mirror of
				https://github.com/crater-invoice/crater.git
				synced 2025-11-03 22:13:18 -05:00 
			
		
		
		
	Compare commits
	
		
			84 Commits
		
	
	
		
			dependabot
			...
			dark-actio
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 150380791f | |||
| 45aaee72d8 | |||
| f4bc78ce98 | |||
| ba769864d4 | |||
| de235a91e5 | |||
| 18dc88682c | |||
| deb039ba02 | |||
| 7e80bfb507 | |||
| 8a5632c7d6 | |||
| 1a4309ca69 | |||
| 3b3da13218 | |||
| 5440b0f354 | |||
| 9249105ad6 | |||
| 238cdb3c25 | |||
| 55bf70c868 | |||
| 0c5adff9b4 | |||
| 3d5732ee26 | |||
| 5aa7decdbe | |||
| 5c67780870 | |||
| ba7f619c67 | |||
| 82efad71c0 | |||
| 826ef72faa | |||
| 2adaa7a84a | |||
| 13557ea075 | |||
| ce88c772d1 | |||
| a32d36363d | |||
| f874b3507d | |||
| c36d25931f | |||
| 1e6c3b287f | |||
| 0462ff60a0 | |||
| 29a135adaf | |||
| 1aceb2c672 | |||
| fb9fab641d | |||
| 8f2edc220b | |||
| d14ab013ad | |||
| f6639f5863 | |||
| c5acf1343e | |||
| 0321ca5515 | |||
| bd345949e5 | |||
| b4b00ebdd6 | |||
| 43316dae28 | |||
| 80e2548b38 | |||
| e8d92cd641 | |||
| 2c8bb38531 | |||
| 04c7ae39a2 | |||
| af2482a69c | |||
| cac35826c2 | |||
| 18b5705372 | |||
| c61fbad5ce | |||
| 15f3f566e3 | |||
| 2e93082282 | |||
| ca55221c4f | |||
| 98196194e2 | |||
| 7447cc24f9 | |||
| 889d22d92c | |||
| bc8f2cd484 | |||
| 4e47f58bad | |||
| d8c429912e | |||
| df04fd9e53 | |||
| 0aaf0e7e75 | |||
| 3d0b89bb4d | |||
| 38c4b9ebce | |||
| 7be59e78e0 | |||
| 189c51cdf4 | |||
| bd5f0fe5cf | |||
| 787619b907 | |||
| fd70ab9a99 | |||
| 33315638df | |||
| bba14bf51a | |||
| ea41989034 | |||
| 3f3f83a00a | |||
| adc5962071 | |||
| c4c00002d7 | |||
| a7c1e12db6 | |||
| 32949d1eec | |||
| 7718f77f3d | |||
| a9e54981bf | |||
| ef35293f8a | |||
| 5c63770b6b | |||
| d130e20c92 | |||
| 586fb1ae10 | |||
| 20c2502e31 | |||
| 0e31f85c18 | |||
| e7301eb7a3 | 
							
								
								
									
										162
									
								
								.github/workflows/uffizzi-build.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										162
									
								
								.github/workflows/uffizzi-build.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@ -0,0 +1,162 @@
 | 
			
		||||
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
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								.github/workflows/uffizzi-preview.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@ -0,0 +1,84 @@
 | 
			
		||||
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:7.4-fpm
 | 
			
		||||
FROM php:8.1-fpm
 | 
			
		||||
 | 
			
		||||
# Arguments defined in docker-compose.yml
 | 
			
		||||
ARG user
 | 
			
		||||
 | 
			
		||||
@ -103,6 +103,7 @@ class CustomerStatsController extends Controller
 | 
			
		||||
        )
 | 
			
		||||
            ->whereCompany()
 | 
			
		||||
            ->whereCustomer($customer->id)
 | 
			
		||||
            ->where('status', '<>', Invoice::STATUS_DRAFT)
 | 
			
		||||
            ->sum('total');
 | 
			
		||||
        $totalReceipts = Payment::whereBetween(
 | 
			
		||||
            'payment_date',
 | 
			
		||||
 | 
			
		||||
@ -104,6 +104,7 @@ class DashboardController extends Controller
 | 
			
		||||
            'invoice_date',
 | 
			
		||||
            [$startDate->format('Y-m-d'), $start->format('Y-m-d')]
 | 
			
		||||
        )
 | 
			
		||||
            ->where('status', '<>', Invoice::STATUS_DRAFT)
 | 
			
		||||
            ->whereCompany()
 | 
			
		||||
            ->sum('base_total');
 | 
			
		||||
 | 
			
		||||
@ -141,6 +142,7 @@ class DashboardController extends Controller
 | 
			
		||||
        $recent_due_invoices = Invoice::with('customer')
 | 
			
		||||
            ->whereCompany()
 | 
			
		||||
            ->where('base_due_amount', '>', 0)
 | 
			
		||||
            ->where('status', '<>', Invoice::STATUS_DRAFT)
 | 
			
		||||
            ->take(5)
 | 
			
		||||
            ->latest()
 | 
			
		||||
            ->get();
 | 
			
		||||
 | 
			
		||||
@ -24,6 +24,7 @@ class InvoicesController extends Controller
 | 
			
		||||
        $limit = $request->has('limit') ? $request->limit : 10;
 | 
			
		||||
 | 
			
		||||
        $invoices = Invoice::whereCompany()
 | 
			
		||||
            ->whereTabFilters($request->tab_status)
 | 
			
		||||
            ->join('customers', 'customers.id', '=', 'invoices.customer_id')
 | 
			
		||||
            ->applyFilters($request->all())
 | 
			
		||||
            ->select('invoices.*', 'customers.name')
 | 
			
		||||
 | 
			
		||||
@ -2,14 +2,15 @@
 | 
			
		||||
 | 
			
		||||
namespace Crater\Http\Controllers\V1\Admin\Report;
 | 
			
		||||
 | 
			
		||||
use PDF;
 | 
			
		||||
use Carbon\Carbon;
 | 
			
		||||
use Crater\Http\Controllers\Controller;
 | 
			
		||||
use Crater\Models\Company;
 | 
			
		||||
use Crater\Models\CompanySetting;
 | 
			
		||||
use Crater\Models\Currency;
 | 
			
		||||
use Crater\Models\Customer;
 | 
			
		||||
use Illuminate\Http\Request;
 | 
			
		||||
use Crater\Models\CompanySetting;
 | 
			
		||||
use Illuminate\Support\Facades\App;
 | 
			
		||||
use PDF;
 | 
			
		||||
use Crater\Http\Controllers\Controller;
 | 
			
		||||
 | 
			
		||||
class CustomerSalesReportController extends Controller
 | 
			
		||||
{
 | 
			
		||||
@ -56,6 +57,7 @@ class CustomerSalesReportController extends Controller
 | 
			
		||||
        $dateFormat = CompanySetting::getSetting('carbon_date_format', $company->id);
 | 
			
		||||
        $from_date = Carbon::createFromFormat('Y-m-d', $request->from_date)->format($dateFormat);
 | 
			
		||||
        $to_date = Carbon::createFromFormat('Y-m-d', $request->to_date)->format($dateFormat);
 | 
			
		||||
        $currency = Currency::findOrFail(CompanySetting::getSetting('currency', $company->id));
 | 
			
		||||
 | 
			
		||||
        $colors = [
 | 
			
		||||
            'primary_text_color',
 | 
			
		||||
@ -80,6 +82,7 @@ class CustomerSalesReportController extends Controller
 | 
			
		||||
            'company' => $company,
 | 
			
		||||
            'from_date' => $from_date,
 | 
			
		||||
            'to_date' => $to_date,
 | 
			
		||||
            'currency' => $currency,
 | 
			
		||||
        ]);
 | 
			
		||||
 | 
			
		||||
        $pdf = PDF::loadView('app.pdf.reports.sales-customers');
 | 
			
		||||
 | 
			
		||||
@ -2,14 +2,15 @@
 | 
			
		||||
 | 
			
		||||
namespace Crater\Http\Controllers\V1\Admin\Report;
 | 
			
		||||
 | 
			
		||||
use Carbon\Carbon;
 | 
			
		||||
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;
 | 
			
		||||
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;
 | 
			
		||||
 | 
			
		||||
class ExpensesReportController extends Controller
 | 
			
		||||
{
 | 
			
		||||
@ -43,6 +44,7 @@ class ExpensesReportController extends Controller
 | 
			
		||||
        $dateFormat = CompanySetting::getSetting('carbon_date_format', $company->id);
 | 
			
		||||
        $from_date = Carbon::createFromFormat('Y-m-d', $request->from_date)->format($dateFormat);
 | 
			
		||||
        $to_date = Carbon::createFromFormat('Y-m-d', $request->to_date)->format($dateFormat);
 | 
			
		||||
        $currency = Currency::findOrFail(CompanySetting::getSetting('currency', $company->id));
 | 
			
		||||
 | 
			
		||||
        $colors = [
 | 
			
		||||
            'primary_text_color',
 | 
			
		||||
@ -66,6 +68,7 @@ class ExpensesReportController extends Controller
 | 
			
		||||
            'company' => $company,
 | 
			
		||||
            'from_date' => $from_date,
 | 
			
		||||
            'to_date' => $to_date,
 | 
			
		||||
            'currency' => $currency,
 | 
			
		||||
        ]);
 | 
			
		||||
        $pdf = PDF::loadView('app.pdf.reports.expenses');
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -2,14 +2,15 @@
 | 
			
		||||
 | 
			
		||||
namespace Crater\Http\Controllers\V1\Admin\Report;
 | 
			
		||||
 | 
			
		||||
use Carbon\Carbon;
 | 
			
		||||
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;
 | 
			
		||||
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;
 | 
			
		||||
 | 
			
		||||
class ItemSalesReportController extends Controller
 | 
			
		||||
{
 | 
			
		||||
@ -43,6 +44,7 @@ class ItemSalesReportController extends Controller
 | 
			
		||||
        $dateFormat = CompanySetting::getSetting('carbon_date_format', $company->id);
 | 
			
		||||
        $from_date = Carbon::createFromFormat('Y-m-d', $request->from_date)->format($dateFormat);
 | 
			
		||||
        $to_date = Carbon::createFromFormat('Y-m-d', $request->to_date)->format($dateFormat);
 | 
			
		||||
        $currency = Currency::findOrFail(CompanySetting::getSetting('currency', $company->id));
 | 
			
		||||
 | 
			
		||||
        $colors = [
 | 
			
		||||
            'primary_text_color',
 | 
			
		||||
@ -66,6 +68,7 @@ class ItemSalesReportController extends Controller
 | 
			
		||||
            'company' => $company,
 | 
			
		||||
            'from_date' => $from_date,
 | 
			
		||||
            'to_date' => $to_date,
 | 
			
		||||
            'currency' => $currency,
 | 
			
		||||
        ]);
 | 
			
		||||
        $pdf = PDF::loadView('app.pdf.reports.sales-items');
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -2,15 +2,16 @@
 | 
			
		||||
 | 
			
		||||
namespace Crater\Http\Controllers\V1\Admin\Report;
 | 
			
		||||
 | 
			
		||||
use PDF;
 | 
			
		||||
use Carbon\Carbon;
 | 
			
		||||
use Crater\Http\Controllers\Controller;
 | 
			
		||||
use Crater\Models\Company;
 | 
			
		||||
use Crater\Models\CompanySetting;
 | 
			
		||||
use Crater\Models\Expense;
 | 
			
		||||
use Crater\Models\Payment;
 | 
			
		||||
use Crater\Models\Currency;
 | 
			
		||||
use Illuminate\Http\Request;
 | 
			
		||||
use Crater\Models\CompanySetting;
 | 
			
		||||
use Illuminate\Support\Facades\App;
 | 
			
		||||
use PDF;
 | 
			
		||||
use Crater\Http\Controllers\Controller;
 | 
			
		||||
 | 
			
		||||
class ProfitLossReportController extends Controller
 | 
			
		||||
{
 | 
			
		||||
@ -49,6 +50,8 @@ class ProfitLossReportController extends Controller
 | 
			
		||||
        $dateFormat = CompanySetting::getSetting('carbon_date_format', $company->id);
 | 
			
		||||
        $from_date = Carbon::createFromFormat('Y-m-d', $request->from_date)->format($dateFormat);
 | 
			
		||||
        $to_date = Carbon::createFromFormat('Y-m-d', $request->to_date)->format($dateFormat);
 | 
			
		||||
        $currency = Currency::findOrFail(CompanySetting::getSetting('currency', $company->id));
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
        $colors = [
 | 
			
		||||
            'primary_text_color',
 | 
			
		||||
@ -74,6 +77,7 @@ class ProfitLossReportController extends Controller
 | 
			
		||||
            'company' => $company,
 | 
			
		||||
            'from_date' => $from_date,
 | 
			
		||||
            'to_date' => $to_date,
 | 
			
		||||
            'currency' => $currency,
 | 
			
		||||
        ]);
 | 
			
		||||
        $pdf = PDF::loadView('app.pdf.reports.profit-loss');
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -2,14 +2,15 @@
 | 
			
		||||
 | 
			
		||||
namespace Crater\Http\Controllers\V1\Admin\Report;
 | 
			
		||||
 | 
			
		||||
use Carbon\Carbon;
 | 
			
		||||
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;
 | 
			
		||||
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;
 | 
			
		||||
 | 
			
		||||
class TaxSummaryReportController extends Controller
 | 
			
		||||
{
 | 
			
		||||
@ -44,6 +45,8 @@ class TaxSummaryReportController extends Controller
 | 
			
		||||
        $dateFormat = CompanySetting::getSetting('carbon_date_format', $company->id);
 | 
			
		||||
        $from_date = Carbon::createFromFormat('Y-m-d', $request->from_date)->format($dateFormat);
 | 
			
		||||
        $to_date = Carbon::createFromFormat('Y-m-d', $request->to_date)->format($dateFormat);
 | 
			
		||||
        $currency = Currency::findOrFail(CompanySetting::getSetting('currency', $company->id));
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
        $colors = [
 | 
			
		||||
            'primary_text_color',
 | 
			
		||||
@ -68,6 +71,7 @@ class TaxSummaryReportController extends Controller
 | 
			
		||||
            'company' => $company,
 | 
			
		||||
            'from_date' => $from_date,
 | 
			
		||||
            'to_date' => $to_date,
 | 
			
		||||
            'currency' => $currency,
 | 
			
		||||
        ]);
 | 
			
		||||
 | 
			
		||||
        $pdf = PDF::loadView('app.pdf.reports.tax-summary');
 | 
			
		||||
 | 
			
		||||
@ -3,7 +3,6 @@
 | 
			
		||||
namespace Crater\Http\Requests;
 | 
			
		||||
 | 
			
		||||
use Illuminate\Foundation\Http\FormRequest;
 | 
			
		||||
use Illuminate\Support\Str;
 | 
			
		||||
use Illuminate\Validation\Rule;
 | 
			
		||||
 | 
			
		||||
class CompaniesRequest extends FormRequest
 | 
			
		||||
@ -34,6 +33,10 @@ class CompaniesRequest extends FormRequest
 | 
			
		||||
            'currency' => [
 | 
			
		||||
                'required'
 | 
			
		||||
            ],
 | 
			
		||||
            'slug' => [
 | 
			
		||||
                'required',
 | 
			
		||||
                Rule::unique('companies')
 | 
			
		||||
            ],
 | 
			
		||||
            'address.name' => [
 | 
			
		||||
                'nullable',
 | 
			
		||||
            ],
 | 
			
		||||
@ -68,11 +71,11 @@ class CompaniesRequest extends FormRequest
 | 
			
		||||
    {
 | 
			
		||||
        return collect($this->validated())
 | 
			
		||||
            ->only([
 | 
			
		||||
                'name'
 | 
			
		||||
                'name',
 | 
			
		||||
                'slug'
 | 
			
		||||
            ])
 | 
			
		||||
            ->merge([
 | 
			
		||||
                'owner_id' => $this->user()->id,
 | 
			
		||||
                'slug' => Str::slug($this->name)
 | 
			
		||||
                'owner_id' => $this->user()->id
 | 
			
		||||
            ])
 | 
			
		||||
            ->toArray();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
@ -30,7 +30,8 @@ class CompanyRequest extends FormRequest
 | 
			
		||||
                Rule::unique('companies')->ignore($this->header('company'), 'id'),
 | 
			
		||||
            ],
 | 
			
		||||
            'slug' => [
 | 
			
		||||
                'nullable'
 | 
			
		||||
                'required',
 | 
			
		||||
                Rule::unique('companies')->ignore($this->header('company'), 'id'),
 | 
			
		||||
            ],
 | 
			
		||||
            'address.country_id' => [
 | 
			
		||||
                'required',
 | 
			
		||||
 | 
			
		||||
@ -23,7 +23,7 @@ class EstimateResource extends JsonResource
 | 
			
		||||
            'reference_number' => $this->reference_number,
 | 
			
		||||
            'tax_per_item' => $this->tax_per_item,
 | 
			
		||||
            'discount_per_item' => $this->discount_per_item,
 | 
			
		||||
            'notes' => $this->getNotes(),
 | 
			
		||||
            'notes' => $this->notes,
 | 
			
		||||
            'discount' => $this->discount,
 | 
			
		||||
            'discount_type' => $this->discount_type,
 | 
			
		||||
            'discount_val' => $this->discount_val,
 | 
			
		||||
 | 
			
		||||
@ -18,7 +18,7 @@ class PaymentResource extends JsonResource
 | 
			
		||||
            'id' => $this->id,
 | 
			
		||||
            'payment_number' => $this->payment_number,
 | 
			
		||||
            'payment_date' => $this->payment_date,
 | 
			
		||||
            'notes' => $this->getNotes(),
 | 
			
		||||
            'notes' => $this->notes,
 | 
			
		||||
            'amount' => $this->amount,
 | 
			
		||||
            'unique_hash' => $this->unique_hash,
 | 
			
		||||
            'invoice_id' => $this->invoice_id,
 | 
			
		||||
 | 
			
		||||
@ -217,7 +217,7 @@ class Company extends Model implements HasMedia
 | 
			
		||||
            'estimate_billing_address_format' => $billingAddressFormat,
 | 
			
		||||
            'payment_company_address_format' => $companyAddressFormat,
 | 
			
		||||
            'payment_from_customer_address_format' => $paymentFromCustomerAddress,
 | 
			
		||||
            'currency' => request()->currency ?? 13,
 | 
			
		||||
            'currency' => request()->currency ?? 1,
 | 
			
		||||
            'time_zone' => 'Asia/Kolkata',
 | 
			
		||||
            'language' => 'en',
 | 
			
		||||
            'fiscal_year' => '1-12',
 | 
			
		||||
 | 
			
		||||
@ -483,7 +483,8 @@ class Estimate extends Model implements HasMedia
 | 
			
		||||
            '{ESTIMATE_DATE}' => $this->formattedEstimateDate,
 | 
			
		||||
            '{ESTIMATE_EXPIRY_DATE}' => $this->formattedExpiryDate,
 | 
			
		||||
            '{ESTIMATE_NUMBER}' => $this->estimate_number,
 | 
			
		||||
            '{ESTIMATE_REF_NUMBER}' => $this->reference_number,
 | 
			
		||||
            '{PDF_LINK}' => $this->estimatePdfUrl,
 | 
			
		||||
            '{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')) {
 | 
			
		||||
            $expense->addMediaFromRequest('attachment_receipt')->toMediaCollection('receipts');
 | 
			
		||||
            $expense->addMediaFromRequest('attachment_receipt')->toMediaCollection('receipts', 'local');
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if ($request->customFields) {
 | 
			
		||||
@ -262,12 +262,12 @@ class Expense extends Model implements HasMedia
 | 
			
		||||
            ExchangeRateLog::addExchangeRateLog($this);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (isset($request->is_attachment_receipt_removed) && (bool) $request->is_attachment_receipt_removed) {
 | 
			
		||||
        if (isset($request->is_attachment_receipt_removed) && $request->is_attachment_receipt_removed == "true") {
 | 
			
		||||
            $this->clearMediaCollection('receipts');
 | 
			
		||||
        }
 | 
			
		||||
        if ($request->hasFile('attachment_receipt')) {
 | 
			
		||||
            $this->clearMediaCollection('receipts');
 | 
			
		||||
            $this->addMediaFromRequest('attachment_receipt')->toMediaCollection('receipts');
 | 
			
		||||
            $this->addMediaFromRequest('attachment_receipt')->toMediaCollection('receipts', 'local');
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if ($request->customFields) {
 | 
			
		||||
 | 
			
		||||
@ -187,16 +187,6 @@ class Invoice extends Model implements HasMedia
 | 
			
		||||
        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)
 | 
			
		||||
    {
 | 
			
		||||
        return $query->whereIn('invoices.paid_status', [
 | 
			
		||||
@ -234,6 +224,40 @@ class Invoice extends Model implements HasMedia
 | 
			
		||||
        $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)
 | 
			
		||||
    {
 | 
			
		||||
        $filters = collect($filters);
 | 
			
		||||
@ -249,17 +273,11 @@ class Invoice extends Model implements HasMedia
 | 
			
		||||
                $filters->get('status') == self::STATUS_PAID
 | 
			
		||||
            ) {
 | 
			
		||||
                $query->wherePaidStatus($filters->get('status'));
 | 
			
		||||
            } elseif ($filters->get('status') == 'DUE') {
 | 
			
		||||
                $query->whereDueStatus($filters->get('status'));
 | 
			
		||||
            } else {
 | 
			
		||||
                $query->whereStatus($filters->get('status'));
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if ($filters->get('paid_status')) {
 | 
			
		||||
            $query->wherePaidStatus($filters->get('status'));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if ($filters->get('invoice_id')) {
 | 
			
		||||
            $query->whereInvoice($filters->get('invoice_id'));
 | 
			
		||||
        }
 | 
			
		||||
@ -651,7 +669,9 @@ class Invoice extends Model implements HasMedia
 | 
			
		||||
            '{INVOICE_DATE}' => $this->formattedInvoiceDate,
 | 
			
		||||
            '{INVOICE_DUE_DATE}' => $this->formattedDueDate,
 | 
			
		||||
            '{INVOICE_NUMBER}' => $this->invoice_number,
 | 
			
		||||
            '{INVOICE_REF_NUMBER}' => $this->reference_number,
 | 
			
		||||
            '{PDF_LINK}' => $this->invoicePdfUrl,
 | 
			
		||||
            '{DUE_AMOUNT}' => format_money_pdf($this->due_amount, $this->customer->currency),
 | 
			
		||||
            '{TOTAL_AMOUNT}' => format_money_pdf($this->total, $this->customer->currency)
 | 
			
		||||
        ];
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -435,7 +435,8 @@ class Payment extends Model implements HasMedia
 | 
			
		||||
            '{PAYMENT_DATE}' => $this->formattedPaymentDate,
 | 
			
		||||
            '{PAYMENT_MODE}' => $this->paymentMethod ? $this->paymentMethod->name : null,
 | 
			
		||||
            '{PAYMENT_NUMBER}' => $this->payment_number,
 | 
			
		||||
            '{PAYMENT_AMOUNT}' => $this->reference_number,
 | 
			
		||||
            '{PDF_LINK}' => $this->paymentPdfUrl,
 | 
			
		||||
            '{PAYMENT_AMOUNT}' => format_money_pdf($this->amount, $this->customer->currency)
 | 
			
		||||
        ];
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -38,7 +38,7 @@
 | 
			
		||||
    "barryvdh/laravel-ide-helper": "^2.6",
 | 
			
		||||
    "beyondcode/laravel-dump-server": "^1.0",
 | 
			
		||||
    "facade/ignition": "^2.3.6",
 | 
			
		||||
    "friendsofphp/php-cs-fixer": "^3.0",
 | 
			
		||||
    "friendsofphp/php-cs-fixer": "^3.8",
 | 
			
		||||
    "fakerphp/faker": "^1.9.1",
 | 
			
		||||
    "mockery/mockery": "^1.3.1",
 | 
			
		||||
    "nunomaduro/collision": "^5.0",
 | 
			
		||||
@ -81,7 +81,10 @@
 | 
			
		||||
  "config": {
 | 
			
		||||
    "optimize-autoloader": true,
 | 
			
		||||
    "preferred-install": "dist",
 | 
			
		||||
    "sort-packages": true
 | 
			
		||||
    "sort-packages": true,
 | 
			
		||||
    "allow-plugins": {
 | 
			
		||||
      "pestphp/pest-plugin": true
 | 
			
		||||
    }
 | 
			
		||||
  },
 | 
			
		||||
  "extra": {
 | 
			
		||||
    "laravel": {
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										2331
									
								
								composer.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										2331
									
								
								composer.lock
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							@ -170,7 +170,7 @@ class CountriesTableSeeder extends Seeder
 | 
			
		||||
        ['id' => 152,'code' => 'NR','name' => "Nauru",'phonecode' => 674],
 | 
			
		||||
        ['id' => 153,'code' => 'NP','name' => "Nepal",'phonecode' => 977],
 | 
			
		||||
        ['id' => 154,'code' => 'AN','name' => "Netherlands Antilles",'phonecode' => 599],
 | 
			
		||||
        ['id' => 155,'code' => 'NL','name' => "Netherlands The",'phonecode' => 31],
 | 
			
		||||
        ['id' => 155,'code' => 'NL','name' => "Netherlands",'phonecode' => 31],
 | 
			
		||||
        ['id' => 156,'code' => 'NC','name' => "New Caledonia",'phonecode' => 687],
 | 
			
		||||
        ['id' => 157,'code' => 'NZ','name' => "New Zealand",'phonecode' => 64],
 | 
			
		||||
        ['id' => 158,'code' => 'NI','name' => "Nicaragua",'phonecode' => 505],
 | 
			
		||||
 | 
			
		||||
@ -27,7 +27,7 @@
 | 
			
		||||
    "vite": "^2.6.1"
 | 
			
		||||
  },
 | 
			
		||||
  "dependencies": {
 | 
			
		||||
    "@headlessui/vue": "^1.4.0",
 | 
			
		||||
    "@headlessui/vue": "^1.5.0",
 | 
			
		||||
    "@heroicons/vue": "^1.0.1",
 | 
			
		||||
    "@popperjs/core": "^2.9.2",
 | 
			
		||||
    "@stripe/stripe-js": "^1.21.2",
 | 
			
		||||
@ -48,7 +48,8 @@
 | 
			
		||||
    "mini-svg-data-uri": "^1.3.3",
 | 
			
		||||
    "moment": "^2.29.1",
 | 
			
		||||
    "pinia": "^2.0.4",
 | 
			
		||||
    "v-money3": "^3.13.5",
 | 
			
		||||
    "v-calendar": "3.0.0-alpha.8",
 | 
			
		||||
    "v-money3": "3.16.1",
 | 
			
		||||
    "v-tooltip": "^4.0.0-alpha.1",
 | 
			
		||||
    "vue": "^3.2.0-beta.5",
 | 
			
		||||
    "vue-flatpickr-component": "^9.0.3",
 | 
			
		||||
 | 
			
		||||
@ -7,6 +7,7 @@
 | 
			
		||||
      py-2
 | 
			
		||||
      rounded-lg
 | 
			
		||||
      bg-opacity-40 bg-gray-300
 | 
			
		||||
      dark:bg-gray-700 dark:border-gray-600
 | 
			
		||||
      whitespace-nowrap
 | 
			
		||||
      flex-col
 | 
			
		||||
      mt-1
 | 
			
		||||
@ -19,6 +20,7 @@
 | 
			
		||||
        text-sm
 | 
			
		||||
        font-medium
 | 
			
		||||
        text-black
 | 
			
		||||
        dark:text-white
 | 
			
		||||
        truncate
 | 
			
		||||
        select-all select-color
 | 
			
		||||
      "
 | 
			
		||||
 | 
			
		||||
@ -43,6 +43,12 @@
 | 
			
		||||
            max-w-full
 | 
			
		||||
            left-0
 | 
			
		||||
            top-3
 | 
			
		||||
            bg-white
 | 
			
		||||
            dark:border
 | 
			
		||||
            dark:border-white/10
 | 
			
		||||
            dark:text-white
 | 
			
		||||
            dark:bg-gray-800
 | 
			
		||||
            dark:shadow-glass
 | 
			
		||||
          "
 | 
			
		||||
        >
 | 
			
		||||
          <div
 | 
			
		||||
@ -53,7 +59,7 @@
 | 
			
		||||
              ring-1 ring-black ring-opacity-5
 | 
			
		||||
            "
 | 
			
		||||
          >
 | 
			
		||||
            <div class="relative grid bg-white">
 | 
			
		||||
            <div class="relative grid bg-white dark:bg-gray-800">
 | 
			
		||||
              <div class="relative p-4">
 | 
			
		||||
                <BaseInput
 | 
			
		||||
                  v-model="textSearch"
 | 
			
		||||
@ -66,7 +72,7 @@
 | 
			
		||||
 | 
			
		||||
              <div
 | 
			
		||||
                v-if="filteredNotes.length > 0"
 | 
			
		||||
                class="relative flex flex-col overflow-auto list max-h-36"
 | 
			
		||||
                class="relative flex flex-col overflow-auto list max-h-36 dark:border-white/10"
 | 
			
		||||
              >
 | 
			
		||||
                <div
 | 
			
		||||
                  v-for="(note, index) in filteredNotes"
 | 
			
		||||
@ -79,6 +85,8 @@
 | 
			
		||||
                    cursor-pointer
 | 
			
		||||
                    hover:bg-gray-100 hover:cursor-pointer
 | 
			
		||||
                    last:border-b-0
 | 
			
		||||
                    dark:border-gray-600
 | 
			
		||||
                    dark:border-white/10 dark:hover:bg-gray-700/30
 | 
			
		||||
                  "
 | 
			
		||||
                  @click="selectNote(index, close)"
 | 
			
		||||
                >
 | 
			
		||||
@ -91,6 +99,7 @@
 | 
			
		||||
                        leading-tight
 | 
			
		||||
                        text-gray-700
 | 
			
		||||
                        cursor-pointer
 | 
			
		||||
                        dark:text-gray-400
 | 
			
		||||
                      "
 | 
			
		||||
                    >
 | 
			
		||||
                      {{ note.name }}
 | 
			
		||||
@ -118,6 +127,10 @@
 | 
			
		||||
                bg-gray-200
 | 
			
		||||
                border-none
 | 
			
		||||
                outline-none
 | 
			
		||||
                dark:bg-gray-600/70
 | 
			
		||||
                dark:backdrop-blur-xl
 | 
			
		||||
                dark:shadow-glass
 | 
			
		||||
                dark:hover:bg-gray-600/80
 | 
			
		||||
              "
 | 
			
		||||
              @click="openNoteModal"
 | 
			
		||||
            >
 | 
			
		||||
 | 
			
		||||
@ -6,8 +6,17 @@
 | 
			
		||||
 | 
			
		||||
<script setup>
 | 
			
		||||
import Chart from 'chart.js'
 | 
			
		||||
import { ref, reactive, computed, onMounted, watchEffect, inject } from 'vue'
 | 
			
		||||
import {
 | 
			
		||||
  ref,
 | 
			
		||||
  reactive,
 | 
			
		||||
  computed,
 | 
			
		||||
  onMounted,
 | 
			
		||||
  watchEffect,
 | 
			
		||||
  inject,
 | 
			
		||||
  watch,
 | 
			
		||||
} from 'vue'
 | 
			
		||||
import { useCompanyStore } from '@/scripts/admin/stores/company'
 | 
			
		||||
import { useGlobalStore } from '@/scripts/admin/stores/global'
 | 
			
		||||
 | 
			
		||||
const utils = inject('utils')
 | 
			
		||||
 | 
			
		||||
@ -44,9 +53,11 @@ const props = defineProps({
 | 
			
		||||
  },
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
const isDarkModeOn = document.documentElement.classList.contains('dark')
 | 
			
		||||
let myLineChart = null
 | 
			
		||||
const graph = ref(null)
 | 
			
		||||
const companyStore = useCompanyStore()
 | 
			
		||||
const globalStore = useGlobalStore()
 | 
			
		||||
const defaultCurrency = computed(() => {
 | 
			
		||||
  return companyStore.selectedCompanyCurrency
 | 
			
		||||
})
 | 
			
		||||
@ -60,6 +71,14 @@ watchEffect(() => {
 | 
			
		||||
  }
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
watch(
 | 
			
		||||
  () => globalStore.isDarkModeOn,
 | 
			
		||||
  () => {
 | 
			
		||||
    myLineChart.reset()
 | 
			
		||||
    updateColors()
 | 
			
		||||
  }
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
onMounted(() => {
 | 
			
		||||
  let context = graph.value.getContext('2d')
 | 
			
		||||
  let options = reactive({
 | 
			
		||||
@ -81,6 +100,8 @@ onMounted(() => {
 | 
			
		||||
    },
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
  const salesColor = globalStore.isDarkModeOn ? '#ffffff' : '#040405'
 | 
			
		||||
 | 
			
		||||
  let data = reactive({
 | 
			
		||||
    labels: props.labels,
 | 
			
		||||
    datasets: [
 | 
			
		||||
@ -89,16 +110,16 @@ onMounted(() => {
 | 
			
		||||
        fill: false,
 | 
			
		||||
        lineTension: 0.3,
 | 
			
		||||
        backgroundColor: 'rgba(230, 254, 249)',
 | 
			
		||||
        borderColor: '#040405',
 | 
			
		||||
        borderColor: salesColor,
 | 
			
		||||
        borderCapStyle: 'butt',
 | 
			
		||||
        borderDash: [],
 | 
			
		||||
        borderDashOffset: 0.0,
 | 
			
		||||
        borderJoinStyle: 'miter',
 | 
			
		||||
        pointBorderColor: '#040405',
 | 
			
		||||
        pointBorderColor: salesColor,
 | 
			
		||||
        pointBackgroundColor: '#fff',
 | 
			
		||||
        pointBorderWidth: 1,
 | 
			
		||||
        pointHoverRadius: 5,
 | 
			
		||||
        pointHoverBackgroundColor: '#040405',
 | 
			
		||||
        pointHoverBackgroundColor: salesColor,
 | 
			
		||||
        pointHoverBorderColor: 'rgba(220,220,220,1)',
 | 
			
		||||
        pointHoverBorderWidth: 2,
 | 
			
		||||
        pointRadius: 4,
 | 
			
		||||
@ -194,4 +215,12 @@ function update() {
 | 
			
		||||
    lazy: true,
 | 
			
		||||
  })
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function updateColors() {
 | 
			
		||||
  const newColor = globalStore.isDarkModeOn ? '#ffffff' : '#040405'
 | 
			
		||||
 | 
			
		||||
  myLineChart.data.datasets[0].borderColor = newColor
 | 
			
		||||
  myLineChart.data.datasets[0].pointBorderColor = newColor
 | 
			
		||||
  myLineChart.data.datasets[0].pointHoverBackgroundColor = newColor
 | 
			
		||||
}
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
@ -50,21 +50,11 @@
 | 
			
		||||
          </BaseInputGroup>
 | 
			
		||||
        </template>
 | 
			
		||||
      </ValidateEach>
 | 
			
		||||
      <div
 | 
			
		||||
        slot="footer"
 | 
			
		||||
        class="
 | 
			
		||||
          z-0
 | 
			
		||||
          flex
 | 
			
		||||
          justify-end
 | 
			
		||||
          mt-4
 | 
			
		||||
          pt-4
 | 
			
		||||
          border-t border-gray-200 border-solid border-modal-bg
 | 
			
		||||
        "
 | 
			
		||||
      >
 | 
			
		||||
      <BaseModalFooter>
 | 
			
		||||
        <BaseButton :loading="isSaving" variant="primary" type="submit">
 | 
			
		||||
          {{ $t('general.save') }}
 | 
			
		||||
        </BaseButton>
 | 
			
		||||
      </div>
 | 
			
		||||
      </BaseModalFooter>
 | 
			
		||||
    </form>
 | 
			
		||||
  </BaseCard>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
@ -64,7 +64,7 @@ function mergeExistingValues() {
 | 
			
		||||
  if (props.isEdit) {
 | 
			
		||||
    props.store[props.storeProp].fields.forEach((field) => {
 | 
			
		||||
      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) {
 | 
			
		||||
 | 
			
		||||
@ -9,7 +9,7 @@ import { computed } from 'vue'
 | 
			
		||||
const props = defineProps({
 | 
			
		||||
  modelValue: {
 | 
			
		||||
    type: String,
 | 
			
		||||
    default: moment().format('YYYY-MM-DD hh:MM'),
 | 
			
		||||
    default: moment().format('YYYY-MM-DD HH:mm'),
 | 
			
		||||
  },
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -7,11 +7,12 @@
 | 
			
		||||
    <!-- edit customField  -->
 | 
			
		||||
    <BaseDropdownItem
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.EDIT_CUSTOM_FIELDS)"
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="editCustomField(row.id)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="PencilIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.edit') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
@ -19,11 +20,12 @@
 | 
			
		||||
    <!-- delete customField  -->
 | 
			
		||||
    <BaseDropdownItem
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.DELETE_CUSTOM_FIELDS)"
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="removeCustomField(row.id)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="TrashIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.delete') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
 | 
			
		||||
@ -12,10 +12,10 @@
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.EDIT_CUSTOMER)"
 | 
			
		||||
      :to="`/admin/customers/${row.id}/edit`"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseDropdownItem>
 | 
			
		||||
      <BaseDropdownItem v-slot="slotProps">
 | 
			
		||||
        <BaseIcon
 | 
			
		||||
          name="PencilIcon"
 | 
			
		||||
          class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
          :class="slotProps.class"
 | 
			
		||||
        />
 | 
			
		||||
        {{ $t('general.edit') }}
 | 
			
		||||
      </BaseDropdownItem>
 | 
			
		||||
@ -29,10 +29,10 @@
 | 
			
		||||
      "
 | 
			
		||||
      :to="`customers/${row.id}/view`"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseDropdownItem>
 | 
			
		||||
      <BaseDropdownItem v-slot="slotProps">
 | 
			
		||||
        <BaseIcon
 | 
			
		||||
          name="EyeIcon"
 | 
			
		||||
          class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
          :class="slotProps.class"
 | 
			
		||||
        />
 | 
			
		||||
        {{ $t('general.view') }}
 | 
			
		||||
      </BaseDropdownItem>
 | 
			
		||||
@ -41,11 +41,12 @@
 | 
			
		||||
    <!-- Delete Customer  -->
 | 
			
		||||
    <BaseDropdownItem
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.DELETE_CUSTOMER)"
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="removeCustomer(row.id)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="TrashIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.delete') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
 | 
			
		||||
@ -10,11 +10,12 @@
 | 
			
		||||
    <!-- Copy PDF url  -->
 | 
			
		||||
    <BaseDropdownItem
 | 
			
		||||
      v-if="route.name === 'estimates.view'"
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="copyPdfUrl"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="LinkIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.copy_pdf_url') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
@ -24,10 +25,10 @@
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.EDIT_ESTIMATE)"
 | 
			
		||||
      :to="`/admin/estimates/${row.id}/edit`"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseDropdownItem>
 | 
			
		||||
      <BaseDropdownItem v-slot="slotProps">
 | 
			
		||||
        <BaseIcon
 | 
			
		||||
          name="PencilIcon"
 | 
			
		||||
          class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
          :class="slotProps.class"
 | 
			
		||||
        />
 | 
			
		||||
        {{ $t('general.edit') }}
 | 
			
		||||
      </BaseDropdownItem>
 | 
			
		||||
@ -36,11 +37,12 @@
 | 
			
		||||
    <!-- Delete Estimate  -->
 | 
			
		||||
    <BaseDropdownItem
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.DELETE_ESTIMATE)"
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="removeEstimate(row.id)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="TrashIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.delete') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
@ -53,10 +55,10 @@
 | 
			
		||||
      "
 | 
			
		||||
      :to="`estimates/${row.id}/view`"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseDropdownItem>
 | 
			
		||||
      <BaseDropdownItem v-slot="slotProps">
 | 
			
		||||
        <BaseIcon
 | 
			
		||||
          name="EyeIcon"
 | 
			
		||||
          class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
          :class="slotProps.class"
 | 
			
		||||
        />
 | 
			
		||||
        {{ $t('general.view') }}
 | 
			
		||||
      </BaseDropdownItem>
 | 
			
		||||
@ -65,11 +67,12 @@
 | 
			
		||||
    <!-- Convert into Invoice  -->
 | 
			
		||||
    <BaseDropdownItem
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.CREATE_INVOICE)"
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="convertInToinvoice(row.id)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="DocumentTextIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('estimates.convert_to_invoice') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
@ -81,11 +84,12 @@
 | 
			
		||||
        route.name !== 'estimates.view' &&
 | 
			
		||||
        userStore.hasAbilities(abilities.SEND_ESTIMATE)
 | 
			
		||||
      "
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="onMarkAsSent(row.id)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="CheckCircleIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('estimates.mark_as_sent') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
@ -97,20 +101,21 @@
 | 
			
		||||
        route.name !== 'estimates.view' &&
 | 
			
		||||
        userStore.hasAbilities(abilities.SEND_ESTIMATE)
 | 
			
		||||
      "
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="sendEstimate(row)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="PaperAirplaneIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('estimates.send_estimate') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
 | 
			
		||||
    <!-- Resend Estimate -->
 | 
			
		||||
    <BaseDropdownItem v-if="canResendEstimate(row)" @click="sendEstimate(row)">
 | 
			
		||||
    <BaseDropdownItem v-if="canResendEstimate(row)" v-slot="slotProps"  @click="sendEstimate(row)">
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="PaperAirplaneIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('estimates.resend_estimate') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
@ -121,11 +126,12 @@
 | 
			
		||||
        row.status !== 'ACCEPTED' &&
 | 
			
		||||
        userStore.hasAbilities(abilities.EDIT_ESTIMATE)
 | 
			
		||||
      "
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="onMarkAsAccepted(row.id)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="CheckCircleIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('estimates.mark_as_accepted') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
@ -136,11 +142,12 @@
 | 
			
		||||
        row.status !== 'REJECTED' &&
 | 
			
		||||
        userStore.hasAbilities(abilities.EDIT_ESTIMATE)
 | 
			
		||||
      "
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="onMarkAsRejected(row.id)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="XCircleIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('estimates.mark_as_rejected') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
 | 
			
		||||
@ -13,11 +13,12 @@
 | 
			
		||||
    <!-- edit expenseCategory  -->
 | 
			
		||||
    <BaseDropdownItem
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.EDIT_EXPENSE)"
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="editExpenseCategory(row.id)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="PencilIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.edit') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
@ -25,11 +26,12 @@
 | 
			
		||||
    <!-- delete expenseCategory  -->
 | 
			
		||||
    <BaseDropdownItem
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.DELETE_EXPENSE)"
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="removeExpenseCategory(row.id)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="TrashIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.delete') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
 | 
			
		||||
@ -12,10 +12,10 @@
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.EDIT_EXPENSE)"
 | 
			
		||||
      :to="`/admin/expenses/${row.id}/edit`"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseDropdownItem>
 | 
			
		||||
      <BaseDropdownItem v-slot="slotProps">
 | 
			
		||||
        <BaseIcon
 | 
			
		||||
          name="PencilIcon"
 | 
			
		||||
          class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
          :class="slotProps.class"
 | 
			
		||||
        />
 | 
			
		||||
        {{ $t('general.edit') }}
 | 
			
		||||
      </BaseDropdownItem>
 | 
			
		||||
@ -24,11 +24,12 @@
 | 
			
		||||
    <!-- delete expense  -->
 | 
			
		||||
    <BaseDropdownItem
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.DELETE_EXPENSE)"
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="removeExpense(row.id)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="TrashIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.delete') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
 | 
			
		||||
@ -12,20 +12,20 @@
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.EDIT_INVOICE)"
 | 
			
		||||
      :to="`/admin/invoices/${row.id}/edit`"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseDropdownItem v-show="row.allow_edit">
 | 
			
		||||
      <BaseDropdownItem v-show="row.allow_edit" v-slot="slotProps">
 | 
			
		||||
        <BaseIcon
 | 
			
		||||
          name="PencilIcon"
 | 
			
		||||
          class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
          :class="slotProps.class"
 | 
			
		||||
        />
 | 
			
		||||
        {{ $t('general.edit') }}
 | 
			
		||||
      </BaseDropdownItem>
 | 
			
		||||
    </router-link>
 | 
			
		||||
 | 
			
		||||
    <!-- Copy PDF url  -->
 | 
			
		||||
    <BaseDropdownItem v-if="route.name === 'invoices.view'" @click="copyPdfUrl">
 | 
			
		||||
    <BaseDropdownItem v-if="route.name === 'invoices.view'" v-slot="slotProps" @click="copyPdfUrl">
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="LinkIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.copy_pdf_url') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
@ -38,29 +38,29 @@
 | 
			
		||||
      "
 | 
			
		||||
      :to="`/admin/invoices/${row.id}/view`"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseDropdownItem>
 | 
			
		||||
      <BaseDropdownItem v-slot="slotProps">
 | 
			
		||||
        <BaseIcon
 | 
			
		||||
          name="EyeIcon"
 | 
			
		||||
          class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
          :class="slotProps.class"
 | 
			
		||||
        />
 | 
			
		||||
        {{ $t('general.view') }}
 | 
			
		||||
      </BaseDropdownItem>
 | 
			
		||||
    </router-link>
 | 
			
		||||
 | 
			
		||||
    <!-- Send Invoice Mail  -->
 | 
			
		||||
    <BaseDropdownItem v-if="canSendInvoice(row)" @click="sendInvoice(row)">
 | 
			
		||||
    <BaseDropdownItem v-if="canSendInvoice(row)" v-slot="slotProps" @click="sendInvoice(row)">
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="PaperAirplaneIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('invoices.send_invoice') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
 | 
			
		||||
    <!-- Resend Invoice -->
 | 
			
		||||
    <BaseDropdownItem v-if="canReSendInvoice(row)" @click="sendInvoice(row)">
 | 
			
		||||
    <BaseDropdownItem v-if="canReSendInvoice(row)" v-slot="slotProps" @click="sendInvoice(row)">
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="PaperAirplaneIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('invoices.resend_invoice') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
@ -69,20 +69,21 @@
 | 
			
		||||
    <router-link :to="`/admin/payments/${row.id}/create`">
 | 
			
		||||
      <BaseDropdownItem
 | 
			
		||||
        v-if="row.status == 'SENT' && route.name !== 'invoices.view'"
 | 
			
		||||
        v-slot="slotProps"
 | 
			
		||||
      >
 | 
			
		||||
        <BaseIcon
 | 
			
		||||
          name="CreditCardIcon"
 | 
			
		||||
          class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
          :class="slotProps.class"
 | 
			
		||||
        />
 | 
			
		||||
        {{ $t('invoices.record_payment') }}
 | 
			
		||||
      </BaseDropdownItem>
 | 
			
		||||
    </router-link>
 | 
			
		||||
 | 
			
		||||
    <!-- Mark as sent Invoice -->
 | 
			
		||||
    <BaseDropdownItem v-if="canSendInvoice(row)" @click="onMarkAsSent(row.id)">
 | 
			
		||||
    <BaseDropdownItem v-if="canSendInvoice(row)" v-slot="slotProps" @click="onMarkAsSent(row.id)">
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="CheckCircleIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('invoices.mark_as_sent') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
@ -90,11 +91,12 @@
 | 
			
		||||
    <!-- Clone Invoice into new invoice  -->
 | 
			
		||||
    <BaseDropdownItem
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.CREATE_INVOICE)"
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="cloneInvoiceData(row)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="DocumentTextIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('invoices.clone_invoice') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
@ -102,11 +104,12 @@
 | 
			
		||||
    <!--  Delete Invoice  -->
 | 
			
		||||
    <BaseDropdownItem
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.DELETE_INVOICE)"
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="removeInvoice(row.id)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="TrashIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.delete') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
 | 
			
		||||
@ -12,11 +12,8 @@
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.EDIT_ITEM)"
 | 
			
		||||
      :to="`/admin/items/${row.id}/edit`"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseDropdownItem>
 | 
			
		||||
        <BaseIcon
 | 
			
		||||
          name="PencilIcon"
 | 
			
		||||
          class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        />
 | 
			
		||||
      <BaseDropdownItem v-slot="slotProps">
 | 
			
		||||
        <BaseIcon name="PencilIcon" :class="slotProps.class" />
 | 
			
		||||
        {{ $t('general.edit') }}
 | 
			
		||||
      </BaseDropdownItem>
 | 
			
		||||
    </router-link>
 | 
			
		||||
@ -24,12 +21,10 @@
 | 
			
		||||
    <!-- delete item  -->
 | 
			
		||||
    <BaseDropdownItem
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.DELETE_ITEM)"
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="removeItem(row.id)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="TrashIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
      />
 | 
			
		||||
      <BaseIcon name="TrashIcon" :class="slotProps.class" />
 | 
			
		||||
      {{ $t('general.delete') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
  </BaseDropdown>
 | 
			
		||||
 | 
			
		||||
@ -10,11 +10,12 @@
 | 
			
		||||
    <!-- edit note  -->
 | 
			
		||||
    <BaseDropdownItem
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.MANAGE_NOTE)"
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="editNote(row.id)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="PencilIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.edit') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
@ -22,11 +23,12 @@
 | 
			
		||||
    <!-- delete note  -->
 | 
			
		||||
    <BaseDropdownItem
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.MANAGE_NOTE)"
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="removeNote(row.id)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="TrashIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.delete') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
 | 
			
		||||
@ -8,30 +8,31 @@
 | 
			
		||||
    </template>
 | 
			
		||||
 | 
			
		||||
    <!-- Copy pdf url  -->
 | 
			
		||||
    <BaseDropdown-item
 | 
			
		||||
    <BaseDropdownItem
 | 
			
		||||
      v-if="
 | 
			
		||||
        route.name === 'payments.view' &&
 | 
			
		||||
        userStore.hasAbilities(abilities.VIEW_PAYMENT)
 | 
			
		||||
      "
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      class="rounded-md"
 | 
			
		||||
      @click="copyPdfUrl"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="LinkIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.copy_pdf_url') }}
 | 
			
		||||
    </BaseDropdown-item>
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
 | 
			
		||||
    <!-- edit payment  -->
 | 
			
		||||
    <router-link
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.EDIT_PAYMENT)"
 | 
			
		||||
      :to="`/admin/payments/${row.id}/edit`"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseDropdownItem>
 | 
			
		||||
      <BaseDropdownItem v-slot="slotProps">
 | 
			
		||||
        <BaseIcon
 | 
			
		||||
          name="PencilIcon"
 | 
			
		||||
          class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
          :class="slotProps.class"
 | 
			
		||||
        />
 | 
			
		||||
        {{ $t('general.edit') }}
 | 
			
		||||
      </BaseDropdownItem>
 | 
			
		||||
@ -45,10 +46,10 @@
 | 
			
		||||
      "
 | 
			
		||||
      :to="`/admin/payments/${row.id}/view`"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseDropdownItem>
 | 
			
		||||
      <BaseDropdownItem v-slot="slotProps">
 | 
			
		||||
        <BaseIcon
 | 
			
		||||
          name="EyeIcon"
 | 
			
		||||
          class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
          :class="slotProps.class"
 | 
			
		||||
        />
 | 
			
		||||
        {{ $t('general.view') }}
 | 
			
		||||
      </BaseDropdownItem>
 | 
			
		||||
@ -61,11 +62,12 @@
 | 
			
		||||
        route.name !== 'payments.view' &&
 | 
			
		||||
        userStore.hasAbilities(abilities.SEND_PAYMENT)
 | 
			
		||||
      "
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="sendPayment(row)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="PaperAirplaneIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('payments.send_payment') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
@ -73,11 +75,12 @@
 | 
			
		||||
    <!-- delete payment  -->
 | 
			
		||||
    <BaseDropdownItem
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.DELETE_PAYMENT)"
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="removePayment(row.id)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="TrashIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.delete') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
 | 
			
		||||
@ -8,19 +8,19 @@
 | 
			
		||||
    </template>
 | 
			
		||||
 | 
			
		||||
    <!-- edit paymentMode  -->
 | 
			
		||||
    <BaseDropdownItem @click="editPaymentMode(row.id)">
 | 
			
		||||
    <BaseDropdownItem v-slot="slotProps"  @click="editPaymentMode(row.id)">
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="PencilIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.edit') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
 | 
			
		||||
    <!-- delete paymentMode  -->
 | 
			
		||||
    <BaseDropdownItem @click="removePaymentMode(row.id)">
 | 
			
		||||
    <BaseDropdownItem v-slot="slotProps"  @click="removePaymentMode(row.id)">
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="TrashIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.delete') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
 | 
			
		||||
@ -15,10 +15,10 @@
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.EDIT_RECURRING_INVOICE)"
 | 
			
		||||
      :to="`/admin/recurring-invoices/${row.id}/edit`"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseDropdownItem>
 | 
			
		||||
      <BaseDropdownItem v-slot="slotProps">
 | 
			
		||||
        <BaseIcon
 | 
			
		||||
          name="PencilIcon"
 | 
			
		||||
          class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
          :class="slotProps.class"
 | 
			
		||||
        />
 | 
			
		||||
        {{ $t('general.edit') }}
 | 
			
		||||
      </BaseDropdownItem>
 | 
			
		||||
@ -32,10 +32,10 @@
 | 
			
		||||
      "
 | 
			
		||||
      :to="`recurring-invoices/${row.id}/view`"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseDropdownItem>
 | 
			
		||||
      <BaseDropdownItem v-slot="slotProps">
 | 
			
		||||
        <BaseIcon
 | 
			
		||||
          name="EyeIcon"
 | 
			
		||||
          class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
          :class="slotProps.class"
 | 
			
		||||
        />
 | 
			
		||||
        {{ $t('general.view') }}
 | 
			
		||||
      </BaseDropdownItem>
 | 
			
		||||
@ -44,11 +44,12 @@
 | 
			
		||||
    <!-- Delete Recurring Invoice  -->
 | 
			
		||||
    <BaseDropdownItem
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.DELETE_RECURRING_INVOICE)"
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="removeMultipleRecurringInvoices(row.id)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="TrashIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.delete') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
 | 
			
		||||
@ -10,11 +10,12 @@
 | 
			
		||||
    <!-- edit role  -->
 | 
			
		||||
    <BaseDropdownItem
 | 
			
		||||
      v-if="userStore.currentUser.is_owner"
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="editRole(row.id)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="PencilIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.edit') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
@ -22,11 +23,12 @@
 | 
			
		||||
    <!-- delete role  -->
 | 
			
		||||
    <BaseDropdownItem
 | 
			
		||||
      v-if="userStore.currentUser.is_owner"
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="removeRole(row.id)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="TrashIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.delete') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
 | 
			
		||||
@ -10,11 +10,12 @@
 | 
			
		||||
    <!-- edit tax-type  -->
 | 
			
		||||
    <BaseDropdownItem
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.EDIT_TAX_TYPE)"
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="editTaxType(row.id)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="PencilIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.edit') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
@ -22,11 +23,12 @@
 | 
			
		||||
    <!-- delete tax-type  -->
 | 
			
		||||
    <BaseDropdownItem
 | 
			
		||||
      v-if="userStore.hasAbilities(abilities.DELETE_TAX_TYPE)"
 | 
			
		||||
      v-slot="slotProps"
 | 
			
		||||
      @click="removeTaxType(row.id)"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="TrashIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.delete') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
 | 
			
		||||
@ -9,20 +9,20 @@
 | 
			
		||||
 | 
			
		||||
    <!-- edit user  -->
 | 
			
		||||
    <router-link :to="`/admin/users/${row.id}/edit`">
 | 
			
		||||
      <BaseDropdownItem>
 | 
			
		||||
      <BaseDropdownItem v-slot="slotProps">
 | 
			
		||||
        <BaseIcon
 | 
			
		||||
          name="PencilIcon"
 | 
			
		||||
          class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
          :class="slotProps.class"
 | 
			
		||||
        />
 | 
			
		||||
        {{ $t('general.edit') }}
 | 
			
		||||
      </BaseDropdownItem>
 | 
			
		||||
    </router-link>
 | 
			
		||||
 | 
			
		||||
    <!-- delete user  -->
 | 
			
		||||
    <BaseDropdownItem @click="removeUser(row.id)">
 | 
			
		||||
    <BaseDropdownItem v-slot="slotProps" @click="removeUser(row.id)">
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="TrashIcon"
 | 
			
		||||
        class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
        :class="slotProps.class"
 | 
			
		||||
      />
 | 
			
		||||
      {{ $t('general.delete') }}
 | 
			
		||||
    </BaseDropdownItem>
 | 
			
		||||
 | 
			
		||||
@ -1,5 +1,13 @@
 | 
			
		||||
<template>
 | 
			
		||||
  <tr class="box-border bg-white border border-gray-200 border-solid rounded-b">
 | 
			
		||||
  <tr
 | 
			
		||||
    class="
 | 
			
		||||
      box-border
 | 
			
		||||
      bg-white
 | 
			
		||||
      border border-gray-200 border-solid
 | 
			
		||||
      rounded-b
 | 
			
		||||
      dark:shadow-glass dark:border dark:border-white/10 dark:bg-gray-800/70
 | 
			
		||||
      "
 | 
			
		||||
    >
 | 
			
		||||
    <td colspan="5" class="p-0 text-left align-top">
 | 
			
		||||
      <table class="w-full">
 | 
			
		||||
        <colgroup>
 | 
			
		||||
@ -17,18 +25,7 @@
 | 
			
		||||
            <td class="px-5 py-4 text-left align-top">
 | 
			
		||||
              <div class="flex justify-start">
 | 
			
		||||
                <div
 | 
			
		||||
                  class="
 | 
			
		||||
                    flex
 | 
			
		||||
                    items-center
 | 
			
		||||
                    justify-center
 | 
			
		||||
                    w-5
 | 
			
		||||
                    h-5
 | 
			
		||||
                    mt-2
 | 
			
		||||
                    text-gray-300
 | 
			
		||||
                    cursor-move
 | 
			
		||||
                    handle
 | 
			
		||||
                    mr-2
 | 
			
		||||
                  "
 | 
			
		||||
                  class="flex items-center justify-center w-5 h-5 mt-2 mr-2 text-gray-300 cursor-move  handle"
 | 
			
		||||
                >
 | 
			
		||||
                  <DragIcon />
 | 
			
		||||
                </div>
 | 
			
		||||
@ -108,7 +105,7 @@
 | 
			
		||||
 | 
			
		||||
                          <BaseIcon
 | 
			
		||||
                            name="ChevronDownIcon"
 | 
			
		||||
                            class="w-4 h-4 text-gray-500 ml-1"
 | 
			
		||||
                            class="w-4 h-4 ml-1 text-gray-500"
 | 
			
		||||
                          />
 | 
			
		||||
                        </span>
 | 
			
		||||
                      </BaseButton>
 | 
			
		||||
@ -141,7 +138,7 @@
 | 
			
		||||
                <div class="flex items-center justify-center w-6 h-10 mx-2">
 | 
			
		||||
                  <BaseIcon
 | 
			
		||||
                    v-if="showRemoveButton"
 | 
			
		||||
                    class="h-5 text-gray-700 cursor-pointer"
 | 
			
		||||
                    class="h-5 dark:text-red-400 cursor-pointer"
 | 
			
		||||
                    name="TrashIcon"
 | 
			
		||||
                    @click="store.removeItem(index)"
 | 
			
		||||
                  />
 | 
			
		||||
@ -155,7 +152,7 @@
 | 
			
		||||
              <BaseContentPlaceholders v-if="loading">
 | 
			
		||||
                <BaseContentPlaceholdersText
 | 
			
		||||
                  :lines="1"
 | 
			
		||||
                  class="w-24 h-8 rounded-md border"
 | 
			
		||||
                  class="w-24 h-8 border rounded-md"
 | 
			
		||||
                />
 | 
			
		||||
              </BaseContentPlaceholders>
 | 
			
		||||
 | 
			
		||||
@ -175,6 +172,7 @@
 | 
			
		||||
                :ability="abilities.CREATE_INVOICE"
 | 
			
		||||
                :store="store"
 | 
			
		||||
                :store-prop="storeProp"
 | 
			
		||||
                :discount="discount"
 | 
			
		||||
                @update="updateTax"
 | 
			
		||||
              />
 | 
			
		||||
            </td>
 | 
			
		||||
 | 
			
		||||
@ -36,18 +36,22 @@
 | 
			
		||||
              justify-center
 | 
			
		||||
              w-full
 | 
			
		||||
              px-2
 | 
			
		||||
              cursor-pointer
 | 
			
		||||
              py-2
 | 
			
		||||
              bg-gray-200
 | 
			
		||||
              border-none
 | 
			
		||||
              outline-none
 | 
			
		||||
              cursor-pointer
 | 
			
		||||
              dark:bg-gray-600/70
 | 
			
		||||
              dark:backdrop-blur-xl
 | 
			
		||||
              dark:shadow-glass
 | 
			
		||||
              dark:hover:bg-gray-600/80
 | 
			
		||||
            "
 | 
			
		||||
            @click="openTaxModal"
 | 
			
		||||
          >
 | 
			
		||||
            <BaseIcon name="CheckCircleIcon" class="h-5 text-primary-400" />
 | 
			
		||||
 | 
			
		||||
            <label
 | 
			
		||||
              class="ml-2 text-sm leading-none text-primary-400 cursor-pointer"
 | 
			
		||||
              class="ml-2 text-sm leading-none cursor-pointer text-primary-400"
 | 
			
		||||
              >{{ $t('invoices.add_new_tax') }}</label
 | 
			
		||||
            >
 | 
			
		||||
          </button>
 | 
			
		||||
@ -64,7 +68,7 @@
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        v-if="taxes.length && index !== taxes.length - 1"
 | 
			
		||||
        name="TrashIcon"
 | 
			
		||||
        class="h-5 text-gray-700 cursor-pointer"
 | 
			
		||||
        class="h-5 text-gray-700 dark:text-red-400 cursor-pointer"
 | 
			
		||||
        @click="removeTax(index)"
 | 
			
		||||
      />
 | 
			
		||||
    </div>
 | 
			
		||||
@ -115,6 +119,10 @@ const props = defineProps({
 | 
			
		||||
    type: Number,
 | 
			
		||||
    default: 0,
 | 
			
		||||
  },
 | 
			
		||||
  discountedTotal: {
 | 
			
		||||
    type: Number,
 | 
			
		||||
    default: 0,
 | 
			
		||||
  },
 | 
			
		||||
  currency: {
 | 
			
		||||
    type: [Object, String],
 | 
			
		||||
    required: true,
 | 
			
		||||
@ -153,19 +161,19 @@ const filteredTypes = computed(() => {
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
const taxAmount = computed(() => {
 | 
			
		||||
  if (localTax.compound_tax && props.total) {
 | 
			
		||||
    return ((props.total + props.totalTax) * localTax.percent) / 100
 | 
			
		||||
  if (localTax.compound_tax && props.discountedTotal) {
 | 
			
		||||
    return ((props.discountedTotal + props.totalTax) * localTax.percent) / 100
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (props.total && localTax.percent) {
 | 
			
		||||
    return (props.total * localTax.percent) / 100
 | 
			
		||||
  if (props.discountedTotal && localTax.percent) {
 | 
			
		||||
    return (props.discountedTotal * localTax.percent) / 100
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return 0
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
watch(
 | 
			
		||||
  () => props.total,
 | 
			
		||||
  () => props.discountedTotal,
 | 
			
		||||
  () => {
 | 
			
		||||
    updateRowTax()
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
@ -1,4 +1,6 @@
 | 
			
		||||
<template>
 | 
			
		||||
  <div class="relative" >
 | 
			
		||||
    <BaseDarkHighlight class="z-[-1]" />
 | 
			
		||||
    <table class="text-center item-table min-w-full">
 | 
			
		||||
      <colgroup>
 | 
			
		||||
        <col style="width: 40%; min-width: 280px" />
 | 
			
		||||
@ -10,7 +12,15 @@
 | 
			
		||||
        />
 | 
			
		||||
        <col style="width: 15%; min-width: 120px" />
 | 
			
		||||
      </colgroup>
 | 
			
		||||
    <thead class="bg-white border border-gray-200 border-solid">
 | 
			
		||||
      <thead
 | 
			
		||||
        class="
 | 
			
		||||
          bg-white
 | 
			
		||||
          border
 | 
			
		||||
          border-gray-200
 | 
			
		||||
          border-solid
 | 
			
		||||
          dark:shadow-glass dark:border dark:border-white/10 dark:bg-gray-800/70
 | 
			
		||||
        "
 | 
			
		||||
      >
 | 
			
		||||
        <tr>
 | 
			
		||||
          <th
 | 
			
		||||
            class="
 | 
			
		||||
@ -22,6 +32,7 @@
 | 
			
		||||
              leading-5
 | 
			
		||||
              text-left text-gray-700
 | 
			
		||||
              border-t border-b border-gray-200 border-solid
 | 
			
		||||
              dark:text-white dark:border-white/10
 | 
			
		||||
            "
 | 
			
		||||
          >
 | 
			
		||||
            <BaseContentPlaceholders v-if="isLoading">
 | 
			
		||||
@ -41,6 +52,7 @@
 | 
			
		||||
              leading-5
 | 
			
		||||
              text-right text-gray-700
 | 
			
		||||
              border-t border-b border-gray-200 border-solid
 | 
			
		||||
              dark:text-white dark:border-white/10
 | 
			
		||||
            "
 | 
			
		||||
          >
 | 
			
		||||
            <BaseContentPlaceholders v-if="isLoading">
 | 
			
		||||
@ -60,6 +72,7 @@
 | 
			
		||||
              leading-5
 | 
			
		||||
              text-left text-gray-700
 | 
			
		||||
              border-t border-b border-gray-200 border-solid
 | 
			
		||||
              dark:text-white dark:border-white/10
 | 
			
		||||
            "
 | 
			
		||||
          >
 | 
			
		||||
            <BaseContentPlaceholders v-if="isLoading">
 | 
			
		||||
@ -80,6 +93,7 @@
 | 
			
		||||
              leading-5
 | 
			
		||||
              text-left text-gray-700
 | 
			
		||||
              border-t border-b border-gray-200 border-solid
 | 
			
		||||
              dark:text-white dark:border-white/10
 | 
			
		||||
            "
 | 
			
		||||
          >
 | 
			
		||||
            <BaseContentPlaceholders v-if="isLoading">
 | 
			
		||||
@ -99,6 +113,7 @@
 | 
			
		||||
              leading-5
 | 
			
		||||
              text-right text-gray-700
 | 
			
		||||
              border-t border-b border-gray-200 border-solid
 | 
			
		||||
              dark:text-white dark:border-white/10
 | 
			
		||||
            "
 | 
			
		||||
          >
 | 
			
		||||
            <BaseContentPlaceholders v-if="isLoading">
 | 
			
		||||
@ -145,12 +160,14 @@
 | 
			
		||||
        cursor-pointer
 | 
			
		||||
        text-primary-400
 | 
			
		||||
        hover:bg-primary-100
 | 
			
		||||
        dark:bg-gray-900/50 dark:border-white/10 dark:hover:bg-gray-900/80
 | 
			
		||||
      "
 | 
			
		||||
      @click="store.addItem"
 | 
			
		||||
    >
 | 
			
		||||
      <BaseIcon name="PlusCircleIcon" class="mr-2" />
 | 
			
		||||
      {{ $t('general.add_new_item') }}
 | 
			
		||||
    </div>
 | 
			
		||||
  </div>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
<script setup>
 | 
			
		||||
 | 
			
		||||
@ -5,7 +5,7 @@
 | 
			
		||||
    >
 | 
			
		||||
      <SelectNotePopup :type="type" @select="onSelectNote" />
 | 
			
		||||
    </div>
 | 
			
		||||
    <label class="text-gray-800 font-medium mb-4 text-sm">
 | 
			
		||||
    <label class="text-gray-800 font-medium mb-4 text-sm dark:text-gray-300">
 | 
			
		||||
      {{ $t('invoices.notes') }}
 | 
			
		||||
    </label>
 | 
			
		||||
    <BaseCustomInput
 | 
			
		||||
 | 
			
		||||
@ -6,6 +6,7 @@
 | 
			
		||||
      mt-6
 | 
			
		||||
      bg-white
 | 
			
		||||
      border border-gray-200 border-solid
 | 
			
		||||
      dark:bg-gray-800/50 dark:border-white/10
 | 
			
		||||
      rounded
 | 
			
		||||
      md:min-w-[390px]
 | 
			
		||||
      min-w-[300px]
 | 
			
		||||
@ -34,7 +35,9 @@
 | 
			
		||||
          items-center
 | 
			
		||||
          justify-center
 | 
			
		||||
          m-0
 | 
			
		||||
          text-lg text-black
 | 
			
		||||
          text-lg
 | 
			
		||||
          text-black
 | 
			
		||||
          dark:text-white
 | 
			
		||||
          uppercase
 | 
			
		||||
        "
 | 
			
		||||
      >
 | 
			
		||||
@ -71,7 +74,9 @@
 | 
			
		||||
          items-center
 | 
			
		||||
          justify-center
 | 
			
		||||
          m-0
 | 
			
		||||
          text-lg text-black
 | 
			
		||||
          text-lg
 | 
			
		||||
          text-black
 | 
			
		||||
          dark:text-white
 | 
			
		||||
          uppercase
 | 
			
		||||
        "
 | 
			
		||||
      >
 | 
			
		||||
@ -98,7 +103,7 @@
 | 
			
		||||
      <BaseContentPlaceholders v-if="isLoading">
 | 
			
		||||
        <BaseContentPlaceholdersText
 | 
			
		||||
          :lines="1"
 | 
			
		||||
          class="w-24 h-8 rounded-md border"
 | 
			
		||||
          class="w-24 h-8 border rounded-md"
 | 
			
		||||
        />
 | 
			
		||||
      </BaseContentPlaceholders>
 | 
			
		||||
      <div v-else class="flex" style="width: 140px" role="group">
 | 
			
		||||
@ -114,7 +119,7 @@
 | 
			
		||||
        <BaseDropdown position="bottom-end">
 | 
			
		||||
          <template #activator>
 | 
			
		||||
            <BaseButton
 | 
			
		||||
              class="rounded-tr-md rounded-br-md p-2 rounded-none"
 | 
			
		||||
              class="p-2 rounded-none rounded-tr-md rounded-br-md"
 | 
			
		||||
              type="button"
 | 
			
		||||
              variant="white"
 | 
			
		||||
            >
 | 
			
		||||
@ -127,7 +132,7 @@
 | 
			
		||||
 | 
			
		||||
                <BaseIcon
 | 
			
		||||
                  name="ChevronDownIcon"
 | 
			
		||||
                  class="w-4 h-4 text-gray-500 ml-1"
 | 
			
		||||
                  class="w-4 h-4 ml-1 text-gray-500"
 | 
			
		||||
                />
 | 
			
		||||
              </span>
 | 
			
		||||
            </BaseButton>
 | 
			
		||||
@ -188,6 +193,7 @@
 | 
			
		||||
        pt-2
 | 
			
		||||
        mt-5
 | 
			
		||||
        border-t border-gray-200 border-solid
 | 
			
		||||
        dark:border-gray-600
 | 
			
		||||
      "
 | 
			
		||||
    >
 | 
			
		||||
      <BaseContentPlaceholders v-if="isLoading">
 | 
			
		||||
@ -195,7 +201,7 @@
 | 
			
		||||
      </BaseContentPlaceholders>
 | 
			
		||||
      <label
 | 
			
		||||
        v-else
 | 
			
		||||
        class="m-0 text-sm font-semibold leading-5 text-gray-400 uppercase"
 | 
			
		||||
        class="m-0 text-sm font-semibold leading-5 text-gray-400 uppercase dark:text-gray-400"
 | 
			
		||||
        >{{ $t('estimates.total') }} {{ $t('estimates.amount') }}:</label
 | 
			
		||||
      >
 | 
			
		||||
 | 
			
		||||
@ -204,14 +210,7 @@
 | 
			
		||||
      </BaseContentPlaceholders>
 | 
			
		||||
      <label
 | 
			
		||||
        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" />
 | 
			
		||||
      </label>
 | 
			
		||||
@ -334,6 +333,7 @@ function selectPercentage() {
 | 
			
		||||
 | 
			
		||||
function onSelectTax(selectedTax) {
 | 
			
		||||
  let amount = 0
 | 
			
		||||
 | 
			
		||||
  if (selectedTax.compound_tax && props.store.getSubtotalWithDiscount) {
 | 
			
		||||
    amount = Math.round(
 | 
			
		||||
      ((props.store.getSubtotalWithDiscount + props.store.getTotalSimpleTax) *
 | 
			
		||||
 | 
			
		||||
@ -1,14 +1,23 @@
 | 
			
		||||
<template>
 | 
			
		||||
  <div class="flex items-center justify-between w-full mt-2 text-sm">
 | 
			
		||||
    <label class="font-semibold leading-5 text-gray-500 uppercase">
 | 
			
		||||
    <label class="font-semibold leading-5 text-gray-500 uppercase dark:text-gray-300">
 | 
			
		||||
      {{ tax.name }} ({{ tax.percent }} %)
 | 
			
		||||
    </label>
 | 
			
		||||
    <label class="flex items-center justify-center text-lg text-black">
 | 
			
		||||
    <label
 | 
			
		||||
      class="
 | 
			
		||||
        flex
 | 
			
		||||
        items-center
 | 
			
		||||
        justify-center
 | 
			
		||||
        text-lg
 | 
			
		||||
        text-black
 | 
			
		||||
        dark:text-white
 | 
			
		||||
      "
 | 
			
		||||
    >
 | 
			
		||||
      <BaseFormatMoney :amount="tax.amount" :currency="currency" />
 | 
			
		||||
 | 
			
		||||
      <BaseIcon
 | 
			
		||||
        name="TrashIcon"
 | 
			
		||||
        class="h-5 ml-2 cursor-pointer"
 | 
			
		||||
        class="h-5 ml-2 cursor-pointer dark:text-red-400"
 | 
			
		||||
        @click="$emit('remove', tax.id)"
 | 
			
		||||
      />
 | 
			
		||||
    </label>
 | 
			
		||||
 | 
			
		||||
@ -44,7 +44,7 @@
 | 
			
		||||
            >
 | 
			
		||||
              <!-- Tax Search Input  -->
 | 
			
		||||
 | 
			
		||||
              <div class="relative bg-white">
 | 
			
		||||
              <div class="relative bg-white dark:bg-gray-800">
 | 
			
		||||
                <div class="relative p-4">
 | 
			
		||||
                  <BaseInput
 | 
			
		||||
                    v-model="textSearch"
 | 
			
		||||
@ -65,13 +65,14 @@
 | 
			
		||||
                    list
 | 
			
		||||
                    max-h-36
 | 
			
		||||
                    border-t border-gray-200
 | 
			
		||||
                    dark:border-gray-600
 | 
			
		||||
                  "
 | 
			
		||||
                >
 | 
			
		||||
                  <div
 | 
			
		||||
                    v-for="(taxType, index) in filteredTaxType"
 | 
			
		||||
                    :key="index"
 | 
			
		||||
                    :class="{
 | 
			
		||||
                      'bg-gray-100 cursor-not-allowed opacity-50 pointer-events-none':
 | 
			
		||||
                      'bg-gray-100 cursor-not-allowed opacity-50 pointer-events-none dark:bg-gray-900':
 | 
			
		||||
                        taxes.find((val) => {
 | 
			
		||||
                          return val.tax_type_id === taxType.id
 | 
			
		||||
                        }),
 | 
			
		||||
@ -84,6 +85,7 @@
 | 
			
		||||
                      cursor-pointer
 | 
			
		||||
                      hover:bg-gray-100 hover:cursor-pointer
 | 
			
		||||
                      last:border-b-0
 | 
			
		||||
                      dark:border-gray-600 dark:hover:bg-gray-700/20
 | 
			
		||||
                    "
 | 
			
		||||
                    @click="selectTaxType(taxType, close)"
 | 
			
		||||
                  >
 | 
			
		||||
@ -96,6 +98,7 @@
 | 
			
		||||
                          leading-tight
 | 
			
		||||
                          text-gray-700
 | 
			
		||||
                          cursor-pointer
 | 
			
		||||
                          dark:text-gray-300
 | 
			
		||||
                        "
 | 
			
		||||
                      >
 | 
			
		||||
                        {{ taxType.name }}
 | 
			
		||||
@ -108,6 +111,7 @@
 | 
			
		||||
                          font-semibold
 | 
			
		||||
                          text-gray-700
 | 
			
		||||
                          cursor-pointer
 | 
			
		||||
                          dark:text-gray-300
 | 
			
		||||
                        "
 | 
			
		||||
                      >
 | 
			
		||||
                        {{ taxType.percent }} %
 | 
			
		||||
@ -138,6 +142,10 @@
 | 
			
		||||
                  bg-gray-200
 | 
			
		||||
                  border-none
 | 
			
		||||
                  outline-none
 | 
			
		||||
                  dark:bg-gray-600/70
 | 
			
		||||
                  dark:backdrop-blur-xl
 | 
			
		||||
                  dark:shadow-glass
 | 
			
		||||
                  dark:hover:bg-gray-600/80
 | 
			
		||||
                "
 | 
			
		||||
                @click="openTaxTypeModal"
 | 
			
		||||
              >
 | 
			
		||||
 | 
			
		||||
@ -1,6 +1,6 @@
 | 
			
		||||
<template>
 | 
			
		||||
  <div>
 | 
			
		||||
    <label class="flex text-gray-800 font-medium text-sm mb-2">
 | 
			
		||||
    <label class="flex text-gray-800 font-medium text-sm mb-2 dark:text-gray-300">
 | 
			
		||||
      {{ $t('general.select_template') }}
 | 
			
		||||
      <span class="text-sm text-red-500"> *</span>
 | 
			
		||||
    </label>
 | 
			
		||||
 | 
			
		||||
@ -57,9 +57,7 @@
 | 
			
		||||
          </BaseInputGroup>
 | 
			
		||||
        </BaseInputGrid>
 | 
			
		||||
      </div>
 | 
			
		||||
      <div
 | 
			
		||||
        class="z-0 flex justify-end p-4 border-t border-gray-200 border-solid"
 | 
			
		||||
      >
 | 
			
		||||
      <BaseModalFooter>
 | 
			
		||||
        <BaseButton
 | 
			
		||||
          class="mr-3"
 | 
			
		||||
          variant="primary-outline"
 | 
			
		||||
@ -84,7 +82,7 @@
 | 
			
		||||
          </template>
 | 
			
		||||
          {{ $t('general.create') }}
 | 
			
		||||
        </BaseButton>
 | 
			
		||||
      </div>
 | 
			
		||||
      </BaseModalFooter>
 | 
			
		||||
    </form>
 | 
			
		||||
  </BaseModal>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
@ -47,15 +47,7 @@
 | 
			
		||||
        </BaseInputGrid>
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      <div
 | 
			
		||||
        class="
 | 
			
		||||
          z-0
 | 
			
		||||
          flex
 | 
			
		||||
          justify-end
 | 
			
		||||
          p-4
 | 
			
		||||
          border-t border-gray-200 border-solid border-modal-bg
 | 
			
		||||
        "
 | 
			
		||||
      >
 | 
			
		||||
      <BaseModalFooter>
 | 
			
		||||
        <BaseButton
 | 
			
		||||
          type="button"
 | 
			
		||||
          variant="primary-outline"
 | 
			
		||||
@ -80,7 +72,7 @@
 | 
			
		||||
          </template>
 | 
			
		||||
          {{ categoryStore.isEdit ? $t('general.update') : $t('general.save') }}
 | 
			
		||||
        </BaseButton>
 | 
			
		||||
      </div>
 | 
			
		||||
      </BaseModalFooter>
 | 
			
		||||
    </form>
 | 
			
		||||
  </BaseModal>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
@ -48,6 +48,24 @@
 | 
			
		||||
            />
 | 
			
		||||
          </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
 | 
			
		||||
            :content-loading="isFetchingInitialData"
 | 
			
		||||
            :label="$tc('settings.company_info.country')"
 | 
			
		||||
@ -98,7 +116,7 @@
 | 
			
		||||
        </BaseInputGrid>
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      <div class="z-0 flex justify-end p-4 bg-gray-50 border-modal-bg">
 | 
			
		||||
       <BaseModalFooter>
 | 
			
		||||
        <BaseButton
 | 
			
		||||
          class="mr-3 text-sm"
 | 
			
		||||
          variant="primary-outline"
 | 
			
		||||
@ -123,14 +141,14 @@
 | 
			
		||||
          </template>
 | 
			
		||||
          {{ $t('general.save') }}
 | 
			
		||||
        </BaseButton>
 | 
			
		||||
      </div>
 | 
			
		||||
      </BaseModalFooter>
 | 
			
		||||
    </form>
 | 
			
		||||
  </BaseModal>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
<script setup>
 | 
			
		||||
import { useModalStore } from '@/scripts/stores/modal'
 | 
			
		||||
import { computed, onMounted, ref, reactive } from 'vue'
 | 
			
		||||
import { computed, onMounted, ref, reactive, watch } from 'vue'
 | 
			
		||||
import { useI18n } from 'vue-i18n'
 | 
			
		||||
import { required, minLength, helpers } from '@vuelidate/validators'
 | 
			
		||||
import { useVuelidate } from '@vuelidate/core'
 | 
			
		||||
@ -152,6 +170,7 @@ let companyLogoName = ref(null)
 | 
			
		||||
 | 
			
		||||
const newCompanyForm = reactive({
 | 
			
		||||
  name: null,
 | 
			
		||||
  slug: null,
 | 
			
		||||
  currency: '',
 | 
			
		||||
  address: {
 | 
			
		||||
    country_id: null,
 | 
			
		||||
@ -162,6 +181,9 @@ const modalActive = computed(() => {
 | 
			
		||||
  return modalStore.active && modalStore.componentName === 'CompanyModal'
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
const slugValidator = (value) => {
 | 
			
		||||
  return value == slugify(value)
 | 
			
		||||
}
 | 
			
		||||
const rules = {
 | 
			
		||||
  newCompanyForm: {
 | 
			
		||||
    name: {
 | 
			
		||||
@ -171,6 +193,17 @@ const rules = {
 | 
			
		||||
        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: {
 | 
			
		||||
      country_id: {
 | 
			
		||||
        required: helpers.withMessage(t('validation.required'), required),
 | 
			
		||||
@ -243,6 +276,7 @@ async function submitCompanyData() {
 | 
			
		||||
 | 
			
		||||
function resetNewCompanyForm() {
 | 
			
		||||
  newCompanyForm.name = ''
 | 
			
		||||
  newCompanyForm.slug = ''
 | 
			
		||||
  newCompanyForm.currency = ''
 | 
			
		||||
  newCompanyForm.address.country_id = ''
 | 
			
		||||
 | 
			
		||||
@ -257,4 +291,24 @@ function closeCompanyModal() {
 | 
			
		||||
    v$.value.$reset()
 | 
			
		||||
  }, 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>
 | 
			
		||||
 | 
			
		||||
@ -122,7 +122,7 @@
 | 
			
		||||
          <BaseTab :title="$t('customers.portal_access')">
 | 
			
		||||
            <BaseInputGrid class="col-span-5 lg:col-span-4">
 | 
			
		||||
              <div class="md:col-span-2">
 | 
			
		||||
                <p class="text-sm text-gray-500">
 | 
			
		||||
                <p class="text-sm text-gray-500 dark:text-gray-300">
 | 
			
		||||
                  {{ $t('customers.portal_access_text') }}
 | 
			
		||||
                </p>
 | 
			
		||||
 | 
			
		||||
@ -425,9 +425,7 @@
 | 
			
		||||
        </BaseTabGroup>
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      <div
 | 
			
		||||
        class="z-0 flex justify-end p-4 border-t border-gray-200 border-solid"
 | 
			
		||||
      >
 | 
			
		||||
      <BaseModalFooter>
 | 
			
		||||
        <BaseButton
 | 
			
		||||
          class="mr-3 text-sm"
 | 
			
		||||
          type="button"
 | 
			
		||||
@ -447,7 +445,7 @@
 | 
			
		||||
          </template>
 | 
			
		||||
          {{ $t('general.save') }}
 | 
			
		||||
        </BaseButton>
 | 
			
		||||
      </div>
 | 
			
		||||
      </BaseModalFooter>
 | 
			
		||||
    </form>
 | 
			
		||||
  </BaseModal>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
@ -38,7 +38,7 @@
 | 
			
		||||
        </BaseInputGroup>
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      <div class="z-0 flex justify-end p-4 bg-gray-50 border-modal-bg">
 | 
			
		||||
      <BaseModalFooter>
 | 
			
		||||
        <BaseButton
 | 
			
		||||
          class="mr-3 text-sm"
 | 
			
		||||
          variant="primary-outline"
 | 
			
		||||
@ -63,7 +63,7 @@
 | 
			
		||||
          </template>
 | 
			
		||||
          {{ $t('general.delete') }}
 | 
			
		||||
        </BaseButton>
 | 
			
		||||
      </div>
 | 
			
		||||
      </BaseModalFooter>
 | 
			
		||||
    </form>
 | 
			
		||||
  </BaseModal>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
@ -150,9 +150,7 @@
 | 
			
		||||
          @Remove="removeUsedSelectedCurrencies"
 | 
			
		||||
        />
 | 
			
		||||
      </div>
 | 
			
		||||
      <div
 | 
			
		||||
        class="z-0 flex justify-end p-4 border-t border-gray-200 border-solid"
 | 
			
		||||
      >
 | 
			
		||||
      <BaseModalFooter>
 | 
			
		||||
        <BaseButton
 | 
			
		||||
          class="mr-3"
 | 
			
		||||
          variant="primary-outline"
 | 
			
		||||
@ -179,7 +177,7 @@
 | 
			
		||||
            exchangeRateStore.isEdit ? $t('general.update') : $t('general.save')
 | 
			
		||||
          }}
 | 
			
		||||
        </BaseButton>
 | 
			
		||||
      </div>
 | 
			
		||||
      </BaseModalFooter>
 | 
			
		||||
    </form>
 | 
			
		||||
  </BaseModal>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
@ -20,15 +20,7 @@
 | 
			
		||||
        @submit="createNewDisk"
 | 
			
		||||
      >
 | 
			
		||||
        <template #default="slotProps">
 | 
			
		||||
          <div
 | 
			
		||||
            class="
 | 
			
		||||
              z-0
 | 
			
		||||
              flex
 | 
			
		||||
              justify-end
 | 
			
		||||
              p-4
 | 
			
		||||
              border-t border-solid border-gray-light
 | 
			
		||||
            "
 | 
			
		||||
          >
 | 
			
		||||
          <BaseModalFooter>
 | 
			
		||||
            <BaseButton
 | 
			
		||||
              class="mr-3 text-sm"
 | 
			
		||||
              variant="primary-outline"
 | 
			
		||||
@ -52,7 +44,7 @@
 | 
			
		||||
 | 
			
		||||
              {{ $t('general.save') }}
 | 
			
		||||
            </BaseButton>
 | 
			
		||||
          </div>
 | 
			
		||||
          </BaseModalFooter>
 | 
			
		||||
        </template>
 | 
			
		||||
      </component>
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
@ -89,9 +89,7 @@
 | 
			
		||||
            </BaseInputGroup>
 | 
			
		||||
          </BaseInputGrid>
 | 
			
		||||
        </div>
 | 
			
		||||
        <div
 | 
			
		||||
          class="z-0 flex justify-end p-4 border-t border-gray-200 border-solid"
 | 
			
		||||
        >
 | 
			
		||||
        <BaseModalFooter>
 | 
			
		||||
          <BaseButton
 | 
			
		||||
            class="mr-3"
 | 
			
		||||
            variant="primary-outline"
 | 
			
		||||
@ -111,7 +109,7 @@
 | 
			
		||||
            </template>
 | 
			
		||||
            {{ itemStore.isEdit ? $t('general.update') : $t('general.save') }}
 | 
			
		||||
          </BaseButton>
 | 
			
		||||
        </div>
 | 
			
		||||
        </BaseModalFooter>
 | 
			
		||||
      </form>
 | 
			
		||||
    </div>
 | 
			
		||||
  </BaseModal>
 | 
			
		||||
 | 
			
		||||
@ -31,15 +31,7 @@
 | 
			
		||||
        </BaseInputGroup>
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      <div
 | 
			
		||||
        class="
 | 
			
		||||
          z-0
 | 
			
		||||
          flex
 | 
			
		||||
          justify-end
 | 
			
		||||
          p-4
 | 
			
		||||
          border-t border-gray-200 border-solid border-modal-bg
 | 
			
		||||
        "
 | 
			
		||||
      >
 | 
			
		||||
      <BaseModalFooter>
 | 
			
		||||
        <BaseButton
 | 
			
		||||
          type="button"
 | 
			
		||||
          variant="primary-outline"
 | 
			
		||||
@ -66,7 +58,7 @@
 | 
			
		||||
            itemStore.isItemUnitEdit ? $t('general.update') : $t('general.save')
 | 
			
		||||
          }}
 | 
			
		||||
        </BaseButton>
 | 
			
		||||
      </div>
 | 
			
		||||
      </BaseModalFooter>
 | 
			
		||||
    </form>
 | 
			
		||||
  </BaseModal>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
@ -62,9 +62,7 @@
 | 
			
		||||
          </BaseInputGroup>
 | 
			
		||||
        </BaseInputGrid>
 | 
			
		||||
      </div>
 | 
			
		||||
      <div
 | 
			
		||||
        class="z-0 flex justify-end p-4 border-t border-gray-200 border-solid"
 | 
			
		||||
      >
 | 
			
		||||
     <BaseModalFooter>
 | 
			
		||||
        <BaseButton
 | 
			
		||||
          variant="primary-outline"
 | 
			
		||||
          type="button"
 | 
			
		||||
@ -84,7 +82,7 @@
 | 
			
		||||
          </template>
 | 
			
		||||
          {{ $t('general.send') }}
 | 
			
		||||
        </BaseButton>
 | 
			
		||||
      </div>
 | 
			
		||||
      </BaseModalFooter>
 | 
			
		||||
    </form>
 | 
			
		||||
  </BaseModal>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
@ -63,16 +63,7 @@
 | 
			
		||||
          </BaseInputGroup>
 | 
			
		||||
        </BaseInputGrid>
 | 
			
		||||
      </div>
 | 
			
		||||
      <div
 | 
			
		||||
        class="
 | 
			
		||||
          z-0
 | 
			
		||||
          flex
 | 
			
		||||
          justify-end
 | 
			
		||||
          px-4
 | 
			
		||||
          py-4
 | 
			
		||||
          border-t border-solid border-gray-light
 | 
			
		||||
        "
 | 
			
		||||
      >
 | 
			
		||||
     <BaseModalFooter>
 | 
			
		||||
        <BaseButton
 | 
			
		||||
          class="mr-2"
 | 
			
		||||
          variant="primary-outline"
 | 
			
		||||
@ -93,7 +84,7 @@
 | 
			
		||||
          </template>
 | 
			
		||||
          {{ noteStore.isEdit ? $t('general.update') : $t('general.save') }}
 | 
			
		||||
        </BaseButton>
 | 
			
		||||
      </div>
 | 
			
		||||
      </BaseModalFooter>
 | 
			
		||||
    </form>
 | 
			
		||||
  </BaseModal>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
@ -29,9 +29,7 @@
 | 
			
		||||
        </BaseInputGroup>
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      <div
 | 
			
		||||
        class="z-0 flex justify-end p-4 border-t border-gray-200 border-solid"
 | 
			
		||||
      >
 | 
			
		||||
      <BaseModalFooter>
 | 
			
		||||
        <BaseButton
 | 
			
		||||
          variant="primary-outline"
 | 
			
		||||
          class="mr-3"
 | 
			
		||||
@ -56,7 +54,7 @@
 | 
			
		||||
              : $t('general.save')
 | 
			
		||||
          }}
 | 
			
		||||
        </BaseButton>
 | 
			
		||||
      </div>
 | 
			
		||||
      </BaseModalFooter>
 | 
			
		||||
    </form>
 | 
			
		||||
  </BaseModal>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
@ -40,6 +40,7 @@
 | 
			
		||||
            px-4
 | 
			
		||||
            md:px-8
 | 
			
		||||
            py-1.5
 | 
			
		||||
            dark:text-gray-200
 | 
			
		||||
          "
 | 
			
		||||
        >
 | 
			
		||||
          {{ $tc('settings.roles.permission', 2) }}
 | 
			
		||||
@ -72,7 +73,7 @@
 | 
			
		||||
        </div>
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      <div class="border-t border-gray-200 py-3">
 | 
			
		||||
      <div class="border-t border-gray-200 dark:border-gray-600 py-3">
 | 
			
		||||
        <div
 | 
			
		||||
          class="
 | 
			
		||||
            grid grid-cols-1
 | 
			
		||||
@ -89,7 +90,7 @@
 | 
			
		||||
            :key="gIndex"
 | 
			
		||||
            class="flex flex-col space-y-1"
 | 
			
		||||
          >
 | 
			
		||||
            <p class="text-sm text-gray-500 border-b border-gray-200 pb-1 mb-2">
 | 
			
		||||
            <p class="text-sm text-gray-500 dark:text-gray-200 border-b dark:border-gray-600 pb-1 mb-2">
 | 
			
		||||
              {{ gIndex }}
 | 
			
		||||
            </p>
 | 
			
		||||
            <div
 | 
			
		||||
@ -116,15 +117,7 @@
 | 
			
		||||
          </span>
 | 
			
		||||
        </div>
 | 
			
		||||
      </div>
 | 
			
		||||
      <div
 | 
			
		||||
        class="
 | 
			
		||||
          z-0
 | 
			
		||||
          flex
 | 
			
		||||
          justify-end
 | 
			
		||||
          p-4
 | 
			
		||||
          border-t border-solid border--200 border-modal-bg
 | 
			
		||||
        "
 | 
			
		||||
      >
 | 
			
		||||
      <BaseModalFooter>
 | 
			
		||||
        <BaseButton
 | 
			
		||||
          class="mr-3 text-sm"
 | 
			
		||||
          variant="primary-outline"
 | 
			
		||||
@ -144,7 +137,7 @@
 | 
			
		||||
          </template>
 | 
			
		||||
          {{ !roleStore.isEdit ? $t('general.save') : $t('general.update') }}
 | 
			
		||||
        </BaseButton>
 | 
			
		||||
      </div>
 | 
			
		||||
      </BaseModalFooter>
 | 
			
		||||
    </form>
 | 
			
		||||
  </BaseModal>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
@ -70,7 +70,7 @@
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
    <div class="z-0 flex justify-end p-4 border-t border-gray-200 border-solid">
 | 
			
		||||
    <BaseModalFooter>
 | 
			
		||||
      <BaseButton class="mr-3" variant="primary-outline" @click="closeModal">
 | 
			
		||||
        {{ $t('general.cancel') }}
 | 
			
		||||
      </BaseButton>
 | 
			
		||||
@ -80,7 +80,7 @@
 | 
			
		||||
        </template>
 | 
			
		||||
        {{ $t('general.choose') }}
 | 
			
		||||
      </BaseButton>
 | 
			
		||||
    </div>
 | 
			
		||||
    </BaseModalFooter>
 | 
			
		||||
  </BaseModal>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -62,9 +62,7 @@
 | 
			
		||||
          </BaseInputGroup>
 | 
			
		||||
        </BaseInputGrid>
 | 
			
		||||
      </div>
 | 
			
		||||
      <div
 | 
			
		||||
        class="z-0 flex justify-end p-4 border-t border-gray-200 border-solid"
 | 
			
		||||
      >
 | 
			
		||||
      <BaseModalFooter>
 | 
			
		||||
        <BaseButton
 | 
			
		||||
          class="mr-3"
 | 
			
		||||
          variant="primary-outline"
 | 
			
		||||
@ -85,7 +83,7 @@
 | 
			
		||||
          <BaseIcon v-if="!isLoading" name="PhotographIcon" class="h-5 mr-2" />
 | 
			
		||||
          {{ $t('general.preview') }}
 | 
			
		||||
        </BaseButton>
 | 
			
		||||
      </div>
 | 
			
		||||
      </BaseModalFooter>
 | 
			
		||||
    </form>
 | 
			
		||||
    <div v-else>
 | 
			
		||||
      <div class="my-6 mx-4 border border-gray-200 relative">
 | 
			
		||||
@ -106,9 +104,7 @@
 | 
			
		||||
        ></iframe>
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      <div
 | 
			
		||||
        class="z-0 flex justify-end p-4 border-t border-gray-200 border-solid"
 | 
			
		||||
      >
 | 
			
		||||
      <BaseModalFooter>
 | 
			
		||||
        <BaseButton
 | 
			
		||||
          class="mr-3"
 | 
			
		||||
          variant="primary-outline"
 | 
			
		||||
@ -127,7 +123,7 @@
 | 
			
		||||
          <BaseIcon v-if="!isLoading" name="PaperAirplaneIcon" class="mr-2" />
 | 
			
		||||
          {{ $t('general.send') }}
 | 
			
		||||
        </BaseButton>
 | 
			
		||||
      </div>
 | 
			
		||||
      </BaseModalFooter>
 | 
			
		||||
    </div>
 | 
			
		||||
  </BaseModal>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
@ -65,9 +65,7 @@
 | 
			
		||||
          </BaseInputGroup>
 | 
			
		||||
        </BaseInputGrid>
 | 
			
		||||
      </div>
 | 
			
		||||
      <div
 | 
			
		||||
        class="z-0 flex justify-end p-4 border-t border-gray-200 border-solid"
 | 
			
		||||
      >
 | 
			
		||||
      <BaseModalFooter>
 | 
			
		||||
        <BaseButton
 | 
			
		||||
          class="mr-3"
 | 
			
		||||
          variant="primary-outline"
 | 
			
		||||
@ -93,7 +91,7 @@
 | 
			
		||||
          </template>
 | 
			
		||||
          {{ $t('general.preview') }}
 | 
			
		||||
        </BaseButton>
 | 
			
		||||
      </div>
 | 
			
		||||
      </BaseModalFooter>
 | 
			
		||||
    </form>
 | 
			
		||||
    <div v-else>
 | 
			
		||||
      <div class="my-6 mx-4 border border-gray-200 relative">
 | 
			
		||||
@ -114,9 +112,7 @@
 | 
			
		||||
          style="min-height: 500px"
 | 
			
		||||
        ></iframe>
 | 
			
		||||
      </div>
 | 
			
		||||
      <div
 | 
			
		||||
        class="z-0 flex justify-end p-4 border-t border-gray-200 border-solid"
 | 
			
		||||
      >
 | 
			
		||||
      <BaseModalFooter>
 | 
			
		||||
        <BaseButton
 | 
			
		||||
          class="mr-3"
 | 
			
		||||
          variant="primary-outline"
 | 
			
		||||
@ -140,7 +136,7 @@
 | 
			
		||||
          />
 | 
			
		||||
          {{ $t('general.send') }}
 | 
			
		||||
        </BaseButton>
 | 
			
		||||
      </div>
 | 
			
		||||
      </BaseModalFooter>
 | 
			
		||||
    </div>
 | 
			
		||||
  </BaseModal>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
@ -65,9 +65,7 @@
 | 
			
		||||
          </BaseInputGroup>
 | 
			
		||||
        </BaseInputGrid>
 | 
			
		||||
      </div>
 | 
			
		||||
      <div
 | 
			
		||||
        class="z-0 flex justify-end p-4 border-t border-gray-200 border-solid"
 | 
			
		||||
      >
 | 
			
		||||
      <BaseModalFooter>
 | 
			
		||||
        <BaseButton
 | 
			
		||||
          class="mr-3"
 | 
			
		||||
          variant="primary-outline"
 | 
			
		||||
@ -93,7 +91,7 @@
 | 
			
		||||
          </template>
 | 
			
		||||
          {{ $t('general.preview') }}
 | 
			
		||||
        </BaseButton>
 | 
			
		||||
      </div>
 | 
			
		||||
      </BaseModalFooter>
 | 
			
		||||
    </form>
 | 
			
		||||
    <div v-else>
 | 
			
		||||
      <div class="my-6 mx-4 border border-gray-200 relative">
 | 
			
		||||
@ -114,9 +112,7 @@
 | 
			
		||||
          style="min-height: 500px"
 | 
			
		||||
        ></iframe>
 | 
			
		||||
      </div>
 | 
			
		||||
      <div
 | 
			
		||||
        class="z-0 flex justify-end p-4 border-t border-gray-200 border-solid"
 | 
			
		||||
      >
 | 
			
		||||
      <BaseModalFooter>
 | 
			
		||||
        <BaseButton
 | 
			
		||||
          class="mr-3"
 | 
			
		||||
          variant="primary-outline"
 | 
			
		||||
@ -140,7 +136,7 @@
 | 
			
		||||
          />
 | 
			
		||||
          {{ $t('general.send') }}
 | 
			
		||||
        </BaseButton>
 | 
			
		||||
      </div>
 | 
			
		||||
      </BaseModalFooter>
 | 
			
		||||
    </div>
 | 
			
		||||
  </BaseModal>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
@ -90,15 +90,7 @@
 | 
			
		||||
          </BaseInputGroup>
 | 
			
		||||
        </BaseInputGrid>
 | 
			
		||||
      </div>
 | 
			
		||||
      <div
 | 
			
		||||
        class="
 | 
			
		||||
          z-0
 | 
			
		||||
          flex
 | 
			
		||||
          justify-end
 | 
			
		||||
          p-4
 | 
			
		||||
          border-t border-solid border--200 border-modal-bg
 | 
			
		||||
        "
 | 
			
		||||
      >
 | 
			
		||||
      <BaseModalFooter>
 | 
			
		||||
        <BaseButton
 | 
			
		||||
          class="mr-3 text-sm"
 | 
			
		||||
          variant="primary-outline"
 | 
			
		||||
@ -122,7 +114,7 @@
 | 
			
		||||
          </template>
 | 
			
		||||
          {{ taxTypeStore.isEdit ? $t('general.update') : $t('general.save') }}
 | 
			
		||||
        </BaseButton>
 | 
			
		||||
      </div>
 | 
			
		||||
      </BaseModalFooter>
 | 
			
		||||
    </form>
 | 
			
		||||
  </BaseModal>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
@ -87,9 +87,7 @@
 | 
			
		||||
        </BaseInputGrid>
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      <div
 | 
			
		||||
        class="z-0 flex justify-end p-4 border-t border-gray-200 border-solid"
 | 
			
		||||
      >
 | 
			
		||||
      <BaseModalFooter>
 | 
			
		||||
        <BaseButton
 | 
			
		||||
          class="mr-3 text-sm"
 | 
			
		||||
          type="button"
 | 
			
		||||
@ -109,7 +107,7 @@
 | 
			
		||||
          </template>
 | 
			
		||||
          {{ $t('general.save') }}
 | 
			
		||||
        </BaseButton>
 | 
			
		||||
      </div>
 | 
			
		||||
      </BaseModalFooter>
 | 
			
		||||
    </form>
 | 
			
		||||
  </BaseModal>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
@ -172,15 +172,7 @@
 | 
			
		||||
        </div>
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      <div
 | 
			
		||||
        class="
 | 
			
		||||
          z-0
 | 
			
		||||
          flex
 | 
			
		||||
          justify-end
 | 
			
		||||
          p-4
 | 
			
		||||
          border-t border-solid border-gray-light border-modal-bg
 | 
			
		||||
        "
 | 
			
		||||
      >
 | 
			
		||||
      <BaseModalFooter>
 | 
			
		||||
        <BaseButton
 | 
			
		||||
          class="mr-3"
 | 
			
		||||
          type="button"
 | 
			
		||||
@ -207,7 +199,7 @@
 | 
			
		||||
            !customFieldStore.isEdit ? $t('general.save') : $t('general.update')
 | 
			
		||||
          }}
 | 
			
		||||
        </BaseButton>
 | 
			
		||||
      </div>
 | 
			
		||||
      </BaseModalFooter>
 | 
			
		||||
    </form>
 | 
			
		||||
  </BaseModal>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
@ -153,7 +153,7 @@
 | 
			
		||||
          <BaseSwitch v-model="set_as_default" class="flex" />
 | 
			
		||||
        </div>
 | 
			
		||||
        <div class="ml-4 right">
 | 
			
		||||
          <p class="p-0 mb-1 text-base leading-snug text-black box-title">
 | 
			
		||||
          <p class="p-0 mb-1 text-base leading-snug text-black dark:text-white box-title">
 | 
			
		||||
            {{ $t('settings.disk.is_default') }}
 | 
			
		||||
          </p>
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
@ -132,7 +132,7 @@
 | 
			
		||||
          <BaseSwitch v-model="set_as_default" class="flex" />
 | 
			
		||||
        </div>
 | 
			
		||||
        <div class="ml-4 right">
 | 
			
		||||
          <p class="p-0 mb-1 text-base leading-snug text-black box-title">
 | 
			
		||||
          <p class="p-0 mb-1 text-base leading-snug text-black dark:text-white box-title">
 | 
			
		||||
            {{ $t('settings.disk.is_default') }}
 | 
			
		||||
          </p>
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
@ -63,7 +63,7 @@
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
        <div class="ml-4 right">
 | 
			
		||||
          <p class="p-0 mb-1 text-base leading-snug text-black box-title">
 | 
			
		||||
          <p class="p-0 mb-1 text-base leading-snug text-black dark:text-white box-title">
 | 
			
		||||
            {{ $t('settings.disk.is_default') }}
 | 
			
		||||
          </p>
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
@ -134,7 +134,7 @@
 | 
			
		||||
          <BaseSwitch v-model="set_as_default" class="flex" />
 | 
			
		||||
        </div>
 | 
			
		||||
        <div class="ml-4 right">
 | 
			
		||||
          <p class="p-0 mb-1 text-base leading-snug text-black box-title">
 | 
			
		||||
          <p class="p-0 mb-1 text-base leading-snug text-black dark:text-white box-title">
 | 
			
		||||
            {{ $t('settings.disk.is_default') }}
 | 
			
		||||
          </p>
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
@ -1,8 +1,8 @@
 | 
			
		||||
<template>
 | 
			
		||||
  <div class="h-screen overflow-y-auto text-base">
 | 
			
		||||
  <div class="h-screen overflow-y-auto text-base dark:bg-gray-800/80 dark:text-white">
 | 
			
		||||
    <NotificationRoot />
 | 
			
		||||
 | 
			
		||||
    <div class="container mx-auto px-4">
 | 
			
		||||
    <div class="container px-4 mx-auto">
 | 
			
		||||
      <router-view />
 | 
			
		||||
    </div>
 | 
			
		||||
  </div>
 | 
			
		||||
 | 
			
		||||
@ -1,5 +1,5 @@
 | 
			
		||||
<template>
 | 
			
		||||
  <div class="grid h-screen grid-cols-12 overflow-y-hidden bg-gray-100">
 | 
			
		||||
  <div class="grid h-screen grid-cols-12 overflow-y-hidden bg-gray-100 dark:bg-gray-900">
 | 
			
		||||
    <NotificationRoot />
 | 
			
		||||
 | 
			
		||||
    <div
 | 
			
		||||
 | 
			
		||||
@ -15,8 +15,17 @@
 | 
			
		||||
      bg-gradient-to-r
 | 
			
		||||
      from-primary-500
 | 
			
		||||
      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
 | 
			
		||||
    "
 | 
			
		||||
  >
 | 
			
		||||
    <BaseDarkHighlight />
 | 
			
		||||
 | 
			
		||||
    <router-link
 | 
			
		||||
      to="/admin/dashboard"
 | 
			
		||||
      class="
 | 
			
		||||
@ -53,6 +62,7 @@
 | 
			
		||||
        cursor-pointer
 | 
			
		||||
        md:hidden md:ml-0
 | 
			
		||||
        hover:bg-gray-100
 | 
			
		||||
        dark:bg-gray-800 dark:border-gray-500 dark:border
 | 
			
		||||
      "
 | 
			
		||||
      @click.prevent="onToggle"
 | 
			
		||||
    >
 | 
			
		||||
@ -78,19 +88,21 @@
 | 
			
		||||
                bg-white
 | 
			
		||||
                rounded
 | 
			
		||||
                md:h-9 md:w-9
 | 
			
		||||
                dark:bg-gray-700 dark:border-gray-500 dark:border
 | 
			
		||||
              "
 | 
			
		||||
            >
 | 
			
		||||
              <BaseIcon name="PlusIcon" class="w-5 h-5 text-gray-600" />
 | 
			
		||||
              <BaseIcon name="PlusIcon" class="w-5 h-5 text-gray-600 dark:text-white" />
 | 
			
		||||
            </div>
 | 
			
		||||
          </template>
 | 
			
		||||
 | 
			
		||||
          <router-link to="/admin/invoices/create">
 | 
			
		||||
            <BaseDropdownItem
 | 
			
		||||
              v-if="userStore.hasAbilities(abilities.CREATE_INVOICE)"
 | 
			
		||||
              v-slot="slotProps"
 | 
			
		||||
            >
 | 
			
		||||
              <BaseIcon
 | 
			
		||||
                name="DocumentTextIcon"
 | 
			
		||||
                class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
                :class="slotProps.class"
 | 
			
		||||
                aria-hidden="true"
 | 
			
		||||
              />
 | 
			
		||||
              {{ $t('invoices.new_invoice') }}
 | 
			
		||||
@ -99,10 +111,11 @@
 | 
			
		||||
          <router-link to="/admin/estimates/create">
 | 
			
		||||
            <BaseDropdownItem
 | 
			
		||||
              v-if="userStore.hasAbilities(abilities.CREATE_ESTIMATE)"
 | 
			
		||||
              v-slot="slotProps"
 | 
			
		||||
            >
 | 
			
		||||
              <BaseIcon
 | 
			
		||||
                name="DocumentIcon"
 | 
			
		||||
                class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
                :class="slotProps.class"
 | 
			
		||||
                aria-hidden="true"
 | 
			
		||||
              />
 | 
			
		||||
              {{ $t('estimates.new_estimate') }}
 | 
			
		||||
@ -112,10 +125,11 @@
 | 
			
		||||
          <router-link to="/admin/customers/create">
 | 
			
		||||
            <BaseDropdownItem
 | 
			
		||||
              v-if="userStore.hasAbilities(abilities.CREATE_CUSTOMER)"
 | 
			
		||||
              v-slot="slotProps"
 | 
			
		||||
            >
 | 
			
		||||
              <BaseIcon
 | 
			
		||||
                name="UserIcon"
 | 
			
		||||
                class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
                :class="slotProps.class"
 | 
			
		||||
                aria-hidden="true"
 | 
			
		||||
              />
 | 
			
		||||
              {{ $t('customers.new_customer') }}
 | 
			
		||||
@ -143,25 +157,25 @@
 | 
			
		||||
          <template #activator>
 | 
			
		||||
            <img
 | 
			
		||||
              :src="previewAvatar"
 | 
			
		||||
              class="block w-8 h-8 rounded md:h-9 md:w-9"
 | 
			
		||||
              class="block w-8 h-8 rounded md:h-9 md:w-9 object-cover"
 | 
			
		||||
            />
 | 
			
		||||
          </template>
 | 
			
		||||
 | 
			
		||||
          <router-link to="/admin/settings/account-settings">
 | 
			
		||||
            <BaseDropdownItem>
 | 
			
		||||
            <BaseDropdownItem v-slot="slotProps">
 | 
			
		||||
              <BaseIcon
 | 
			
		||||
                name="CogIcon"
 | 
			
		||||
                class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
                :class="slotProps.class"
 | 
			
		||||
                aria-hidden="true"
 | 
			
		||||
              />
 | 
			
		||||
              {{ $t('navigation.settings') }}
 | 
			
		||||
            </BaseDropdownItem>
 | 
			
		||||
          </router-link>
 | 
			
		||||
 | 
			
		||||
          <BaseDropdownItem @click="logout">
 | 
			
		||||
          <BaseDropdownItem v-slot="slotProps" @click="logout">
 | 
			
		||||
            <BaseIcon
 | 
			
		||||
              name="LogoutIcon"
 | 
			
		||||
              class="w-5 h-5 mr-3 text-gray-400 group-hover:text-gray-500"
 | 
			
		||||
              :class="slotProps.class"
 | 
			
		||||
              aria-hidden="true"
 | 
			
		||||
            />
 | 
			
		||||
            {{ $t('navigation.logout') }}
 | 
			
		||||
 | 
			
		||||
@ -15,7 +15,9 @@
 | 
			
		||||
        leave-from="opacity-100"
 | 
			
		||||
        leave-to="opacity-0"
 | 
			
		||||
      >
 | 
			
		||||
        <DialogOverlay class="fixed inset-0 bg-gray-600 bg-opacity-75" />
 | 
			
		||||
        <DialogOverlay
 | 
			
		||||
          class="fixed inset-0 bg-gray-600 bg-opacity-75 dark:bg-gray-900/90"
 | 
			
		||||
        />
 | 
			
		||||
      </TransitionChild>
 | 
			
		||||
 | 
			
		||||
      <TransitionChild
 | 
			
		||||
@ -27,7 +29,9 @@
 | 
			
		||||
        leave-from="translate-x-0"
 | 
			
		||||
        leave-to="-translate-x-full"
 | 
			
		||||
      >
 | 
			
		||||
        <div class="relative flex flex-col flex-1 w-full max-w-xs bg-white">
 | 
			
		||||
        <div
 | 
			
		||||
          class="relative flex flex-col flex-1 w-full max-w-xs bg-white dark:bg-gray-800"
 | 
			
		||||
        >
 | 
			
		||||
          <TransitionChild
 | 
			
		||||
            as="template"
 | 
			
		||||
            enter="ease-in-out duration-300"
 | 
			
		||||
@ -50,8 +54,7 @@
 | 
			
		||||
                focus:outline-none
 | 
			
		||||
                focus:ring-2
 | 
			
		||||
                focus:ring-inset
 | 
			
		||||
                  focus:ring-white
 | 
			
		||||
                "
 | 
			
		||||
                focus:ring-white"
 | 
			
		||||
                @click="globalStore.setSidebarVisibility(false)"
 | 
			
		||||
              >
 | 
			
		||||
                <span class="sr-only">Close sidebar</span>
 | 
			
		||||
@ -65,10 +68,8 @@
 | 
			
		||||
          </TransitionChild>
 | 
			
		||||
          <div class="flex-1 h-0 pt-5 pb-4 overflow-y-auto">
 | 
			
		||||
            <div class="flex items-center shrink-0 px-4 mb-10">
 | 
			
		||||
              <MainLogo
 | 
			
		||||
                class="block h-auto max-w-full w-36 text-primary-400"
 | 
			
		||||
                alt="Crater Logo"
 | 
			
		||||
              />
 | 
			
		||||
              <img class="h-auto max-w-full w-36 hidden dark:block" :src="getDarkLogo"/>
 | 
			
		||||
              <img class="h-auto max-w-full w-36 block dark:hidden" :src="getLightLogo"/>
 | 
			
		||||
            </div>
 | 
			
		||||
 | 
			
		||||
            <nav
 | 
			
		||||
@ -82,8 +83,8 @@
 | 
			
		||||
                :to="item.link"
 | 
			
		||||
                :class="[
 | 
			
		||||
                  hasActiveUrl(item.link)
 | 
			
		||||
                    ? 'text-primary-500 border-primary-500 bg-gray-100 '
 | 
			
		||||
                    : 'text-black',
 | 
			
		||||
                    ? '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-black dark:text-gray-300',
 | 
			
		||||
                  '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)"
 | 
			
		||||
@ -92,7 +93,7 @@
 | 
			
		||||
                  :name="item.icon"
 | 
			
		||||
                  :class="[
 | 
			
		||||
                    hasActiveUrl(item.link)
 | 
			
		||||
                      ? 'text-primary-500 '
 | 
			
		||||
                      ? 'text-primary-500 dark:text-primary-400'
 | 
			
		||||
                      : 'text-gray-400',
 | 
			
		||||
                    'mr-4 shrink-0 h-5 w-5',
 | 
			
		||||
                  ]"
 | 
			
		||||
@ -100,6 +101,10 @@
 | 
			
		||||
                />
 | 
			
		||||
                {{ $t(item.title) }}
 | 
			
		||||
              </router-link>
 | 
			
		||||
              <LightDarkSwitch
 | 
			
		||||
                :show-label="false"
 | 
			
		||||
                class="absolute right-6 top-6 !w-auto"
 | 
			
		||||
              />
 | 
			
		||||
            </nav>
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
@ -116,14 +121,13 @@
 | 
			
		||||
    hidden
 | 
			
		||||
    w-56
 | 
			
		||||
    h-screen
 | 
			
		||||
      pb-32
 | 
			
		||||
      overflow-y-auto
 | 
			
		||||
    bg-white
 | 
			
		||||
    border-r border-gray-200 border-solid
 | 
			
		||||
    xl:w-64
 | 
			
		||||
    md:fixed md:flex md:flex-col md:inset-y-0
 | 
			
		||||
    pt-16
 | 
			
		||||
    "
 | 
			
		||||
    dark:border-gray-800
 | 
			
		||||
    dark:bg-gray-800/80"
 | 
			
		||||
  >
 | 
			
		||||
    <div
 | 
			
		||||
      v-for="menu in globalStore.menuGroups"
 | 
			
		||||
@ -136,8 +140,8 @@
 | 
			
		||||
        :to="item.link"
 | 
			
		||||
        :class="[
 | 
			
		||||
          hasActiveUrl(item.link)
 | 
			
		||||
            ? 'text-primary-500 border-primary-500 bg-gray-100 '
 | 
			
		||||
            : 'text-black',
 | 
			
		||||
            ? '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-black dark:hover:bg-transparent dark:hover:text-white dark:text-gray-300',
 | 
			
		||||
          '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',
 | 
			
		||||
        ]"
 | 
			
		||||
      >
 | 
			
		||||
@ -145,8 +149,8 @@
 | 
			
		||||
          :name="item.icon"
 | 
			
		||||
          :class="[
 | 
			
		||||
            hasActiveUrl(item.link)
 | 
			
		||||
              ? 'text-primary-500 group-hover:text-primary-500 '
 | 
			
		||||
              : 'text-gray-400 group-hover:text-black',
 | 
			
		||||
              ? 'text-primary-500 dark:text-primary-400'
 | 
			
		||||
              : 'text-gray-400 group-hover:text-black dark:text-gray-400 dark:group-hover:text-white',
 | 
			
		||||
            'mr-4 shrink-0 h-5 w-5 ',
 | 
			
		||||
          ]"
 | 
			
		||||
        />
 | 
			
		||||
@ -154,12 +158,14 @@
 | 
			
		||||
        {{ $t(item.title) }}
 | 
			
		||||
      </router-link>
 | 
			
		||||
    </div>
 | 
			
		||||
    <LightDarkSwitch
 | 
			
		||||
      class="absolute bottom-0 py-4 border-t border-gray-200 dark:border-gray-700"
 | 
			
		||||
    />
 | 
			
		||||
  </div>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
<script setup>
 | 
			
		||||
import MainLogo from '@/scripts/components/icons/MainLogo.vue'
 | 
			
		||||
 | 
			
		||||
import { computed } from 'vue'
 | 
			
		||||
import {
 | 
			
		||||
  Dialog,
 | 
			
		||||
  DialogOverlay,
 | 
			
		||||
@ -169,10 +175,14 @@ import {
 | 
			
		||||
 | 
			
		||||
import { useRoute } from 'vue-router'
 | 
			
		||||
import { useGlobalStore } from '@/scripts/admin/stores/global'
 | 
			
		||||
import LightDarkSwitch from '@/scripts/components/LightDarkSwitcher.vue'
 | 
			
		||||
 | 
			
		||||
const route = useRoute()
 | 
			
		||||
const globalStore = useGlobalStore()
 | 
			
		||||
 | 
			
		||||
const getDarkLogo = computed(() => new URL('/img/logo-white.png', import.meta.url))
 | 
			
		||||
const getLightLogo = computed(() => new URL('/img/crater-logo.png', import.meta.url))
 | 
			
		||||
 | 
			
		||||
function hasActiveUrl(url) {
 | 
			
		||||
  return route.path.indexOf(url) > -1
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@ -184,6 +184,20 @@ export const useCompanyStore = (useWindow = false) => {
 | 
			
		||||
      setDefaultCurrency(data) {
 | 
			
		||||
        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,7 @@ export const useGlobalStore = (useWindow = false) => {
 | 
			
		||||
      isAppLoaded: false,
 | 
			
		||||
      isSidebarOpen: false,
 | 
			
		||||
      areCurrenciesLoading: false,
 | 
			
		||||
 | 
			
		||||
      isDarkModeOn: localStorage.getItem('theme') === 'dark' || document.documentElement.classList.contains('dark'),
 | 
			
		||||
      downloadReport: null,
 | 
			
		||||
    }),
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -32,7 +32,7 @@
 | 
			
		||||
    <div class="mt-4 mb-4 text-sm">
 | 
			
		||||
      <router-link
 | 
			
		||||
        to="/login"
 | 
			
		||||
        class="text-sm text-primary-400 hover:text-gray-700"
 | 
			
		||||
        class="text-sm text-primary-400 hover:text-gray-700 dark:hover:text-primary-500"
 | 
			
		||||
      >
 | 
			
		||||
        {{ $t('general.back_to_login') }}
 | 
			
		||||
      </router-link>
 | 
			
		||||
 | 
			
		||||
@ -49,7 +49,7 @@
 | 
			
		||||
      <div class="mb-4">
 | 
			
		||||
        <router-link
 | 
			
		||||
          to="forgot-password"
 | 
			
		||||
          class="text-sm text-primary-400 hover:text-gray-700"
 | 
			
		||||
          class="text-sm text-primary-400 hover:text-gray-700 dark:hover:text-primary-500"
 | 
			
		||||
        >
 | 
			
		||||
          {{ $t('login.forgot_password') }}
 | 
			
		||||
        </router-link>
 | 
			
		||||
 | 
			
		||||
@ -171,7 +171,7 @@
 | 
			
		||||
 | 
			
		||||
          <BaseInputGrid class="col-span-5 lg:col-span-4">
 | 
			
		||||
            <div class="md:col-span-2">
 | 
			
		||||
              <p class="text-sm text-gray-500">
 | 
			
		||||
              <p class="text-sm text-gray-500 dark:text-gray-300">
 | 
			
		||||
                {{ $t('customers.portal_access_text') }}
 | 
			
		||||
              </p>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -113,7 +113,7 @@
 | 
			
		||||
            </span>
 | 
			
		||||
          </template>
 | 
			
		||||
          <BaseDropdownItem @click="removeMultipleCustomers">
 | 
			
		||||
            <BaseIcon name="TrashIcon" class="mr-3 text-gray-600" />
 | 
			
		||||
            <BaseIcon name="TrashIcon" class="mr-3 text-gray-600 dark:text-gray-400 dark:group-hover:text-white" />
 | 
			
		||||
            {{ $t('general.delete') }}
 | 
			
		||||
          </BaseDropdownItem>
 | 
			
		||||
        </BaseDropdown>
 | 
			
		||||
@ -154,7 +154,7 @@
 | 
			
		||||
              :text="row.data.name"
 | 
			
		||||
              :length="30"
 | 
			
		||||
              tag="span"
 | 
			
		||||
              class="font-medium text-primary-500 flex flex-col"
 | 
			
		||||
              class="font-medium text-primary-500 flex flex-col dark:text-primary-400"
 | 
			
		||||
            />
 | 
			
		||||
            <BaseText
 | 
			
		||||
              :text="row.data.contact_name ? row.data.contact_name : ''"
 | 
			
		||||
 | 
			
		||||
@ -34,8 +34,8 @@
 | 
			
		||||
            v-if="userStore.hasAbilities(abilities.CREATE_ESTIMATE)"
 | 
			
		||||
            :to="`/admin/estimates/create?customer=${$route.params.id}`"
 | 
			
		||||
          >
 | 
			
		||||
            <BaseDropdownItem class="">
 | 
			
		||||
              <BaseIcon name="DocumentIcon" class="mr-3 text-gray-600" />
 | 
			
		||||
            <BaseDropdownItem v-slot="slotProps">
 | 
			
		||||
              <BaseIcon name="DocumentIcon" :class="slotProps.class" />
 | 
			
		||||
              {{ $t('estimates.new_estimate') }}
 | 
			
		||||
            </BaseDropdownItem>
 | 
			
		||||
          </router-link>
 | 
			
		||||
@ -44,8 +44,8 @@
 | 
			
		||||
            v-if="userStore.hasAbilities(abilities.CREATE_INVOICE)"
 | 
			
		||||
            :to="`/admin/invoices/create?customer=${$route.params.id}`"
 | 
			
		||||
          >
 | 
			
		||||
            <BaseDropdownItem>
 | 
			
		||||
              <BaseIcon name="DocumentTextIcon" class="mr-3 text-gray-600" />
 | 
			
		||||
            <BaseDropdownItem v-slot="slotProps">
 | 
			
		||||
              <BaseIcon name="DocumentTextIcon" :class="slotProps.class" />
 | 
			
		||||
              {{ $t('invoices.new_invoice') }}
 | 
			
		||||
            </BaseDropdownItem>
 | 
			
		||||
          </router-link>
 | 
			
		||||
@ -54,8 +54,8 @@
 | 
			
		||||
            v-if="userStore.hasAbilities(abilities.CREATE_PAYMENT)"
 | 
			
		||||
            :to="`/admin/payments/create?customer=${$route.params.id}`"
 | 
			
		||||
          >
 | 
			
		||||
            <BaseDropdownItem>
 | 
			
		||||
              <BaseIcon name="CreditCardIcon" class="mr-3 text-gray-600" />
 | 
			
		||||
            <BaseDropdownItem v-slot="slotProps">
 | 
			
		||||
              <BaseIcon name="CreditCardIcon" :class="slotProps.class" />
 | 
			
		||||
              {{ $t('payments.new_payment') }}
 | 
			
		||||
            </BaseDropdownItem>
 | 
			
		||||
          </router-link>
 | 
			
		||||
@ -64,8 +64,8 @@
 | 
			
		||||
            v-if="userStore.hasAbilities(abilities.CREATE_EXPENSE)"
 | 
			
		||||
            :to="`/admin/expenses/create?customer=${$route.params.id}`"
 | 
			
		||||
          >
 | 
			
		||||
            <BaseDropdownItem>
 | 
			
		||||
              <BaseIcon name="CalculatorIcon" class="mr-3 text-gray-600" />
 | 
			
		||||
            <BaseDropdownItem v-slot="slotProps">
 | 
			
		||||
              <BaseIcon name="CalculatorIcon" :class="slotProps.class" />
 | 
			
		||||
              {{ $t('expenses.new_expense') }}
 | 
			
		||||
            </BaseDropdownItem>
 | 
			
		||||
          </router-link>
 | 
			
		||||
 | 
			
		||||
@ -1,5 +1,5 @@
 | 
			
		||||
<template>
 | 
			
		||||
  <div class="pt-6 mt-5 border-t border-solid lg:pt-8 md:pt-4 border-gray-200">
 | 
			
		||||
  <div class="pt-6 mt-5 border-t border-solid lg:pt-8 md:pt-4 border-gray-200 dark:border-gray-600">
 | 
			
		||||
    <!-- Basic Info -->
 | 
			
		||||
    <BaseHeading>
 | 
			
		||||
      {{ $t('customers.basic_info') }}
 | 
			
		||||
 | 
			
		||||
@ -13,6 +13,7 @@
 | 
			
		||||
      xl:ml-64
 | 
			
		||||
      w-88
 | 
			
		||||
      xl:block
 | 
			
		||||
      dark:bg-gray-800
 | 
			
		||||
    "
 | 
			
		||||
  >
 | 
			
		||||
    <div
 | 
			
		||||
@ -25,6 +26,7 @@
 | 
			
		||||
        pb-2
 | 
			
		||||
        border border-gray-200 border-solid
 | 
			
		||||
        height-full
 | 
			
		||||
        dark:border-gray-600
 | 
			
		||||
      "
 | 
			
		||||
    >
 | 
			
		||||
      <BaseInput
 | 
			
		||||
@ -59,6 +61,7 @@
 | 
			
		||||
              mb-2
 | 
			
		||||
              text-sm
 | 
			
		||||
              border-b border-gray-200 border-solid
 | 
			
		||||
              dark:border-gray-600
 | 
			
		||||
            "
 | 
			
		||||
          >
 | 
			
		||||
            {{ $t('general.sort_by') }}
 | 
			
		||||
@ -114,6 +117,7 @@
 | 
			
		||||
        border-l border-gray-200 border-solid
 | 
			
		||||
        sidebar
 | 
			
		||||
        base-scroll
 | 
			
		||||
        dark:border-gray-600
 | 
			
		||||
      "
 | 
			
		||||
    >
 | 
			
		||||
      <div v-for="(customer, index) in customerList" :key="index">
 | 
			
		||||
@ -122,9 +126,9 @@
 | 
			
		||||
          :id="'customer-' + customer.id"
 | 
			
		||||
          :to="`/admin/customers/${customer.id}/view`"
 | 
			
		||||
          :class="[
 | 
			
		||||
            'flex justify-between p-4 items-center cursor-pointer hover:bg-gray-100 border-l-4 border-transparent',
 | 
			
		||||
            'flex justify-between p-4 items-center cursor-pointer hover:bg-gray-100 border-l-4 border-transparent dark:hover:bg-gray-700',
 | 
			
		||||
            {
 | 
			
		||||
              'bg-gray-100 border-l-4 border-primary-500 border-solid':
 | 
			
		||||
              'bg-gray-100 border-l-4 border-primary-500 border-solid dark:border-primary-400 dark:bg-gray-700':
 | 
			
		||||
                hasActiveUrl(customer.id),
 | 
			
		||||
            },
 | 
			
		||||
          ]"
 | 
			
		||||
@ -143,6 +147,7 @@
 | 
			
		||||
                text-black
 | 
			
		||||
                capitalize
 | 
			
		||||
                truncate
 | 
			
		||||
                dark:text-white
 | 
			
		||||
              "
 | 
			
		||||
            />
 | 
			
		||||
 | 
			
		||||
@ -157,6 +162,7 @@
 | 
			
		||||
                font-medium
 | 
			
		||||
                leading-5
 | 
			
		||||
                text-gray-600
 | 
			
		||||
                dark:text-gray-400
 | 
			
		||||
              "
 | 
			
		||||
            />
 | 
			
		||||
          </div>
 | 
			
		||||
@ -175,7 +181,7 @@
 | 
			
		||||
      </div>
 | 
			
		||||
      <p
 | 
			
		||||
        v-if="!customerList?.length && !isFetching"
 | 
			
		||||
        class="flex justify-center px-4 mt-5 text-sm text-gray-600"
 | 
			
		||||
        class="flex justify-center px-4 mt-5 text-sm text-gray-600 dark:text-gray-300"
 | 
			
		||||
      >
 | 
			
		||||
        {{ $t('customers.no_matching_customers') }}
 | 
			
		||||
      </p>
 | 
			
		||||
 | 
			
		||||
@ -2,8 +2,23 @@
 | 
			
		||||
  <div>
 | 
			
		||||
    <div
 | 
			
		||||
      v-if="dashboardStore.isDashboardDataLoaded"
 | 
			
		||||
      class="grid grid-cols-10 mt-8 bg-white rounded shadow"
 | 
			
		||||
      class="
 | 
			
		||||
        grid
 | 
			
		||||
        grid-cols-10
 | 
			
		||||
        mt-8
 | 
			
		||||
        bg-white
 | 
			
		||||
        rounded shadow
 | 
			
		||||
        dark:text-white
 | 
			
		||||
        dark:backdrop-blur-xl
 | 
			
		||||
        dark:shadow-glass
 | 
			
		||||
        dark:border
 | 
			
		||||
        dark:bg-opacity-70
 | 
			
		||||
        dark:border-white/10
 | 
			
		||||
        dark:bg-gray-800
 | 
			
		||||
        relative
 | 
			
		||||
      "
 | 
			
		||||
    >
 | 
			
		||||
      <BaseDarkHighlight />
 | 
			
		||||
      <!-- Chart -->
 | 
			
		||||
      <div
 | 
			
		||||
        class="
 | 
			
		||||
@ -54,6 +69,7 @@
 | 
			
		||||
          lg:border-t-0 lg:text-right lg:col-span-3
 | 
			
		||||
          xl:col-span-2
 | 
			
		||||
          lg:grid-cols-1
 | 
			
		||||
          dark:border-white/10
 | 
			
		||||
        "
 | 
			
		||||
      >
 | 
			
		||||
        <div class="p-6">
 | 
			
		||||
@ -96,15 +112,7 @@
 | 
			
		||||
          </span>
 | 
			
		||||
          <br />
 | 
			
		||||
          <span
 | 
			
		||||
            class="
 | 
			
		||||
              block
 | 
			
		||||
              mt-1
 | 
			
		||||
              text-xl
 | 
			
		||||
              font-semibold
 | 
			
		||||
              leading-8
 | 
			
		||||
              lg:text-2xl
 | 
			
		||||
              text-red-400
 | 
			
		||||
            "
 | 
			
		||||
            class="block mt-1 text-xl font-semibold leading-8 lg:text-2xl text-red-400"
 | 
			
		||||
          >
 | 
			
		||||
            <BaseFormatMoney
 | 
			
		||||
              :amount="dashboardStore.totalExpenses"
 | 
			
		||||
@ -116,8 +124,10 @@
 | 
			
		||||
          class="
 | 
			
		||||
            col-span-3
 | 
			
		||||
            p-6
 | 
			
		||||
            border-t border-gray-200 border-solid
 | 
			
		||||
            border-t
 | 
			
		||||
            border-gray-200 border-solid
 | 
			
		||||
            lg:col-span-1
 | 
			
		||||
            dark:border-white/10
 | 
			
		||||
          "
 | 
			
		||||
        >
 | 
			
		||||
          <span class="text-xs leading-5 lg:text-sm">
 | 
			
		||||
@ -132,7 +142,7 @@
 | 
			
		||||
              font-semibold
 | 
			
		||||
              leading-8
 | 
			
		||||
              lg:text-2xl
 | 
			
		||||
              text-primary-500
 | 
			
		||||
              text-primary-500 dark:text-primary-400
 | 
			
		||||
            "
 | 
			
		||||
          >
 | 
			
		||||
            <BaseFormatMoney
 | 
			
		||||
 | 
			
		||||
@ -1,6 +1,6 @@
 | 
			
		||||
<template>
 | 
			
		||||
  <BaseContentPlaceholders
 | 
			
		||||
    class="grid grid-cols-10 mt-8 bg-white rounded shadow"
 | 
			
		||||
    class="grid grid-cols-10 mt-8 bg-white rounded shadow dark:bg-gray-800"
 | 
			
		||||
  >
 | 
			
		||||
    <!-- Chart -->
 | 
			
		||||
    <div
 | 
			
		||||
@ -29,6 +29,7 @@
 | 
			
		||||
        text-center
 | 
			
		||||
        border-t border-l border-gray-200 border-solid
 | 
			
		||||
        lg:border-t-0 lg:text-right lg:col-span-3
 | 
			
		||||
        dark:border-gray-600
 | 
			
		||||
        xl:col-span-2
 | 
			
		||||
        lg:grid-cols-1
 | 
			
		||||
      "
 | 
			
		||||
@ -77,6 +78,7 @@
 | 
			
		||||
          col-span-3
 | 
			
		||||
          p-6
 | 
			
		||||
          border-t border-gray-200 border-solid
 | 
			
		||||
          dark:border-gray-600
 | 
			
		||||
          lg:justify-end lg:items-end lg:col-span-1
 | 
			
		||||
        "
 | 
			
		||||
      >
 | 
			
		||||
 | 
			
		||||
@ -12,18 +12,24 @@
 | 
			
		||||
      hover:bg-gray-50
 | 
			
		||||
      xl:p-4
 | 
			
		||||
      lg:col-span-2
 | 
			
		||||
      dark:backdrop-blur-xl
 | 
			
		||||
      dark:shadow-glass
 | 
			
		||||
      dark:border
 | 
			
		||||
      dark:border-white/10
 | 
			
		||||
      dark:bg-gray-800/70
 | 
			
		||||
    "
 | 
			
		||||
    :class="{ 'lg:!col-span-3': large }"
 | 
			
		||||
    :to="route"
 | 
			
		||||
  >
 | 
			
		||||
    <div>
 | 
			
		||||
      <span class="text-xl font-semibold leading-tight text-black xl:text-3xl">
 | 
			
		||||
      <span class="text-xl font-semibold leading-tight text-black xl:text-3xl dark:text-white">
 | 
			
		||||
        <slot />
 | 
			
		||||
      </span>
 | 
			
		||||
      <span class="block mt-1 text-sm leading-tight text-gray-500 xl:text-lg">
 | 
			
		||||
      <span class="block mt-1 text-sm leading-tight text-gray-500 xl:text-lg dark:text-gray-300">
 | 
			
		||||
        {{ label }}
 | 
			
		||||
      </span>
 | 
			
		||||
    </div>
 | 
			
		||||
    <BaseDarkHighlight class="!bg-highlight/[.17] !top-5" />
 | 
			
		||||
    <div class="flex items-center">
 | 
			
		||||
      <component :is="iconComponent" class="w-10 h-10 xl:w-12 xl:h-12" />
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
@ -1,7 +1,7 @@
 | 
			
		||||
<template>
 | 
			
		||||
  <BaseContentPlaceholders
 | 
			
		||||
    :rounded="true"
 | 
			
		||||
    class="relative flex justify-between w-full p-3 bg-white rounded shadow lg:col-span-3 xl:p-4"
 | 
			
		||||
    class="relative flex justify-between w-full p-3 bg-white rounded shadow lg:col-span-3 xl:p-4 dark:bg-gray-800"
 | 
			
		||||
  >
 | 
			
		||||
    <div>
 | 
			
		||||
      <BaseContentPlaceholdersText
 | 
			
		||||
 | 
			
		||||
@ -12,6 +12,7 @@
 | 
			
		||||
      shadow
 | 
			
		||||
      lg:col-span-2
 | 
			
		||||
      xl:p-4
 | 
			
		||||
      dark:bg-gray-800
 | 
			
		||||
    "
 | 
			
		||||
  >
 | 
			
		||||
    <div>
 | 
			
		||||
 | 
			
		||||
@ -28,7 +28,7 @@
 | 
			
		||||
          <template #cell-user="{ row }">
 | 
			
		||||
            <router-link
 | 
			
		||||
              :to="{ path: `invoices/${row.data.id}/view` }"
 | 
			
		||||
              class="font-medium text-primary-500"
 | 
			
		||||
              class="font-medium text-primary-500 dark:text-primary-400"
 | 
			
		||||
            >
 | 
			
		||||
              {{ row.data.customer.name }}
 | 
			
		||||
            </router-link>
 | 
			
		||||
@ -78,7 +78,7 @@
 | 
			
		||||
          <template #cell-user="{ row }">
 | 
			
		||||
            <router-link
 | 
			
		||||
              :to="{ path: `estimates/${row.data.id}/view` }"
 | 
			
		||||
              class="font-medium text-primary-500"
 | 
			
		||||
              class="font-medium text-primary-500 dark:text-primary-400"
 | 
			
		||||
            >
 | 
			
		||||
              {{ row.data.customer.name }}
 | 
			
		||||
            </router-link>
 | 
			
		||||
 | 
			
		||||
@ -130,6 +130,7 @@
 | 
			
		||||
          mt-5
 | 
			
		||||
          list-none
 | 
			
		||||
          border-b-2 border-gray-200 border-solid
 | 
			
		||||
          dark:border-gray-600
 | 
			
		||||
        "
 | 
			
		||||
      >
 | 
			
		||||
        <!-- Tabs -->
 | 
			
		||||
@ -163,7 +164,7 @@
 | 
			
		||||
          </template>
 | 
			
		||||
 | 
			
		||||
          <BaseDropdownItem @click="removeMultipleEstimates">
 | 
			
		||||
            <BaseIcon name="TrashIcon" class="mr-3 text-gray-600" />
 | 
			
		||||
            <BaseIcon name="TrashIcon" class="mr-3 text-gray-600 dark:text-gray-400 dark:group-hover:text-white" />
 | 
			
		||||
            {{ $t('general.delete') }}
 | 
			
		||||
          </BaseDropdownItem>
 | 
			
		||||
        </BaseDropdown>
 | 
			
		||||
@ -204,7 +205,7 @@
 | 
			
		||||
        <template #cell-estimate_number="{ row }">
 | 
			
		||||
          <router-link
 | 
			
		||||
            :to="{ path: `estimates/${row.data.id}/view` }"
 | 
			
		||||
            class="font-medium text-primary-500"
 | 
			
		||||
            class="font-medium text-primary-500 dark:text-primary-400"
 | 
			
		||||
          >
 | 
			
		||||
            {{ row.data.estimate_number }}
 | 
			
		||||
          </router-link>
 | 
			
		||||
 | 
			
		||||
@ -50,6 +50,7 @@
 | 
			
		||||
        xl:ml-64
 | 
			
		||||
        w-88
 | 
			
		||||
        xl:block
 | 
			
		||||
        dark:bg-gray-800
 | 
			
		||||
      "
 | 
			
		||||
    >
 | 
			
		||||
      <div
 | 
			
		||||
@ -62,6 +63,7 @@
 | 
			
		||||
          pb-2
 | 
			
		||||
          border border-gray-200 border-solid
 | 
			
		||||
          height-full
 | 
			
		||||
          dark:border-gray-600
 | 
			
		||||
        "
 | 
			
		||||
      >
 | 
			
		||||
        <div class="mb-6">
 | 
			
		||||
@ -96,9 +98,10 @@
 | 
			
		||||
                px-4
 | 
			
		||||
                py-1
 | 
			
		||||
                pb-2
 | 
			
		||||
                mb-1 mb-2
 | 
			
		||||
                mb-2
 | 
			
		||||
                text-sm
 | 
			
		||||
                border-b border-gray-200 border-solid
 | 
			
		||||
                dark:border-gray-600
 | 
			
		||||
              "
 | 
			
		||||
            >
 | 
			
		||||
              {{ $t('general.sort_by') }}
 | 
			
		||||
@ -161,6 +164,7 @@
 | 
			
		||||
          overflow-y-scroll
 | 
			
		||||
          border-l border-gray-200 border-solid
 | 
			
		||||
          base-scroll
 | 
			
		||||
          dark:border-gray-600
 | 
			
		||||
        "
 | 
			
		||||
      >
 | 
			
		||||
        <div v-for="(estimate, index) in estimateList" :key="index">
 | 
			
		||||
@ -169,9 +173,9 @@
 | 
			
		||||
            :id="'estimate-' + estimate.id"
 | 
			
		||||
            :to="`/admin/estimates/${estimate.id}/view`"
 | 
			
		||||
            :class="[
 | 
			
		||||
              'flex justify-between side-estimate p-4 cursor-pointer hover:bg-gray-100 items-center border-l-4 border-transparent',
 | 
			
		||||
              'flex justify-between side-estimate p-4 cursor-pointer hover:bg-gray-100 items-center border-l-4 border-transparent dark:hover:bg-gray-700',
 | 
			
		||||
              {
 | 
			
		||||
                'bg-gray-100 border-l-4 border-primary-500 border-solid':
 | 
			
		||||
                'bg-gray-100 border-l-4 border-primary-500 border-solid dark:border-primary-400 dark:bg-gray-700':
 | 
			
		||||
                  hasActiveUrl(estimate.id),
 | 
			
		||||
              },
 | 
			
		||||
            ]"
 | 
			
		||||
@ -191,6 +195,7 @@
 | 
			
		||||
                  text-black
 | 
			
		||||
                  capitalize
 | 
			
		||||
                  truncate
 | 
			
		||||
                  dark:text-white
 | 
			
		||||
                "
 | 
			
		||||
              />
 | 
			
		||||
 | 
			
		||||
@ -203,6 +208,7 @@
 | 
			
		||||
                  font-medium
 | 
			
		||||
                  leading-5
 | 
			
		||||
                  text-gray-600
 | 
			
		||||
                  dark:text-gray-400
 | 
			
		||||
                "
 | 
			
		||||
              >
 | 
			
		||||
                {{ estimate.estimate_number }}
 | 
			
		||||
@ -228,6 +234,7 @@
 | 
			
		||||
                  font-semibold
 | 
			
		||||
                  leading-8
 | 
			
		||||
                  text-right text-gray-900
 | 
			
		||||
                  dark:text-white
 | 
			
		||||
                "
 | 
			
		||||
              />
 | 
			
		||||
 | 
			
		||||
@ -239,6 +246,7 @@
 | 
			
		||||
                  leading-5
 | 
			
		||||
                  text-right text-gray-600
 | 
			
		||||
                  est-date
 | 
			
		||||
                  dark:text-gray-400
 | 
			
		||||
                "
 | 
			
		||||
              >
 | 
			
		||||
                {{ estimate.formatted_estimate_date }}
 | 
			
		||||
@ -251,7 +259,7 @@
 | 
			
		||||
        </div>
 | 
			
		||||
        <p
 | 
			
		||||
          v-if="!estimateList?.length && !isLoading"
 | 
			
		||||
          class="flex justify-center px-4 mt-5 text-sm text-gray-600"
 | 
			
		||||
          class="flex justify-center px-4 mt-5 text-sm text-gray-600 dark:text-gray-300"
 | 
			
		||||
        >
 | 
			
		||||
          {{ $t('estimates.no_matching_estimates') }}
 | 
			
		||||
        </p>
 | 
			
		||||
 | 
			
		||||
@ -32,6 +32,8 @@
 | 
			
		||||
          :content-loading="isLoading"
 | 
			
		||||
          :calendar-button="true"
 | 
			
		||||
          calendar-button-icon="calendar"
 | 
			
		||||
          :show-extra-options="true"
 | 
			
		||||
          :source-date="estimateStore.newEstimate.estimate_date"
 | 
			
		||||
        />
 | 
			
		||||
      </BaseInputGroup>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -135,7 +135,7 @@
 | 
			
		||||
            v-if="userStore.hasAbilities(abilities.DELETE_EXPENSE)"
 | 
			
		||||
            @click="removeMultipleExpenses"
 | 
			
		||||
          >
 | 
			
		||||
            <BaseIcon name="TrashIcon" class="h-5 mr-3 text-gray-600" />
 | 
			
		||||
            <BaseIcon name="TrashIcon" class="h-5 mr-3 text-gray-600 dark:text-gray-400 dark:group-hover:text-white" />
 | 
			
		||||
            {{ $t('general.delete') }}
 | 
			
		||||
          </BaseDropdownItem>
 | 
			
		||||
        </BaseDropdown>
 | 
			
		||||
@ -171,7 +171,7 @@
 | 
			
		||||
        <template #cell-name="{ row }">
 | 
			
		||||
          <router-link
 | 
			
		||||
            :to="{ path: `expenses/${row.data.id}/edit` }"
 | 
			
		||||
            class="font-medium text-primary-500"
 | 
			
		||||
            class="font-medium text-primary-500 dark:text-primary-400"
 | 
			
		||||
          >
 | 
			
		||||
            {{ row.data.expense_category.name }}
 | 
			
		||||
          </router-link>
 | 
			
		||||
 | 
			
		||||
@ -1,5 +1,6 @@
 | 
			
		||||
<template>
 | 
			
		||||
  <div class="flex flex-col items-center justify-between w-full pt-10">
 | 
			
		||||
    <LightDarkButton class="absolute z-10 top-2 right-2"/>
 | 
			
		||||
    <img
 | 
			
		||||
      id="logo-crater"
 | 
			
		||||
      src="/img/crater-logo.png"
 | 
			
		||||
@ -29,6 +30,7 @@ import Step7CompanyInfo from './Step7CompanyInfo.vue'
 | 
			
		||||
import Step8CompanyPreferences from './Step8CompanyPreferences.vue'
 | 
			
		||||
import { useInstallationStore } from '@/scripts/admin/stores/installation'
 | 
			
		||||
import { useRouter } from 'vue-router'
 | 
			
		||||
import LightDarkButton from '@/scripts/components/LightDarkButton.vue'
 | 
			
		||||
 | 
			
		||||
export default {
 | 
			
		||||
  components: {
 | 
			
		||||
@ -40,6 +42,7 @@ export default {
 | 
			
		||||
    step_6: Step6AccountSettings,
 | 
			
		||||
    step_7: Step7CompanyInfo,
 | 
			
		||||
    step_8: Step8CompanyPreferences,
 | 
			
		||||
    LightDarkButton
 | 
			
		||||
  },
 | 
			
		||||
 | 
			
		||||
  setup() {
 | 
			
		||||
 | 
			
		||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user