Compare commits

..

11 Commits

Author SHA1 Message Date
9608ab6207 remove compound interest and remove unused code 2023-08-28 15:49:20 +05:30
c14eb1b414 minor fix 2023-08-28 10:41:43 +05:30
6d0edb4b5a fix table pagination filter issue 2023-08-26 10:48:43 +05:30
0dc8941975 fix minimum total issue 2023-08-26 10:46:35 +05:30
f11437ce63 add validation on requests 2023-08-25 18:30:08 +05:30
dbd75bbe68 add changes in tax per item calculation 2023-08-25 17:45:43 +05:30
4fc67c74e4 remove commit in estimate storage 2023-08-25 09:44:43 +05:30
27660c6bce fix initial tax per item issue 2023-08-25 09:33:30 +05:30
05d5ce26fd Adds Zip to the required PHP Extension check (#1146)
* Ignore .DS_Store

* Checks for required ZIP extension
2023-02-19 11:42:34 +05:30
393fe20010 Fix incorrect invoice creation message (#1109)
Creation message now includes the views disk in its path.
2022-12-17 18:20:51 +05:30
57bdbd2897 feat: front-end files bulding (#1098)
Co-authored-by: Aramayis <>
2022-12-17 18:19:49 +05:30
68 changed files with 1789 additions and 1746 deletions

View File

@ -14,8 +14,6 @@ jobs:
steps: steps:
- name: Checkout git repo - name: Checkout git repo
uses: actions/checkout@v3 uses: actions/checkout@v3
- name: Set up Docker Buildx
uses: docker/setup-buildx-action@v2
- name: Generate UUID image name - name: Generate UUID image name
id: uuid id: uuid
run: echo "UUID_TAG_APP=$(uuidgen)" >> $GITHUB_ENV run: echo "UUID_TAG_APP=$(uuidgen)" >> $GITHUB_ENV
@ -33,11 +31,10 @@ jobs:
tags: ${{ steps.meta.outputs.tags }} tags: ${{ steps.meta.outputs.tags }}
labels: ${{ steps.meta.outputs.labels }} labels: ${{ steps.meta.outputs.labels }}
file: ./uffizzi/Dockerfile file: ./uffizzi/Dockerfile
cache-from: type=gha
cache-to: type=gha,mode=max
build-nginx: build-nginx:
needs:
- build-application
name: Build and Push `nginx` name: Build and Push `nginx`
runs-on: ubuntu-latest runs-on: ubuntu-latest
if: ${{ github.event_name != 'pull_request' || github.event.action != 'closed' }} if: ${{ github.event_name != 'pull_request' || github.event.action != 'closed' }}
@ -65,6 +62,8 @@ jobs:
tags: ${{ steps.meta.outputs.tags }} tags: ${{ steps.meta.outputs.tags }}
labels: ${{ steps.meta.outputs.labels }} labels: ${{ steps.meta.outputs.labels }}
file: ./uffizzi/nginx/Dockerfile file: ./uffizzi/nginx/Dockerfile
build-args: |
BASE_IMAGE=${{ needs.build-application.outputs.tags }}
cache-from: type=gha cache-from: type=gha
cache-to: type=gha,mode=max cache-to: type=gha,mode=max

3
.gitignore vendored
View File

@ -16,4 +16,5 @@ Homestead.yaml
.gitkeep .gitkeep
/public/docs /public/docs
/.scribe /.scribe
!storage/fonts/.gitkeep !storage/fonts/.gitkeep
.DS_Store

View File

@ -55,7 +55,7 @@ class CreateTemplateCommand extends Command
copy(public_path("/build/img/PDF/{$type}1.png"), public_path("/build/img/PDF/{$templateName}.png")); copy(public_path("/build/img/PDF/{$type}1.png"), public_path("/build/img/PDF/{$templateName}.png"));
copy(resource_path("/static/img/PDF/{$type}1.png"), resource_path("/static/img/PDF/{$templateName}.png")); copy(resource_path("/static/img/PDF/{$type}1.png"), resource_path("/static/img/PDF/{$templateName}.png"));
$path = resource_path("app/pdf/{$type}/{$templateName}.blade.php"); $path = resource_path("views/app/pdf/{$type}/{$templateName}.blade.php");
$type = ucfirst($type); $type = ucfirst($type);
$this->info("{$type} Template created successfully at ".$path); $this->info("{$type} Template created successfully at ".$path);

View File

@ -103,7 +103,6 @@ class CustomerStatsController extends Controller
) )
->whereCompany() ->whereCompany()
->whereCustomer($customer->id) ->whereCustomer($customer->id)
->where('status', '<>', Invoice::STATUS_DRAFT)
->sum('total'); ->sum('total');
$totalReceipts = Payment::whereBetween( $totalReceipts = Payment::whereBetween(
'payment_date', 'payment_date',

View File

@ -104,7 +104,6 @@ class DashboardController extends Controller
'invoice_date', 'invoice_date',
[$startDate->format('Y-m-d'), $start->format('Y-m-d')] [$startDate->format('Y-m-d'), $start->format('Y-m-d')]
) )
->where('status', '<>', Invoice::STATUS_DRAFT)
->whereCompany() ->whereCompany()
->sum('base_total'); ->sum('base_total');
@ -142,7 +141,6 @@ class DashboardController extends Controller
$recent_due_invoices = Invoice::with('customer') $recent_due_invoices = Invoice::with('customer')
->whereCompany() ->whereCompany()
->where('base_due_amount', '>', 0) ->where('base_due_amount', '>', 0)
->where('status', '<>', Invoice::STATUS_DRAFT)
->take(5) ->take(5)
->latest() ->latest()
->get(); ->get();

View File

@ -24,7 +24,6 @@ class InvoicesController extends Controller
$limit = $request->has('limit') ? $request->limit : 10; $limit = $request->has('limit') ? $request->limit : 10;
$invoices = Invoice::whereCompany() $invoices = Invoice::whereCompany()
->whereTabFilters($request->tab_status)
->join('customers', 'customers.id', '=', 'invoices.customer_id') ->join('customers', 'customers.id', '=', 'invoices.customer_id')
->applyFilters($request->all()) ->applyFilters($request->all())
->select('invoices.*', 'customers.name') ->select('invoices.*', 'customers.name')

View File

@ -3,6 +3,7 @@
namespace Crater\Http\Requests; namespace Crater\Http\Requests;
use Illuminate\Foundation\Http\FormRequest; use Illuminate\Foundation\Http\FormRequest;
use Illuminate\Support\Str;
use Illuminate\Validation\Rule; use Illuminate\Validation\Rule;
class CompaniesRequest extends FormRequest class CompaniesRequest extends FormRequest
@ -33,10 +34,6 @@ class CompaniesRequest extends FormRequest
'currency' => [ 'currency' => [
'required' 'required'
], ],
'slug' => [
'required',
Rule::unique('companies')
],
'address.name' => [ 'address.name' => [
'nullable', 'nullable',
], ],
@ -71,11 +68,11 @@ class CompaniesRequest extends FormRequest
{ {
return collect($this->validated()) return collect($this->validated())
->only([ ->only([
'name', 'name'
'slug'
]) ])
->merge([ ->merge([
'owner_id' => $this->user()->id 'owner_id' => $this->user()->id,
'slug' => Str::slug($this->name)
]) ])
->toArray(); ->toArray();
} }

View File

@ -30,8 +30,7 @@ class CompanyRequest extends FormRequest
Rule::unique('companies')->ignore($this->header('company'), 'id'), Rule::unique('companies')->ignore($this->header('company'), 'id'),
], ],
'slug' => [ 'slug' => [
'required', 'nullable'
Rule::unique('companies')->ignore($this->header('company'), 'id'),
], ],
'address.country_id' => [ 'address.country_id' => [
'required', 'required',

View File

@ -45,15 +45,21 @@ class EstimatesRequest extends FormRequest
'nullable' 'nullable'
], ],
'discount' => [ 'discount' => [
'numeric',
'required', 'required',
], ],
'discount_val' => [ 'discount_val' => [
'integer',
'required', 'required',
], ],
'sub_total' => [ 'sub_total' => [
'integer',
'required', 'required',
], ],
'total' => [ 'total' => [
'integer',
'numeric',
'max:99999999',
'required', 'required',
], ],
'tax' => [ 'tax' => [
@ -77,9 +83,11 @@ class EstimatesRequest extends FormRequest
'required', 'required',
], ],
'items.*.quantity' => [ 'items.*.quantity' => [
'integer',
'required', 'required',
], ],
'items.*.price' => [ 'items.*.price' => [
'integer',
'required', 'required',
], ],
]; ];

View File

@ -45,15 +45,21 @@ class InvoicesRequest extends FormRequest
'nullable' 'nullable'
], ],
'discount' => [ 'discount' => [
'numeric',
'required', 'required',
], ],
'discount_val' => [ 'discount_val' => [
'integer',
'required', 'required',
], ],
'sub_total' => [ 'sub_total' => [
'integer',
'required', 'required',
], ],
'total' => [ 'total' => [
'integer',
'numeric',
'max:99999999',
'required', 'required',
], ],
'tax' => [ 'tax' => [
@ -77,9 +83,11 @@ class InvoicesRequest extends FormRequest
'required', 'required',
], ],
'items.*.quantity' => [ 'items.*.quantity' => [
'integer',
'required', 'required',
], ],
'items.*.price' => [ 'items.*.price' => [
'integer',
'required', 'required',
], ],
]; ];

View File

@ -43,15 +43,21 @@ class RecurringInvoiceRequest extends FormRequest
'nullable' 'nullable'
], ],
'discount' => [ 'discount' => [
'numeric',
'required', 'required',
], ],
'discount_val' => [ 'discount_val' => [
'integer',
'required', 'required',
], ],
'sub_total' => [ 'sub_total' => [
'integer',
'required', 'required',
], ],
'total' => [ 'total' => [
'integer',
'numeric',
'max:99999999',
'required', 'required',
], ],
'tax' => [ 'tax' => [

View File

@ -23,7 +23,7 @@ class EstimateResource extends JsonResource
'reference_number' => $this->reference_number, 'reference_number' => $this->reference_number,
'tax_per_item' => $this->tax_per_item, 'tax_per_item' => $this->tax_per_item,
'discount_per_item' => $this->discount_per_item, 'discount_per_item' => $this->discount_per_item,
'notes' => $this->notes, 'notes' => $this->getNotes(),
'discount' => $this->discount, 'discount' => $this->discount,
'discount_type' => $this->discount_type, 'discount_type' => $this->discount_type,
'discount_val' => $this->discount_val, 'discount_val' => $this->discount_val,

View File

@ -18,7 +18,7 @@ class PaymentResource extends JsonResource
'id' => $this->id, 'id' => $this->id,
'payment_number' => $this->payment_number, 'payment_number' => $this->payment_number,
'payment_date' => $this->payment_date, 'payment_date' => $this->payment_date,
'notes' => $this->notes, 'notes' => $this->getNotes(),
'amount' => $this->amount, 'amount' => $this->amount,
'unique_hash' => $this->unique_hash, 'unique_hash' => $this->unique_hash,
'invoice_id' => $this->invoice_id, 'invoice_id' => $this->invoice_id,

View File

@ -217,7 +217,7 @@ class Company extends Model implements HasMedia
'estimate_billing_address_format' => $billingAddressFormat, 'estimate_billing_address_format' => $billingAddressFormat,
'payment_company_address_format' => $companyAddressFormat, 'payment_company_address_format' => $companyAddressFormat,
'payment_from_customer_address_format' => $paymentFromCustomerAddress, 'payment_from_customer_address_format' => $paymentFromCustomerAddress,
'currency' => request()->currency ?? 1, 'currency' => request()->currency ?? 13,
'time_zone' => 'Asia/Kolkata', 'time_zone' => 'Asia/Kolkata',
'language' => 'en', 'language' => 'en',
'fiscal_year' => '1-12', 'fiscal_year' => '1-12',

View File

@ -483,8 +483,7 @@ class Estimate extends Model implements HasMedia
'{ESTIMATE_DATE}' => $this->formattedEstimateDate, '{ESTIMATE_DATE}' => $this->formattedEstimateDate,
'{ESTIMATE_EXPIRY_DATE}' => $this->formattedExpiryDate, '{ESTIMATE_EXPIRY_DATE}' => $this->formattedExpiryDate,
'{ESTIMATE_NUMBER}' => $this->estimate_number, '{ESTIMATE_NUMBER}' => $this->estimate_number,
'{PDF_LINK}' => $this->estimatePdfUrl, '{ESTIMATE_REF_NUMBER}' => $this->reference_number,
'{TOTAL_AMOUNT}' => format_money_pdf($this->total, $this->customer->currency)
]; ];
} }

View File

@ -240,7 +240,7 @@ class Expense extends Model implements HasMedia
} }
if ($request->hasFile('attachment_receipt')) { if ($request->hasFile('attachment_receipt')) {
$expense->addMediaFromRequest('attachment_receipt')->toMediaCollection('receipts', 'local'); $expense->addMediaFromRequest('attachment_receipt')->toMediaCollection('receipts');
} }
if ($request->customFields) { if ($request->customFields) {
@ -262,12 +262,12 @@ class Expense extends Model implements HasMedia
ExchangeRateLog::addExchangeRateLog($this); ExchangeRateLog::addExchangeRateLog($this);
} }
if (isset($request->is_attachment_receipt_removed) && $request->is_attachment_receipt_removed == "true") { if (isset($request->is_attachment_receipt_removed) && (bool) $request->is_attachment_receipt_removed) {
$this->clearMediaCollection('receipts'); $this->clearMediaCollection('receipts');
} }
if ($request->hasFile('attachment_receipt')) { if ($request->hasFile('attachment_receipt')) {
$this->clearMediaCollection('receipts'); $this->clearMediaCollection('receipts');
$this->addMediaFromRequest('attachment_receipt')->toMediaCollection('receipts', 'local'); $this->addMediaFromRequest('attachment_receipt')->toMediaCollection('receipts');
} }
if ($request->customFields) { if ($request->customFields) {

View File

@ -187,6 +187,16 @@ class Invoice extends Model implements HasMedia
return Carbon::parse($this->invoice_date)->format($dateFormat); return Carbon::parse($this->invoice_date)->format($dateFormat);
} }
public function scopeWhereStatus($query, $status)
{
return $query->where('invoices.status', $status);
}
public function scopeWherePaidStatus($query, $status)
{
return $query->where('invoices.paid_status', $status);
}
public function scopeWhereDueStatus($query, $status) public function scopeWhereDueStatus($query, $status)
{ {
return $query->whereIn('invoices.paid_status', [ return $query->whereIn('invoices.paid_status', [
@ -224,40 +234,6 @@ class Invoice extends Model implements HasMedia
$query->orderBy($orderByField, $orderBy); $query->orderBy($orderByField, $orderBy);
} }
public function scopeWhereStatus($query, $status)
{
return $query->where('invoices.status', $status);
}
public function scopeWherePaidStatus($query, $status)
{
return $query->where('invoices.paid_status', $status);
}
public function scopeWhereTabFilters($query, $status)
{
if ($status == "DRAFT") {
return $query->where('invoices.status', $status);
}
if ($status == "SENT") {
return $query->whereIn('invoices.status', [
self::STATUS_SENT,
self::STATUS_VIEWED,
self::STATUS_COMPLETED
]);
}
if ($status == 'DUE') {
return $query->whereIn('invoices.paid_status', [
self::STATUS_UNPAID,
self::STATUS_PARTIALLY_PAID,
]);
}
return ;
}
public function scopeApplyFilters($query, array $filters) public function scopeApplyFilters($query, array $filters)
{ {
$filters = collect($filters); $filters = collect($filters);
@ -273,11 +249,17 @@ class Invoice extends Model implements HasMedia
$filters->get('status') == self::STATUS_PAID $filters->get('status') == self::STATUS_PAID
) { ) {
$query->wherePaidStatus($filters->get('status')); $query->wherePaidStatus($filters->get('status'));
} elseif ($filters->get('status') == 'DUE') {
$query->whereDueStatus($filters->get('status'));
} else { } else {
$query->whereStatus($filters->get('status')); $query->whereStatus($filters->get('status'));
} }
} }
if ($filters->get('paid_status')) {
$query->wherePaidStatus($filters->get('status'));
}
if ($filters->get('invoice_id')) { if ($filters->get('invoice_id')) {
$query->whereInvoice($filters->get('invoice_id')); $query->whereInvoice($filters->get('invoice_id'));
} }
@ -669,9 +651,7 @@ class Invoice extends Model implements HasMedia
'{INVOICE_DATE}' => $this->formattedInvoiceDate, '{INVOICE_DATE}' => $this->formattedInvoiceDate,
'{INVOICE_DUE_DATE}' => $this->formattedDueDate, '{INVOICE_DUE_DATE}' => $this->formattedDueDate,
'{INVOICE_NUMBER}' => $this->invoice_number, '{INVOICE_NUMBER}' => $this->invoice_number,
'{PDF_LINK}' => $this->invoicePdfUrl, '{INVOICE_REF_NUMBER}' => $this->reference_number,
'{DUE_AMOUNT}' => format_money_pdf($this->due_amount, $this->customer->currency),
'{TOTAL_AMOUNT}' => format_money_pdf($this->total, $this->customer->currency)
]; ];
} }

View File

@ -435,8 +435,7 @@ class Payment extends Model implements HasMedia
'{PAYMENT_DATE}' => $this->formattedPaymentDate, '{PAYMENT_DATE}' => $this->formattedPaymentDate,
'{PAYMENT_MODE}' => $this->paymentMethod ? $this->paymentMethod->name : null, '{PAYMENT_MODE}' => $this->paymentMethod ? $this->paymentMethod->name : null,
'{PAYMENT_NUMBER}' => $this->payment_number, '{PAYMENT_NUMBER}' => $this->payment_number,
'{PDF_LINK}' => $this->paymentPdfUrl, '{PAYMENT_AMOUNT}' => $this->reference_number,
'{PAYMENT_AMOUNT}' => format_money_pdf($this->amount, $this->customer->currency)
]; ];
} }

View File

@ -27,6 +27,7 @@ return [
'tokenizer', 'tokenizer',
'JSON', 'JSON',
'cURL', 'cURL',
'zip',
], ],
'apache' => [ 'apache' => [
'mod_rewrite', 'mod_rewrite',

View File

@ -27,7 +27,7 @@
"vite": "^2.6.1" "vite": "^2.6.1"
}, },
"dependencies": { "dependencies": {
"@headlessui/vue": "^1.5.0", "@headlessui/vue": "^1.4.0",
"@heroicons/vue": "^1.0.1", "@heroicons/vue": "^1.0.1",
"@popperjs/core": "^2.9.2", "@popperjs/core": "^2.9.2",
"@stripe/stripe-js": "^1.21.2", "@stripe/stripe-js": "^1.21.2",
@ -48,8 +48,7 @@
"mini-svg-data-uri": "^1.3.3", "mini-svg-data-uri": "^1.3.3",
"moment": "^2.29.1", "moment": "^2.29.1",
"pinia": "^2.0.4", "pinia": "^2.0.4",
"v-calendar": "3.0.0-alpha.8", "v-money3": "^3.13.5",
"v-money3": "3.16.1",
"v-tooltip": "^4.0.0-alpha.1", "v-tooltip": "^4.0.0-alpha.1",
"vue": "^3.2.0-beta.5", "vue": "^3.2.0-beta.5",
"vue-flatpickr-component": "^9.0.3", "vue-flatpickr-component": "^9.0.3",

View File

@ -6,17 +6,8 @@
<script setup> <script setup>
import Chart from 'chart.js' import Chart from 'chart.js'
import { import { ref, reactive, computed, onMounted, watchEffect, inject } from 'vue'
ref,
reactive,
computed,
onMounted,
watchEffect,
inject,
watch,
} from 'vue'
import { useCompanyStore } from '@/scripts/admin/stores/company' import { useCompanyStore } from '@/scripts/admin/stores/company'
import { useGlobalStore } from '@/scripts/admin/stores/global'
const utils = inject('utils') const utils = inject('utils')
@ -53,11 +44,9 @@ const props = defineProps({
}, },
}) })
const isDarkModeOn = document.documentElement.classList.contains('dark')
let myLineChart = null let myLineChart = null
const graph = ref(null) const graph = ref(null)
const companyStore = useCompanyStore() const companyStore = useCompanyStore()
const globalStore = useGlobalStore()
const defaultCurrency = computed(() => { const defaultCurrency = computed(() => {
return companyStore.selectedCompanyCurrency return companyStore.selectedCompanyCurrency
}) })
@ -71,14 +60,6 @@ watchEffect(() => {
} }
}) })
watch(
() => globalStore.isDarkModeOn,
() => {
myLineChart.reset()
updateColors()
}
)
onMounted(() => { onMounted(() => {
let context = graph.value.getContext('2d') let context = graph.value.getContext('2d')
let options = reactive({ let options = reactive({
@ -100,8 +81,6 @@ onMounted(() => {
}, },
}) })
const salesColor = globalStore.isDarkModeOn ? '#ffffff' : '#040405'
let data = reactive({ let data = reactive({
labels: props.labels, labels: props.labels,
datasets: [ datasets: [
@ -110,16 +89,16 @@ onMounted(() => {
fill: false, fill: false,
lineTension: 0.3, lineTension: 0.3,
backgroundColor: 'rgba(230, 254, 249)', backgroundColor: 'rgba(230, 254, 249)',
borderColor: salesColor, borderColor: '#040405',
borderCapStyle: 'butt', borderCapStyle: 'butt',
borderDash: [], borderDash: [],
borderDashOffset: 0.0, borderDashOffset: 0.0,
borderJoinStyle: 'miter', borderJoinStyle: 'miter',
pointBorderColor: salesColor, pointBorderColor: '#040405',
pointBackgroundColor: '#fff', pointBackgroundColor: '#fff',
pointBorderWidth: 1, pointBorderWidth: 1,
pointHoverRadius: 5, pointHoverRadius: 5,
pointHoverBackgroundColor: salesColor, pointHoverBackgroundColor: '#040405',
pointHoverBorderColor: 'rgba(220,220,220,1)', pointHoverBorderColor: 'rgba(220,220,220,1)',
pointHoverBorderWidth: 2, pointHoverBorderWidth: 2,
pointRadius: 4, pointRadius: 4,
@ -215,12 +194,4 @@ function update() {
lazy: true, 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> </script>

View File

@ -64,7 +64,7 @@ function mergeExistingValues() {
if (props.isEdit) { if (props.isEdit) {
props.store[props.storeProp].fields.forEach((field) => { props.store[props.storeProp].fields.forEach((field) => {
const existingIndex = props.store[props.storeProp].customFields.findIndex( const existingIndex = props.store[props.storeProp].customFields.findIndex(
(f) => f.id == field.custom_field_id (f) => f.id === field.custom_field_id
) )
if (existingIndex > -1) { if (existingIndex > -1) {

View File

@ -9,7 +9,7 @@ import { computed } from 'vue'
const props = defineProps({ const props = defineProps({
modelValue: { modelValue: {
type: String, type: String,
default: moment().format('YYYY-MM-DD HH:mm'), default: moment().format('YYYY-MM-DD hh:MM'),
}, },
}) })

View File

@ -271,23 +271,19 @@ const price = computed({
} else { } else {
updateItemAttribute('price', newValue) updateItemAttribute('price', newValue)
} }
setDiscount()
}, },
}) })
const subtotal = computed(() => props.itemData.price * props.itemData.quantity) const subtotal = computed(() => Math.round(props.itemData.price * props.itemData.quantity))
const discount = computed({ const discount = computed({
get: () => { get: () => {
return props.itemData.discount return props.itemData.discount
}, },
set: (newValue) => { set: (newValue) => {
if (props.itemData.discount_type === 'percentage') {
updateItemAttribute('discount_val', (subtotal.value * newValue) / 100)
} else {
updateItemAttribute('discount_val', Math.round(newValue * 100))
}
updateItemAttribute('discount', newValue) updateItemAttribute('discount', newValue)
setDiscount()
}, },
}) })
@ -313,7 +309,7 @@ const showRemoveButton = computed(() => {
const totalSimpleTax = computed(() => { const totalSimpleTax = computed(() => {
return Math.round( return Math.round(
sumBy(props.itemData.taxes, function (tax) { sumBy(props.itemData.taxes, function (tax) {
if (!tax.compound_tax) { if (tax.amount) {
return tax.amount return tax.amount
} }
return 0 return 0
@ -321,18 +317,7 @@ const totalSimpleTax = computed(() => {
) )
}) })
const totalCompoundTax = computed(() => { const totalTax = computed(() => totalSimpleTax.value)
return Math.round(
sumBy(props.itemData.taxes, function (tax) {
if (tax.compound_tax) {
return tax.amount
}
return 0
})
)
})
const totalTax = computed(() => totalSimpleTax.value + totalCompoundTax.value)
const rules = { const rules = {
name: { name: {
@ -399,7 +384,7 @@ const v$ = useVuelidate(
function updateTax(data) { function updateTax(data) {
props.store.$patch((state) => { props.store.$patch((state) => {
state[props.storeProp].items[props.index]['taxes'][data.index] = data.item state[props.storeProp].items[props.index]['taxes'][data.index] = data.item
}) })
let lastTax = props.itemData.taxes[props.itemData.taxes.length - 1] let lastTax = props.itemData.taxes[props.itemData.taxes.length - 1]
@ -416,6 +401,16 @@ function updateTax(data) {
syncItemToStore() syncItemToStore()
} }
function setDiscount() {
const newValue = props.store[props.storeProp].items[props.index].discount
if (props.itemData.discount_type === 'percentage'){
updateItemAttribute('discount_val', Math.round((subtotal.value * newValue) / 100))
}else{
updateItemAttribute('discount_val', Math.round(newValue * 100))
}
}
function searchVal(val) { function searchVal(val) {
updateItemAttribute('name', val) updateItemAttribute('name', val)
} }
@ -485,10 +480,12 @@ function syncItemToStore() {
total: total.value, total: total.value,
sub_total: subtotal.value, sub_total: subtotal.value,
totalSimpleTax: totalSimpleTax.value, totalSimpleTax: totalSimpleTax.value,
totalCompoundTax: totalCompoundTax.value,
totalTax: totalTax.value, totalTax: totalTax.value,
tax: totalTax.value, tax: totalTax.value,
taxes: [...itemTaxes], taxes: [...itemTaxes],
tax_type_ids: itemTaxes.flatMap(_t =>
_t.tax_type_id ? _t.tax_type_id : [],
),
} }
props.store.updateItem(data) props.store.updateItem(data)

View File

@ -146,14 +146,14 @@ const filteredTypes = computed(() => {
}) })
const taxAmount = computed(() => { const taxAmount = computed(() => {
if (localTax.compound_tax && props.discountedTotal) {
return ((props.discountedTotal + props.totalTax) * localTax.percent) / 100
}
if (props.discountedTotal && localTax.percent) { if (props.discountedTotal && localTax.percent) {
const taxPerItemEnabled = props.store[props.storeProp].tax_per_item === 'YES'
const discountPerItemEnabled = props.store[props.storeProp].discount_per_item === 'YES'
if (taxPerItemEnabled && !discountPerItemEnabled){
return getTaxAmount()
}
return (props.discountedTotal * localTax.percent) / 100 return (props.discountedTotal * localTax.percent) / 100
} }
return 0 return 0
}) })
@ -171,6 +171,13 @@ watch(
} }
) )
watch(
() => taxAmount.value,
() => {
updateRowTax()
},
)
// Set SelectedTax // Set SelectedTax
if (props.taxData.tax_type_id > 0) { if (props.taxData.tax_type_id > 0) {
selectedTax.value = taxTypeStore.taxTypes.find( selectedTax.value = taxTypeStore.taxTypes.find(
@ -183,7 +190,6 @@ updateRowTax()
function onSelectTax(val) { function onSelectTax(val) {
localTax.percent = val.percent localTax.percent = val.percent
localTax.tax_type_id = val.id localTax.tax_type_id = val.id
localTax.compound_tax = val.compound_tax
localTax.name = val.name localTax.name = val.name
updateRowTax() updateRowTax()
@ -220,6 +226,27 @@ function openTaxModal() {
function removeTax(index) { function removeTax(index) {
props.store.$patch((state) => { props.store.$patch((state) => {
state[props.storeProp].items[props.itemIndex].taxes.splice(index, 1) state[props.storeProp].items[props.itemIndex].taxes.splice(index, 1)
state[props.storeProp].items[props.itemIndex].tax = 0
state[props.storeProp].items[props.itemIndex].totalTax = 0
}) })
} }
function getTaxAmount() {
let total = 0
let discount = 0
const itemTotal = props.discountedTotal
const modelDiscount = props.store[props.storeProp].discount ? props.store[props.storeProp].discount : 0
const type = props.store[props.storeProp].discount_type
if (modelDiscount > 0) {
props.store[props.storeProp].items.forEach((_i) => {
total += _i.total
})
const proportion = (itemTotal / total).toFixed(2)
discount = type === 'fixed' ? modelDiscount * 100 : (total * modelDiscount) / 100
const itemDiscount = Math.round(discount * proportion)
const discounted = itemTotal - itemDiscount
return Math.round((discounted * localTax.percent) / 100)
}
return Math.round((props.discountedTotal * localTax.percent) / 100)
}
</script> </script>

View File

@ -191,7 +191,7 @@
</template> </template>
<script setup> <script setup>
import { computed, inject, ref } from 'vue' import { computed, inject, ref, watch } from 'vue'
import Guid from 'guid' import Guid from 'guid'
import Tax from './CreateTotalTaxes.vue' import Tax from './CreateTotalTaxes.vue'
import TaxStub from '@/scripts/admin/stub/abilities' import TaxStub from '@/scripts/admin/stub/abilities'
@ -227,19 +227,20 @@ const utils = inject('$utils')
const companyStore = useCompanyStore() const companyStore = useCompanyStore()
watch(
() => props.store[props.storeProp].items,
(val) => {
setDiscount()
}, { deep: true },
)
const totalDiscount = computed({ const totalDiscount = computed({
get: () => { get: () => {
return props.store[props.storeProp].discount return props.store[props.storeProp].discount
}, },
set: (newValue) => { set: (newValue) => {
if (props.store[props.storeProp].discount_type === 'percentage') {
props.store[props.storeProp].discount_val = Math.round(
(props.store.getSubTotal * newValue) / 100
)
} else {
props.store[props.storeProp].discount_val = Math.round(newValue * 100)
}
props.store[props.storeProp].discount = newValue props.store[props.storeProp].discount = newValue
setDiscount()
}, },
}) })
@ -265,7 +266,7 @@ const itemWiseTaxes = computed(() => {
} else if (tax.tax_type_id) { } else if (tax.tax_type_id) {
taxes.push({ taxes.push({
tax_type_id: tax.tax_type_id, tax_type_id: tax.tax_type_id,
amount: tax.amount, amount: Math.round(tax.amount),
percent: tax.percent, percent: tax.percent,
name: tax.name, name: tax.name,
}) })
@ -284,6 +285,19 @@ const defaultCurrency = computed(() => {
} }
}) })
function setDiscount() {
const newValue = props.store[props.storeProp].discount
if (props.store[props.storeProp].discount_type === 'percentage') {
props.store[props.storeProp].discount_val
= Math.round((props.store.getSubTotal * newValue) / 100)
return
}
props.store[props.storeProp].discount_val = Math.round(newValue * 100)
}
function selectFixed() { function selectFixed() {
if (props.store[props.storeProp].discount_type === 'fixed') { if (props.store[props.storeProp].discount_type === 'fixed') {
return return
@ -295,24 +309,21 @@ function selectFixed() {
} }
function selectPercentage() { function selectPercentage() {
if (props.store[props.storeProp].discount_type === 'percentage') { if (props.store[props.storeProp].discount_type === 'percentage'){
return return
} }
props.store[props.storeProp].discount_val =
(props.store.getSubTotal * props.store[props.storeProp].discount) / 100 const val = Math.round(props.store[props.storeProp].discount * 100) / 100
props.store[props.storeProp].discount_val
= Math.round((props.store.getSubTotal * val) / 100)
props.store[props.storeProp].discount_type = 'percentage' props.store[props.storeProp].discount_type = 'percentage'
} }
function onSelectTax(selectedTax) { function onSelectTax(selectedTax) {
let amount = 0 let amount = 0
if (props.store.getSubtotalWithDiscount && selectedTax.percent) {
if (selectedTax.compound_tax && props.store.getSubtotalWithDiscount) {
amount = Math.round(
((props.store.getSubtotalWithDiscount + props.store.getTotalSimpleTax) *
selectedTax.percent) /
100
)
} else if (props.store.getSubtotalWithDiscount && selectedTax.percent) {
amount = Math.round( amount = Math.round(
(props.store.getSubtotalWithDiscount * selectedTax.percent) / 100 (props.store.getSubtotalWithDiscount * selectedTax.percent) / 100
) )
@ -323,7 +334,6 @@ function onSelectTax(selectedTax) {
id: Guid.raw(), id: Guid.raw(),
name: selectedTax.name, name: selectedTax.name,
percent: selectedTax.percent, percent: selectedTax.percent,
compound_tax: selectedTax.compound_tax,
tax_type_id: selectedTax.id, tax_type_id: selectedTax.id,
amount, amount,
} }

View File

@ -48,24 +48,6 @@
/> />
</BaseInputGroup> </BaseInputGroup>
<BaseInputGroup
:label="$tc('settings.company_info.company_slug')"
:help-text="$t('settings.company_info.company_slug_help_text')"
:error="
v$.newCompanyForm.slug.$error &&
v$.newCompanyForm.slug.$errors[0].$message
"
:content-loading="isFetchingInitialData"
required
>
<BaseInput
v-model="newCompanyForm.slug"
:invalid="v$.newCompanyForm.slug.$error"
:content-loading="isFetchingInitialData"
@input="v$.newCompanyForm.slug.$touch()"
/>
</BaseInputGroup>
<BaseInputGroup <BaseInputGroup
:content-loading="isFetchingInitialData" :content-loading="isFetchingInitialData"
:label="$tc('settings.company_info.country')" :label="$tc('settings.company_info.country')"
@ -148,7 +130,7 @@
<script setup> <script setup>
import { useModalStore } from '@/scripts/stores/modal' import { useModalStore } from '@/scripts/stores/modal'
import { computed, onMounted, ref, reactive, watch } from 'vue' import { computed, onMounted, ref, reactive } from 'vue'
import { useI18n } from 'vue-i18n' import { useI18n } from 'vue-i18n'
import { required, minLength, helpers } from '@vuelidate/validators' import { required, minLength, helpers } from '@vuelidate/validators'
import { useVuelidate } from '@vuelidate/core' import { useVuelidate } from '@vuelidate/core'
@ -170,7 +152,6 @@ let companyLogoName = ref(null)
const newCompanyForm = reactive({ const newCompanyForm = reactive({
name: null, name: null,
slug: null,
currency: '', currency: '',
address: { address: {
country_id: null, country_id: null,
@ -181,9 +162,6 @@ const modalActive = computed(() => {
return modalStore.active && modalStore.componentName === 'CompanyModal' return modalStore.active && modalStore.componentName === 'CompanyModal'
}) })
const slugValidator = (value) => {
return value == slugify(value)
}
const rules = { const rules = {
newCompanyForm: { newCompanyForm: {
name: { name: {
@ -193,17 +171,6 @@ const rules = {
minLength(3) minLength(3)
), ),
}, },
slug: {
required: helpers.withMessage(t('validation.required'), required),
minLength: helpers.withMessage(
t('validation.name_min_length', { count: 3 }),
minLength(3)
),
slugValidator: helpers.withMessage(
t('validation.invalid_slug'),
slugValidator
),
},
address: { address: {
country_id: { country_id: {
required: helpers.withMessage(t('validation.required'), required), required: helpers.withMessage(t('validation.required'), required),
@ -276,7 +243,6 @@ async function submitCompanyData() {
function resetNewCompanyForm() { function resetNewCompanyForm() {
newCompanyForm.name = '' newCompanyForm.name = ''
newCompanyForm.slug = ''
newCompanyForm.currency = '' newCompanyForm.currency = ''
newCompanyForm.address.country_id = '' newCompanyForm.address.country_id = ''
@ -291,24 +257,4 @@ function closeCompanyModal() {
v$.value.$reset() v$.value.$reset()
}, 300) }, 300)
} }
// watcher for if change company name then auto fill company slug value
watch(
() => newCompanyForm.name,
(currentValue) => {
newCompanyForm.slug = slugify(currentValue)
}
)
function slugify(string) {
return string
.toString()
.trim()
.toLowerCase()
.replace(/\s+/g, '-')
.replace(/[^\w\-]+/g, '')
.replace(/\-\-+/g, '-')
.replace(/^-+/, '')
.replace(/-+$/, '')
}
</script> </script>

View File

@ -77,17 +77,6 @@
@input="v$.currentTaxType.description.$touch()" @input="v$.currentTaxType.description.$touch()"
/> />
</BaseInputGroup> </BaseInputGroup>
<BaseInputGroup
:label="$t('tax_types.compound_tax')"
variant="horizontal"
class="flex flex-row-reverse"
>
<BaseSwitch
v-model="taxTypeStore.currentTaxType.compound_tax"
class="flex items-center"
/>
</BaseInputGroup>
</BaseInputGrid> </BaseInputGrid>
</div> </div>
<div <div
@ -209,14 +198,7 @@ async function submitTaxTypeData() {
function SelectTax(taxData) { function SelectTax(taxData) {
let amount = 0 let amount = 0
if (taxData.compound_tax && estimateStore.getSubtotalWithDiscount) { if (estimateStore.getSubtotalWithDiscount && taxData.percent) {
amount = Math.round(
((estimateStore.getSubtotalWithDiscount +
estimateStore.getTotalSimpleTax) *
taxData.percent) /
100
)
} else if (estimateStore.getSubtotalWithDiscount && taxData.percent) {
amount = Math.round( amount = Math.round(
(estimateStore.getSubtotalWithDiscount * taxData.percent) / 100 (estimateStore.getSubtotalWithDiscount * taxData.percent) / 100
) )
@ -226,7 +208,6 @@ function SelectTax(taxData) {
id: Guid.raw(), id: Guid.raw(),
name: taxData.name, name: taxData.name,
percent: taxData.percent, percent: taxData.percent,
compound_tax: taxData.compound_tax,
tax_type_id: taxData.id, tax_type_id: taxData.id,
amount, amount,
} }
@ -242,7 +223,6 @@ function selectItemTax(taxData) {
id: Guid.raw(), id: Guid.raw(),
name: taxData.name, name: taxData.name,
percent: taxData.percent, percent: taxData.percent,
compound_tax: taxData.compound_tax,
tax_type_id: taxData.id, tax_type_id: taxData.id,
} }
modalStore.refreshData(data) modalStore.refreshData(data)

View File

@ -15,13 +15,6 @@
bg-gradient-to-r bg-gradient-to-r
from-primary-500 from-primary-500
to-primary-400 to-primary-400
dark:from-gray-700/70 dark:to-gray-800/70
bg-primary-500
dark:bg-transparent
dark:backdrop-blur-xl
dark:shadow-glass
dark:border
dark:border-white/10
" "
> >
<router-link <router-link
@ -60,7 +53,6 @@
cursor-pointer cursor-pointer
md:hidden md:ml-0 md:hidden md:ml-0
hover:bg-gray-100 hover:bg-gray-100
dark:bg-gray-800 dark:border-gray-500 dark:border
" "
@click.prevent="onToggle" @click.prevent="onToggle"
> >

View File

@ -15,9 +15,7 @@
leave-from="opacity-100" leave-from="opacity-100"
leave-to="opacity-0" leave-to="opacity-0"
> >
<DialogOverlay <DialogOverlay class="fixed inset-0 bg-gray-600 bg-opacity-75" />
class="fixed inset-0 bg-gray-600 bg-opacity-75 dark:bg-gray-900/90"
/>
</TransitionChild> </TransitionChild>
<TransitionChild <TransitionChild
@ -29,9 +27,7 @@
leave-from="translate-x-0" leave-from="translate-x-0"
leave-to="-translate-x-full" leave-to="-translate-x-full"
> >
<div <div class="relative flex flex-col flex-1 w-full max-w-xs bg-white">
class="relative flex flex-col flex-1 w-full max-w-xs bg-white dark:bg-gray-800"
>
<TransitionChild <TransitionChild
as="template" as="template"
enter="ease-in-out duration-300" enter="ease-in-out duration-300"
@ -44,17 +40,18 @@
<div class="absolute top-0 right-0 pt-2 -mr-12"> <div class="absolute top-0 right-0 pt-2 -mr-12">
<button <button
class=" class="
flex flex
items-center items-center
justify-center justify-center
w-10 w-10
h-10 h-10
ml-1 ml-1
rounded-full rounded-full
focus:outline-none focus:outline-none
focus:ring-2 focus:ring-2
focus:ring-inset focus:ring-inset
focus:ring-white" focus:ring-white
"
@click="globalStore.setSidebarVisibility(false)" @click="globalStore.setSidebarVisibility(false)"
> >
<span class="sr-only">Close sidebar</span> <span class="sr-only">Close sidebar</span>
@ -85,8 +82,8 @@
:to="item.link" :to="item.link"
:class="[ :class="[
hasActiveUrl(item.link) hasActiveUrl(item.link)
? 'text-primary-500 border-primary-500 bg-gray-100 dark:shadow-glass dark:backdrop-blur-xl dark:hover:bg-gray-700 dark:bg-gray-700/50 dark:text-primary-400 dark:font-medium' ? 'text-primary-500 border-primary-500 bg-gray-100 '
: 'text-black dark:text-gray-300', : 'text-black',
'cursor-pointer px-0 pl-4 py-3 border-transparent flex items-center border-l-4 border-solid text-sm not-italic font-medium', 'cursor-pointer px-0 pl-4 py-3 border-transparent flex items-center border-l-4 border-solid text-sm not-italic font-medium',
]" ]"
@click="globalStore.setSidebarVisibility(false)" @click="globalStore.setSidebarVisibility(false)"
@ -103,10 +100,6 @@
/> />
{{ $t(item.title) }} {{ $t(item.title) }}
</router-link> </router-link>
<LightDarkSwitch
:show-label="false"
class="absolute right-6 top-6 !w-auto"
/>
</nav> </nav>
</div> </div>
</div> </div>
@ -120,16 +113,17 @@
<!-- DESKTOP MENU --> <!-- DESKTOP MENU -->
<div <div
class=" class="
hidden hidden
w-56 w-56
h-screen h-screen
bg-white pb-32
border-r border-gray-200 border-solid overflow-y-auto
xl:w-64 bg-white
md:fixed md:flex md:flex-col md:inset-y-0 border-r border-gray-200 border-solid
pt-16 xl:w-64
dark:border-gray-800 md:fixed md:flex md:flex-col md:inset-y-0
dark:bg-gray-800/80" pt-16
"
> >
<div <div
v-for="menu in globalStore.menuGroups" v-for="menu in globalStore.menuGroups"
@ -142,8 +136,8 @@
:to="item.link" :to="item.link"
:class="[ :class="[
hasActiveUrl(item.link) hasActiveUrl(item.link)
? 'text-primary-500 border-primary-500 bg-gray-100 dark:border-primary-400 dark:shadow-glass dark:backdrop-blur-xl dark:hover:bg-gray-700 dark:bg-gray-700/50 dark:text-primary-400 dark:font-medium' ? 'text-primary-500 border-primary-500 bg-gray-100 '
: 'text-black dark:hover:bg-transparent dark:hover:text-white dark:text-gray-300', : 'text-black',
'cursor-pointer px-0 pl-6 hover:bg-gray-50 py-3 group flex items-center border-l-4 border-solid border-transparent text-sm not-italic font-medium', 'cursor-pointer px-0 pl-6 hover:bg-gray-50 py-3 group flex items-center border-l-4 border-solid border-transparent text-sm not-italic font-medium',
]" ]"
> >
@ -151,8 +145,8 @@
:name="item.icon" :name="item.icon"
:class="[ :class="[
hasActiveUrl(item.link) hasActiveUrl(item.link)
? 'text-primary-500 group-hover:text-primary-500 dark:text-primary-400 dark:group-hover:text-primary-500 ' ? 'text-primary-500 group-hover:text-primary-500 '
: 'text-gray-400 group-hover:text-black dark:text-gray-400 dark:group-hover:text-white', : 'text-gray-400 group-hover:text-black',
'mr-4 shrink-0 h-5 w-5 ', 'mr-4 shrink-0 h-5 w-5 ',
]" ]"
/> />
@ -160,9 +154,6 @@
{{ $t(item.title) }} {{ $t(item.title) }}
</router-link> </router-link>
</div> </div>
<LightDarkSwitch
class="absolute bottom-0 py-4 border-t border-gray-200 dark:border-gray-700"
/>
</div> </div>
</template> </template>
@ -178,7 +169,6 @@ import {
import { useRoute } from 'vue-router' import { useRoute } from 'vue-router'
import { useGlobalStore } from '@/scripts/admin/stores/global' import { useGlobalStore } from '@/scripts/admin/stores/global'
import LightDarkSwitch from '@/scripts/components/LightDarkSwitcher.vue'
const route = useRoute() const route = useRoute()
const globalStore = useGlobalStore() const globalStore = useGlobalStore()

View File

@ -184,20 +184,6 @@ export const useCompanyStore = (useWindow = false) => {
setDefaultCurrency(data) { setDefaultCurrency(data) {
this.defaultCurrency = data.currency this.defaultCurrency = data.currency
}, },
checkCompanyHasCurrencyTransactions() {
return new Promise((resolve, reject) => {
axios
.get(`/api/v1/company/has-transactions`)
.then((response) => {
resolve(response)
})
.catch((err) => {
handleError(err)
reject(err)
})
})
},
}, },
})() })()
} }

View File

@ -143,7 +143,8 @@ export const useEstimateStore = (useWindow = false) => {
axios axios
.get(`/api/v1/estimates/${id}`) .get(`/api/v1/estimates/${id}`)
.then((response) => { .then((response) => {
Object.assign(this.newEstimate, response.data.data) this.setEstimateData(response.data.data)
this.setCustomerAddresses(this.newEstimate.customer)
resolve(response) resolve(response)
}) })
.catch((err) => { .catch((err) => {
@ -154,6 +155,41 @@ export const useEstimateStore = (useWindow = false) => {
}) })
}, },
setEstimateData(estimate) {
Object.assign(this.newEstimate, estimate)
if (this.newEstimate.tax_per_item === 'YES') {
this.newEstimate.items.forEach((_i) => {
if (_i.taxes && !_i.taxes.length){
_i.taxes.push({ ...taxStub, id: Guid.raw() })
}
})
}
if (this.newEstimate.discount_per_item === 'YES') {
this.newEstimate.items.forEach((_i, index) => {
if (_i.discount_type === 'fixed'){
this.newEstimate.items[index].discount = _i.discount / 100
}
})
}
else {
if (this.newEstimate.discount_type === 'fixed'){
this.newEstimate.discount = this.newEstimate.discount / 100
}
}
},
setCustomerAddresses(customer) {
const customer_business = customer.customer_business
if (customer_business?.billing_address){
this.newEstimate.customer.billing_address = customer_business.billing_address
}
if (customer_business?.shipping_address){
this.newEstimate.customer.shipping_address = customer_business.shipping_address
}
},
addSalesTaxUs() { addSalesTaxUs() {
const taxTypeStore = useTaxTypeStore() const taxTypeStore = useTaxTypeStore()
let salesTax = { ...taxStub } let salesTax = { ...taxStub }

View File

@ -34,7 +34,6 @@ export const useGlobalStore = (useWindow = false) => {
isAppLoaded: false, isAppLoaded: false,
isSidebarOpen: false, isSidebarOpen: false,
areCurrenciesLoading: false, areCurrenciesLoading: false,
isDarkModeOn: false,
downloadReport: null, downloadReport: null,
}), }),
@ -71,8 +70,8 @@ export const useGlobalStore = (useWindow = false) => {
moduleStore.apiToken = response.data.global_settings.api_token moduleStore.apiToken = response.data.global_settings.api_token
moduleStore.enableModules = response.data.modules moduleStore.enableModules = response.data.modules
// company store // company store
companyStore.companies = response.data.companies companyStore.companies = response.data.companies
companyStore.selectedCompany = response.data.current_company companyStore.selectedCompany = response.data.current_company
companyStore.setSelectedCompany(response.data.current_company) companyStore.setSelectedCompany(response.data.current_company)
companyStore.selectedCompanySettings = companyStore.selectedCompanySettings =

View File

@ -133,8 +133,8 @@ export const useInvoiceStore = (useWindow = false) => {
axios axios
.get(`/api/v1/invoices/${id}`) .get(`/api/v1/invoices/${id}`)
.then((response) => { .then((response) => {
Object.assign(this.newInvoice, response.data.data) this.setInvoiceData(response.data.data)
this.newInvoice.customer = response.data.data.customer this.setCustomerAddresses(this.newInvoice.customer)
resolve(response) resolve(response)
}) })
.catch((err) => { .catch((err) => {
@ -144,6 +144,38 @@ export const useInvoiceStore = (useWindow = false) => {
}) })
}, },
setInvoiceData(invoice) {
Object.assign(this.newInvoice, invoice)
if (this.newInvoice.tax_per_item === 'YES') {
this.newInvoice.items.forEach((_i) => {
if (_i.taxes && !_i.taxes.length)
_i.taxes.push({ ...taxStub, id: Guid.raw() })
})
}
if (this.newInvoice.discount_per_item === 'YES') {
this.newInvoice.items.forEach((_i, index) => {
if (_i.discount_type === 'fixed')
this.newInvoice.items[index].discount = _i.discount / 100
})
}
else {
if (this.newInvoice.discount_type === 'fixed')
this.newInvoice.discount = this.newInvoice.discount / 100
}
},
setCustomerAddresses(customer) {
const customer_business = customer.customer_business
if (customer_business?.billing_address)
this.newInvoice.customer.billing_address = customer_business.billing_address
if (customer_business?.shipping_address)
this.newInvoice.customer.shipping_address = customer_business.shipping_address
},
addSalesTaxUs() { addSalesTaxUs() {
const taxTypeStore = useTaxTypeStore() const taxTypeStore = useTaxTypeStore()
let salesTax = { ...taxStub } let salesTax = { ...taxStub }

View File

@ -2,23 +2,8 @@
<div> <div>
<div <div
v-if="dashboardStore.isDashboardDataLoaded" v-if="dashboardStore.isDashboardDataLoaded"
class=" class="grid grid-cols-10 mt-8 bg-white rounded shadow"
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 --> <!-- Chart -->
<div <div
class=" class="
@ -69,7 +54,6 @@
lg:border-t-0 lg:text-right lg:col-span-3 lg:border-t-0 lg:text-right lg:col-span-3
xl:col-span-2 xl:col-span-2
lg:grid-cols-1 lg:grid-cols-1
dark:border-white/10
" "
> >
<div class="p-6"> <div class="p-6">
@ -112,7 +96,15 @@
</span> </span>
<br /> <br />
<span <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 <BaseFormatMoney
:amount="dashboardStore.totalExpenses" :amount="dashboardStore.totalExpenses"
@ -124,10 +116,8 @@
class=" class="
col-span-3 col-span-3
p-6 p-6
border-t border-t border-gray-200 border-solid
border-gray-200 border-solid
lg:col-span-1 lg:col-span-1
dark:border-white/10
" "
> >
<span class="text-xs leading-5 lg:text-sm"> <span class="text-xs leading-5 lg:text-sm">
@ -142,7 +132,7 @@
font-semibold font-semibold
leading-8 leading-8
lg:text-2xl lg:text-2xl
text-primary-500 dark:text-primary-400 text-primary-500
" "
> >
<BaseFormatMoney <BaseFormatMoney

View File

@ -1,6 +1,6 @@
<template> <template>
<BaseContentPlaceholders <BaseContentPlaceholders
class="grid grid-cols-10 mt-8 bg-white rounded shadow dark:bg-gray-800" class="grid grid-cols-10 mt-8 bg-white rounded shadow"
> >
<!-- Chart --> <!-- Chart -->
<div <div
@ -29,7 +29,6 @@
text-center text-center
border-t border-l border-gray-200 border-solid border-t border-l border-gray-200 border-solid
lg:border-t-0 lg:text-right lg:col-span-3 lg:border-t-0 lg:text-right lg:col-span-3
dark:border-gray-600
xl:col-span-2 xl:col-span-2
lg:grid-cols-1 lg:grid-cols-1
" "
@ -78,7 +77,6 @@
col-span-3 col-span-3
p-6 p-6
border-t border-gray-200 border-solid border-t border-gray-200 border-solid
dark:border-gray-600
lg:justify-end lg:items-end lg:col-span-1 lg:justify-end lg:items-end lg:col-span-1
" "
> >

View File

@ -12,24 +12,18 @@
hover:bg-gray-50 hover:bg-gray-50
xl:p-4 xl:p-4
lg:col-span-2 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 }" :class="{ 'lg:!col-span-3': large }"
:to="route" :to="route"
> >
<div> <div>
<span class="text-xl font-semibold leading-tight text-black xl:text-3xl dark:text-white"> <span class="text-xl font-semibold leading-tight text-black xl:text-3xl">
<slot /> <slot />
</span> </span>
<span class="block mt-1 text-sm leading-tight text-gray-500 xl:text-lg dark:text-gray-300"> <span class="block mt-1 text-sm leading-tight text-gray-500 xl:text-lg">
{{ label }} {{ label }}
</span> </span>
</div> </div>
<BaseDarkHighlight class="!bg-highlight/[.17] !top-5" />
<div class="flex items-center"> <div class="flex items-center">
<component :is="iconComponent" class="w-10 h-10 xl:w-12 xl:h-12" /> <component :is="iconComponent" class="w-10 h-10 xl:w-12 xl:h-12" />
</div> </div>

View File

@ -1,7 +1,7 @@
<template> <template>
<BaseContentPlaceholders <BaseContentPlaceholders
:rounded="true" :rounded="true"
class="relative flex justify-between w-full p-3 bg-white rounded shadow lg:col-span-3 xl:p-4 dark:bg-gray-800" class="relative flex justify-between w-full p-3 bg-white rounded shadow lg:col-span-3 xl:p-4"
> >
<div> <div>
<BaseContentPlaceholdersText <BaseContentPlaceholdersText

View File

@ -12,7 +12,6 @@
shadow shadow
lg:col-span-2 lg:col-span-2
xl:p-4 xl:p-4
dark:bg-gray-800
" "
> >
<div> <div>

View File

@ -138,6 +138,7 @@
<script setup> <script setup>
import { computed, ref, watch, onMounted } from 'vue' import { computed, ref, watch, onMounted } from 'vue'
import { cloneDeep } from 'lodash'
import { useRoute, useRouter } from 'vue-router' import { useRoute, useRouter } from 'vue-router'
import { useI18n } from 'vue-i18n' import { useI18n } from 'vue-i18n'
import { import {
@ -257,11 +258,30 @@ async function submitForm() {
isSaving.value = true isSaving.value = true
let data = { let data = cloneDeep({
...estimateStore.newEstimate, ...estimateStore.newEstimate,
sub_total: estimateStore.getSubTotal, sub_total: estimateStore.getSubTotal,
total: estimateStore.getTotal, total: estimateStore.getTotal,
tax: estimateStore.getTotalTax, tax: estimateStore.getTotalTax,
})
if (data.discount_per_item === 'YES') {
data.items.forEach((item, index) => {
if (item.discount_type === 'fixed'){
data.items[index].discount = Math.round(item.discount * 100)
}
})
}
else {
if (data.discount_type === 'fixed'){
data.discount = Math.round(data.discount * 100)
}
}
if (
!estimateStore.newEstimate.tax_per_item === 'YES'
&& data.taxes.length
){
data.tax_type_ids = data.taxes.map(_t => _t.tax_type_id)
} }
const action = isEdit.value const action = isEdit.value

View File

@ -32,8 +32,6 @@
:content-loading="isLoading" :content-loading="isLoading"
:calendar-button="true" :calendar-button="true"
calendar-button-icon="calendar" calendar-button-icon="calendar"
:show-extra-options="true"
:source-date="estimateStore.newEstimate.estimate_date"
/> />
</BaseInputGroup> </BaseInputGroup>

View File

@ -34,24 +34,6 @@
/> />
</BaseInputGroup> </BaseInputGroup>
<BaseInputGroup
:label="$tc('wizard.company_slug')"
:help-text="$t('wizard.company_slug_help_text')"
:error="
v$.companyForm.slug.$error &&
v$.companyForm.slug.$errors[0].$message
"
required
>
<BaseInput
v-model="companyForm.slug"
:invalid="v$.companyForm.slug.$error"
type="text"
name="slug"
@input="v$.companyForm.slug.$touch()"
/>
</BaseInputGroup>
<BaseInputGroup <BaseInputGroup
:label="$t('wizard.country')" :label="$t('wizard.country')"
:error=" :error="
@ -75,7 +57,9 @@
track-by="name" track-by="name"
/> />
</BaseInputGroup> </BaseInputGroup>
</div>
<div class="grid grid-cols-1 gap-4 mb-4 md:grid-cols-2 md:mb-6">
<BaseInputGroup :label="$t('wizard.state')"> <BaseInputGroup :label="$t('wizard.state')">
<BaseInput <BaseInput
v-model="companyForm.address.state" v-model="companyForm.address.state"
@ -160,9 +144,9 @@
</template> </template>
<script setup> <script setup>
import { ref, computed, onMounted, reactive, watch } from 'vue' import { ref, computed, onMounted, reactive } from 'vue'
import { useI18n } from 'vue-i18n' import { useI18n } from 'vue-i18n'
import { required, minLength, maxLength, helpers } from '@vuelidate/validators' import { required, maxLength, helpers } from '@vuelidate/validators'
import { useVuelidate } from '@vuelidate/core' import { useVuelidate } from '@vuelidate/core'
import { useGlobalStore } from '@/scripts/admin/stores/global' import { useGlobalStore } from '@/scripts/admin/stores/global'
import { useCompanyStore } from '@/scripts/admin/stores/company' import { useCompanyStore } from '@/scripts/admin/stores/company'
@ -178,7 +162,6 @@ let logoFileName = ref(null)
const companyForm = reactive({ const companyForm = reactive({
name: null, name: null,
slug: null,
address: { address: {
address_street_1: '', address_street_1: '',
address_street_2: '', address_street_2: '',
@ -205,28 +188,10 @@ onMounted(async () => {
})?.id })?.id
}) })
const slugValidator = (value) => {
return value == slugify(value)
}
const rules = { const rules = {
companyForm: { companyForm: {
name: { name: {
required: helpers.withMessage(t('validation.required'), required), required: helpers.withMessage(t('validation.required'), required),
minLength: helpers.withMessage(
t('validation.name_min_length', { count: 3 }),
minLength(3)
),
},
slug: {
required: helpers.withMessage(t('validation.required'), required),
minLength: helpers.withMessage(
t('validation.name_min_length', { count: 3 }),
minLength(3)
),
slugValidator: helpers.withMessage(
t('validation.invalid_slug'),
slugValidator
),
}, },
address: { address: {
country_id: { country_id: {
@ -284,24 +249,4 @@ async function next() {
emit('next', 7) emit('next', 7)
} }
} }
// watcher for if change company name then auto fill company slug value
watch(
() => companyForm.name,
(currentValue) => {
companyForm.slug = slugify(currentValue)
}
)
function slugify(string) {
return string
.toString()
.trim()
.toLowerCase()
.replace(/\s+/g, '-')
.replace(/[^\w\-]+/g, '')
.replace(/\-\-+/g, '-')
.replace(/^-+/, '')
.replace(/-+$/, '')
}
</script> </script>

View File

@ -56,7 +56,7 @@
<BaseMultiselect <BaseMultiselect
v-model="filters.status" v-model="filters.status"
:groups="true" :groups="true"
:options="invoiceStatus" :options="status"
searchable searchable
:placeholder="$t('general.select_a_status')" :placeholder="$t('general.select_a_status')"
@update:modelValue="setActiveTab" @update:modelValue="setActiveTab"
@ -130,27 +130,11 @@
" "
> >
<!-- Tabs --> <!-- Tabs -->
<BaseTabGroup <BaseTabGroup class="-mb-5" @change="setStatusFilter">
class="-mb-5" <BaseTab :title="$t('general.all')" filter="" />
:selected-index="selectedIndex" <BaseTab :title="$t('general.draft')" filter="DRAFT" />
@change="changeTabStatus" <BaseTab :title="$t('general.sent')" filter="SENT" />
> <BaseTab :title="$t('general.due')" filter="DUE" />
<BaseTab
:title="invoiceTabStatus[0].title"
:tab-status="invoiceTabStatus[0].value"
/>
<BaseTab
:title="invoiceTabStatus[1].title"
:tab-status="invoiceTabStatus[1].value"
/>
<BaseTab
:title="invoiceTabStatus[2].title"
:tab-status="invoiceTabStatus[2].value"
/>
<BaseTab
:title="invoiceTabStatus[3].title"
:tab-status="invoiceTabStatus[3].value"
/>
</BaseTabGroup> </BaseTabGroup>
<BaseDropdown <BaseDropdown
@ -305,10 +289,10 @@ const utils = inject('$utils')
const table = ref(null) const table = ref(null)
const showFilters = ref(false) const showFilters = ref(false)
const invoiceStatus = ref([ const status = ref([
{ {
label: 'Status', label: 'Status',
options: ['DRAFT', 'SENT', 'VIEWED', 'COMPLETED'], options: ['DRAFT', 'DUE', 'SENT', 'VIEWED', 'COMPLETED'],
}, },
{ {
label: 'Paid Status', label: 'Paid Status',
@ -316,29 +300,10 @@ const invoiceStatus = ref([
}, },
, ,
]) ])
const invoiceTabStatus = {
0: {
title: t('general.all'),
value: '',
},
1: {
title: t('general.draft'),
value: 'DRAFT',
},
2: {
title: t('general.sent'),
value: 'SENT',
},
3: {
title: t('general.due'),
value: 'DUE',
},
}
const isRequestOngoing = ref(true) const isRequestOngoing = ref(true)
const activeTab = ref('general.draft')
const router = useRouter() const router = useRouter()
const userStore = useUserStore() const userStore = useUserStore()
const selectedIndex = ref(0)
let filters = reactive({ let filters = reactive({
customer_id: '', customer_id: '',
@ -346,7 +311,6 @@ let filters = reactive({
from_date: '', from_date: '',
to_date: '', to_date: '',
invoice_number: '', invoice_number: '',
tab_status: '',
}) })
const showEmptyScreen = computed( const showEmptyScreen = computed(
@ -437,7 +401,6 @@ async function fetchData({ page, filter, sort }) {
from_date: filters.from_date, from_date: filters.from_date,
to_date: filters.to_date, to_date: filters.to_date,
invoice_number: filters.invoice_number, invoice_number: filters.invoice_number,
tab_status: filters.tab_status,
orderByField: sort.fieldName || 'created_at', orderByField: sort.fieldName || 'created_at',
orderBy: sort.order || 'desc', orderBy: sort.order || 'desc',
page, page,
@ -460,9 +423,29 @@ async function fetchData({ page, filter, sort }) {
} }
} }
function changeTabStatus(val, index) { function setStatusFilter(val) {
filters.tab_status = val['tab-status'] if (activeTab.value == val.title) {
selectedIndex.value = index return true
}
activeTab.value = val.title
switch (val.title) {
case t('general.draft'):
filters.status = 'DRAFT'
break
case t('general.sent'):
filters.status = 'SENT'
break
case t('general.due'):
filters.status = 'DUE'
break
default:
filters.status = ''
break
}
} }
function setFilters() { function setFilters() {
@ -480,6 +463,8 @@ function clearFilter() {
filters.from_date = '' filters.from_date = ''
filters.to_date = '' filters.to_date = ''
filters.invoice_number = '' filters.invoice_number = ''
activeTab.value = t('general.all')
} }
async function removeMultipleInvoices() { async function removeMultipleInvoices() {
@ -520,21 +505,39 @@ function toggleFilter() {
function setActiveTab(val) { function setActiveTab(val) {
switch (val) { switch (val) {
case 'DRAFT': case 'DRAFT':
selectedIndex.value = 1 activeTab.value = t('general.draft')
break
case 'SENT':
activeTab.value = t('general.sent')
break
case 'DUE':
activeTab.value = t('general.due')
break break
case 'SENT':
case 'VIEWED':
case 'COMPLETED': case 'COMPLETED':
activeTab.value = t('invoices.completed')
break
case 'PAID': case 'PAID':
selectedIndex.value = 2 activeTab.value = t('invoices.paid')
break break
case 'UNPAID': case 'UNPAID':
activeTab.value = t('invoices.unpaid')
break
case 'PARTIALLY_PAID': case 'PARTIALLY_PAID':
selectedIndex.value = 3 activeTab.value = t('invoices.partially_paid')
break
case 'VIEWED':
activeTab.value = t('invoices.viewed')
break
default:
activeTab.value = t('general.all')
break break
} }
filters.tab_status = invoiceTabStatus[selectedIndex.value].value
} }
</script> </script>

View File

@ -147,6 +147,7 @@ import {
decimal, decimal,
} from '@vuelidate/validators' } from '@vuelidate/validators'
import useVuelidate from '@vuelidate/core' import useVuelidate from '@vuelidate/core'
import { cloneDeep } from 'lodash'
import { useInvoiceStore } from '@/scripts/admin/stores/invoice' import { useInvoiceStore } from '@/scripts/admin/stores/invoice'
import { useModuleStore } from '@/scripts/admin/stores/module' import { useModuleStore } from '@/scripts/admin/stores/module'
@ -258,11 +259,29 @@ async function submitForm() {
isSaving.value = true isSaving.value = true
let data = { let data = cloneDeep({
...invoiceStore.newInvoice, ...invoiceStore.newInvoice,
sub_total: invoiceStore.getSubTotal, sub_total: invoiceStore.getSubTotal,
total: invoiceStore.getTotal, total: invoiceStore.getTotal,
tax: invoiceStore.getTotalTax, tax: invoiceStore.getTotalTax,
})
if (data.discount_per_item === 'YES') {
data.items.forEach((item, index) => {
if (item.discount_type === 'fixed'){
data.items[index].discount = item.discount * 100
}
})
}
else {
if (data.discount_type === 'fixed'){
data.discount = data.discount * 100
}
}
if (
!invoiceStore.newInvoice.tax_per_item === 'YES'
&& data.taxes.length
){
data.tax_type_ids = data.taxes.map(_t => _t.tax_type_id)
} }
try { try {

View File

@ -32,8 +32,6 @@
:content-loading="isLoading" :content-loading="isLoading"
:calendar-button="true" :calendar-button="true"
calendar-button-icon="calendar" calendar-button-icon="calendar"
:show-extra-options="true"
:source-date="invoiceStore.newInvoice.invoice_date"
/> />
</BaseInputGroup> </BaseInputGroup>

View File

@ -82,9 +82,9 @@
required required
> >
<BaseCustomerSelectInput <BaseCustomerSelectInput
v-if="!isLoadingContent"
v-model="paymentStore.currentPayment.customer_id" v-model="paymentStore.currentPayment.customer_id"
:content-loading="isLoadingContent" :content-loading="isLoadingContent"
v-if="!isLoadingContent"
:invalid="v$.currentPayment.customer_id.$error" :invalid="v$.currentPayment.customer_id.$error"
:placeholder="$t('customers.select_a_customer')" :placeholder="$t('customers.select_a_customer')"
show-action show-action
@ -423,7 +423,7 @@ function onCustomerChange(customer_id) {
if (customer_id) { if (customer_id) {
let data = { let data = {
customer_id: customer_id, customer_id: customer_id,
tab_status: 'DUE', status: 'DUE',
limit: 'all', limit: 'all',
} }
@ -446,11 +446,7 @@ function onCustomerChange(customer_id) {
paymentStore.currentPayment.selectedCustomer = res2.data.data paymentStore.currentPayment.selectedCustomer = res2.data.data
paymentStore.currentPayment.customer = res2.data.data paymentStore.currentPayment.customer = res2.data.data
paymentStore.currentPayment.currency = res2.data.data.currency paymentStore.currentPayment.currency = res2.data.data.currency
if ( if(isEdit.value && !customerStore.editCustomer && paymentStore.currentPayment.customer_id) {
isEdit.value &&
!customerStore.editCustomer &&
paymentStore.currentPayment.customer_id
) {
customerStore.editCustomer = res2.data.data customerStore.editCustomer = res2.data.data
} }
} }

View File

@ -28,19 +28,6 @@
/> />
</BaseInputGroup> </BaseInputGroup>
<BaseInputGroup
:label="$tc('settings.company_info.company_slug')"
:help-text="$t('settings.company_info.company_slug_help_text')"
:error="v$.slug.$error && v$.slug.$errors[0].$message"
required
>
<BaseInput
v-model="companyForm.slug"
:invalid="v$.slug.$error"
@blur="v$.slug.$touch()"
/>
</BaseInputGroup>
<BaseInputGroup :label="$tc('settings.company_info.phone')"> <BaseInputGroup :label="$tc('settings.company_info.phone')">
<BaseInput v-model="companyForm.address.phone" /> <BaseInput v-model="companyForm.address.phone" />
</BaseInputGroup> </BaseInputGroup>
@ -173,7 +160,6 @@ let isSaving = ref(false)
const companyForm = reactive({ const companyForm = reactive({
name: null, name: null,
slug: null,
logo: null, logo: null,
address: { address: {
address_street_1: '', address_street_1: '',
@ -207,14 +193,7 @@ const rules = computed(() => {
name: { name: {
required: helpers.withMessage(t('validation.required'), required), required: helpers.withMessage(t('validation.required'), required),
minLength: helpers.withMessage( minLength: helpers.withMessage(
t('validation.name_min_length', { count: 3 }), t('validation.name_min_length'),
minLength(3)
),
},
slug: {
required: helpers.withMessage(t('validation.required'), required),
minLength: helpers.withMessage(
t('validation.name_min_length', { count: 3 }),
minLength(3) minLength(3)
), ),
}, },

View File

@ -8,11 +8,7 @@
<BaseInputGroup <BaseInputGroup
:content-loading="isFetchingInitialData" :content-loading="isFetchingInitialData"
:label="$tc('settings.preferences.currency')" :label="$tc('settings.preferences.currency')"
:help-text=" :help-text="$t('settings.preferences.company_currency_unchangeable')"
isCurrencyDisabled
? $t('settings.preferences.company_currency_unchangeable')
: ''
"
:error="v$.currency.$error && v$.currency.$errors[0].$message" :error="v$.currency.$error && v$.currency.$errors[0].$message"
required required
> >
@ -25,7 +21,7 @@
:searchable="true" :searchable="true"
track-by="name" track-by="name"
:invalid="v$.currency.$error" :invalid="v$.currency.$error"
:disabled="isCurrencyDisabled" disabled
class="w-full" class="w-full"
> >
</BaseMultiselect> </BaseMultiselect>
@ -191,7 +187,6 @@ const { t, tm } = useI18n()
let isSaving = ref(false) let isSaving = ref(false)
let isDataSaving = ref(false) let isDataSaving = ref(false)
let isFetchingInitialData = ref(false) let isFetchingInitialData = ref(false)
let isCurrencyDisabled = ref(true)
const settingsForm = reactive({ ...companyStore.selectedCompanySettings }) const settingsForm = reactive({ ...companyStore.selectedCompanySettings })
@ -287,14 +282,10 @@ setInitialData()
async function setInitialData() { async function setInitialData() {
isFetchingInitialData.value = true isFetchingInitialData.value = true
Promise.all([ Promise.all([
companyStore.checkCompanyHasCurrencyTransactions(),
globalStore.fetchCurrencies(), globalStore.fetchCurrencies(),
globalStore.fetchDateFormats(), globalStore.fetchDateFormats(),
globalStore.fetchTimeZones(), globalStore.fetchTimeZones(),
]).then(([res1]) => { ]).then(([res1]) => {
if (res1.data?.has_transactions == false) {
isCurrencyDisabled.value = false
}
isFetchingInitialData.value = false isFetchingInitialData.value = false
}) })
} }

View File

@ -20,21 +20,6 @@
:data="fetchData" :data="fetchData"
:columns="taxTypeColumns" :columns="taxTypeColumns"
> >
<template #cell-compound_tax="{ row }">
<BaseBadge
:bg-color="
utils.getBadgeStatusColor(row.data.compound_tax ? 'YES' : 'NO')
.bgColor
"
:color="
utils.getBadgeStatusColor(row.data.compound_tax ? 'YES' : 'NO')
.color
"
>
{{ row.data.compound_tax ? 'Yes' : 'No'.replace('_', ' ') }}
</BaseBadge>
</template>
<template #cell-percent="{ row }"> {{ row.data.percent }} % </template> <template #cell-percent="{ row }"> {{ row.data.percent }} % </template>
<template v-if="hasAtleastOneAbility()" #cell-actions="{ row }"> <template v-if="hasAtleastOneAbility()" #cell-actions="{ row }">
@ -91,11 +76,6 @@ const taxTypeColumns = computed(() => {
thClass: 'extra', thClass: 'extra',
tdClass: 'font-medium text-gray-900', tdClass: 'font-medium text-gray-900',
}, },
{
key: 'compound_tax',
label: t('settings.tax_types.compound_tax'),
tdClass: 'font-medium text-gray-900',
},
{ {
key: 'percent', key: 'percent',
label: t('settings.tax_types.percent'), label: t('settings.tax_types.percent'),

View File

@ -1,101 +0,0 @@
<!-- This example requires Tailwind CSS v2.0+ -->
<script lang="ts" setup>
import { Switch, SwitchGroup, SwitchLabel } from '@headlessui/vue'
import { useGlobalStore } from '@/scripts/admin/stores/global'
import { computed, ref } from 'vue'
defineProps({
showLabel: {
type: Boolean,
default: true,
},
vertical: {
type: Boolean,
default: false,
},
})
const globalStore = useGlobalStore()
const enabled = ref(
localStorage.getItem('theme') === 'dark' ||
document.documentElement.classList.contains('dark')
)
globalStore.isDarkModeOn = enabled
function onChange(val) {
if (val) {
localStorage.theme = 'dark'
document.documentElement.classList.add('dark')
document.documentElement.style.setProperty('color-scheme', 'dark')
globalStore.isDarkModeOn = true
} else {
localStorage.theme = 'light'
document.documentElement.classList.remove('dark')
document.documentElement.style.setProperty('color-scheme', 'light')
globalStore.isDarkModeOn = false
}
}
</script>
<template>
<div class="w-full flex justify-center">
<SwitchGroup
as="div"
class="flex items-center"
:class="vertical ? 'flex-col justify-center' : 'flex-row'"
>
<Switch
v-model="enabled"
class="relative inline-flex flex-shrink-0 h-6 w-11 border-2 border-transparent rounded-full cursor-pointer transition-colors ease-in-out duration-200 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-primary-500 dark:ring-offset-gray-700"
:class="[enabled ? 'bg-primary-600' : 'bg-gray-200']"
@update:modelValue="onChange"
>
<span class="sr-only">Use setting</span>
<span
class="pointer-events-none relative inline-block h-5 w-5 rounded-full bg-white shadow transform ring-0 transition ease-in-out duration-200"
:class="[enabled ? 'translate-x-5' : 'translate-x-0']"
>
<span
class="absolute inset-0 h-full w-full flex items-center justify-center transition-opacity"
:class="[
enabled
? 'opacity-0 ease-out duration-100'
: 'opacity-100 ease-in duration-200',
]"
aria-hidden="true"
>
<BaseIcon class="h-3 w-3 text-yellow-500" name="SunIcon" />
</span>
<span
class="absolute inset-0 h-full w-full flex items-center justify-center transition-opacity"
:class="[
enabled
? 'opacity-100 ease-in duration-200'
: 'opacity-0 ease-out duration-100',
]"
aria-hidden="true"
>
<BaseIcon class="h-3 w-3 text-primary-500" name="MoonIcon" />
</span>
</span>
</Switch>
<SwitchLabel
v-if="showLabel"
as="span"
class="cursor-pointer"
:class="vertical ? 'px-1 text-center mt-2' : 'ml-3'"
>
<span
v-if="enabled"
class="text-sm font-medium text-gray-500 dark:text-gray-400"
>
Dark Mode
</span>
<span v-else class="text-sm font-medium text-gray-500">
Light Mode
</span>
</SwitchLabel>
</SwitchGroup>
</div>
</template>

View File

@ -126,7 +126,7 @@ onMounted(() => {
}) })
const value = computed({ const value = computed({
get: () => (props.modelValue ? props.modelValue : ''), get: () => props.modelValue,
set: (value) => { set: (value) => {
emit('update:modelValue', value) emit('update:modelValue', value)
}, },
@ -195,9 +195,7 @@ async function getFields() {
{ label: 'Date', value: 'INVOICE_DATE' }, { label: 'Date', value: 'INVOICE_DATE' },
{ label: 'Due Date', value: 'INVOICE_DUE_DATE' }, { label: 'Due Date', value: 'INVOICE_DUE_DATE' },
{ label: 'Number', value: 'INVOICE_NUMBER' }, { label: 'Number', value: 'INVOICE_NUMBER' },
{ label: 'PDF Link', value: 'PDF_LINK' }, { label: 'Ref Number', value: 'INVOICE_REF_NUMBER' },
{ label: 'Due Amount', value: 'DUE_AMOUNT' },
{ label: 'Total Amount', value: 'TOTAL_AMOUNT' },
...invoiceFields.value.map((i) => ({ ...invoiceFields.value.map((i) => ({
label: i.label, label: i.label,
value: i.slug, value: i.slug,
@ -213,8 +211,7 @@ async function getFields() {
{ label: 'Date', value: 'ESTIMATE_DATE' }, { label: 'Date', value: 'ESTIMATE_DATE' },
{ label: 'Expiry Date', value: 'ESTIMATE_EXPIRY_DATE' }, { label: 'Expiry Date', value: 'ESTIMATE_EXPIRY_DATE' },
{ label: 'Number', value: 'ESTIMATE_NUMBER' }, { label: 'Number', value: 'ESTIMATE_NUMBER' },
{ label: 'PDF Link', value: 'PDF_LINK' }, { label: 'Ref Number', value: 'ESTIMATE_REF_NUMBER' },
{ label: 'Total Amount', value: 'TOTAL_AMOUNT' },
...estimateFields.value.map((i) => ({ ...estimateFields.value.map((i) => ({
label: i.label, label: i.label,
value: i.slug, value: i.slug,
@ -231,7 +228,6 @@ async function getFields() {
{ label: 'Number', value: 'PAYMENT_NUMBER' }, { label: 'Number', value: 'PAYMENT_NUMBER' },
{ label: 'Mode', value: 'PAYMENT_MODE' }, { label: 'Mode', value: 'PAYMENT_MODE' },
{ label: 'Amount', value: 'PAYMENT_AMOUNT' }, { label: 'Amount', value: 'PAYMENT_AMOUNT' },
{ label: 'PDF Link', value: 'PDF_LINK' },
...paymentFields.value.map((i) => ({ ...paymentFields.value.map((i) => ({
label: i.label, label: i.label,
value: i.slug, value: i.slug,

View File

@ -7,108 +7,52 @@
/> />
</BaseContentPlaceholders> </BaseContentPlaceholders>
<div v-else :class="computedContainerClass"> <div v-else :class="computedContainerClass" class="relative flex flex-row">
<date-picker <svg
ref="vCalendar" v-if="showCalendarIcon && !hasIconSlot"
v-model="date" viewBox="0 0 20 20"
:mode="mode" fill="currentColor"
:is24hr="time24hr" class="
class="w-full" absolute
color="indigo" w-4
:input-debounce="500" h-4
:update-on-input="false" mx-2
:is-range="false" my-2.5
trim-weeks text-sm
:is-required="isRequired" not-italic
:popover="{ font-black
visibility: disabled ? 'hidden' : 'focus', text-gray-400
showDelay: 0, cursor-pointer
hideDelay: 1, "
}" @click="onClickDp"
:attributes="attrs"
:model-config="config"
:masks="masks"
:locale="global.locale"
> >
<template <path
#default="{ inputValue, inputEvents, togglePopover, hidePopover }" fill-rule="evenodd"
> d="M6 2a1 1 0 00-1 1v1H4a2 2 0 00-2 2v10a2 2 0 002 2h12a2 2 0 002-2V6a2 2 0 00-2-2h-1V3a1 1 0 10-2 0v1H7V3a1 1 0 00-1-1zm0 5a1 1 0 000 2h8a1 1 0 100-2H6z"
<!-- calendar icon --> clip-rule="evenodd"
<svg ></path>
v-if="showCalendarIcon && !hasIconSlot" </svg>
viewBox="0 0 20 20"
fill="currentColor"
class="
absolute
w-4
h-4
mx-2
my-2.5
text-sm
not-italic
font-black
text-gray-400
cursor-pointer
"
@click="togglePopover()"
>
<path
fill-rule="evenodd"
d="M6 2a1 1 0 00-1 1v1H4a2 2 0 00-2 2v10a2 2 0 002 2h12a2 2 0 002-2V6a2 2 0 00-2-2h-1V3a1 1 0 10-2 0v1H7V3a1 1 0 00-1-1zm0 5a1 1 0 000 2h8a1 1 0 100-2H6z"
clip-rule="evenodd"
></path>
</svg>
<slot v-if="showCalendarIcon && hasIconSlot" name="icon" /> <slot v-if="showCalendarIcon && hasIconSlot" name="icon" />
<input <FlatPickr
:value="inputValue" ref="dp"
:class="[defaultInputClass, inputInvalidClass, inputDisabledClass]" v-model="date"
readonly v-bind="$attrs"
v-on="inputEvents" :disabled="disabled"
@blur="hidePopover()" :config="config"
/> :class="[defaultInputClass, inputInvalidClass, inputDisabledClass]"
</template> />
<template v-if="showExtraOptions" #footer>
<div
class="bg-gray-100 grid grid-cols-3 gap-2 p-2 border-t rounded-b-lg"
>
<button type="button" class="extra-button" @click="moveToDate(sourceDate)">
{{ global.t('date_picker.same_day') }}
</button>
<button type="button" class="extra-button" @click="withInDays(7)">
{{ global.t('date_picker.within_7_days') }}
</button>
<button type="button" class="extra-button" @click="withInDays(15)">
{{ global.t('date_picker.within_15_days') }}
</button>
<button type="button" class="extra-button" @click="withInDays(30)">
{{ global.t('date_picker.within_30_days') }}
</button>
<button type="button" class="extra-button" @click="withInDays(45)">
{{ global.t('date_picker.within_45_days') }}
</button>
<button type="button" class="extra-button" @click="withInDays(60)">
{{ global.t('date_picker.within_60_days') }}
</button>
</div>
</template>
</date-picker>
</div> </div>
</template> </template>
<script type="text/babel" setup> <script type="text/babel" setup>
import { Calendar, DatePicker } from 'v-calendar' import FlatPickr from 'vue-flatpickr-component'
import 'v-calendar/dist/style.css' import 'flatpickr/dist/flatpickr.css'
import { computed, reactive, watch, ref, useSlots } from 'vue' import { computed, reactive, watch, ref, useSlots } from 'vue'
import { useCompanyStore } from '@/scripts/admin/stores/company' import { useCompanyStore } from '@/scripts/admin/stores/company'
import moment from 'moment'
const dp = ref(null)
const props = defineProps({ const props = defineProps({
modelValue: { modelValue: {
@ -146,31 +90,36 @@ const props = defineProps({
defaultInputClass: { defaultInputClass: {
type: String, type: String,
default: default:
'border-2 font-base pl-8 py-2 outline-none focus:ring-primary-400 focus:outline-none focus:border-primary-400 block w-full sm:text-sm border-gray-200 rounded-md text-black', 'font-base pl-8 py-2 outline-none focus:ring-primary-400 focus:outline-none focus:border-primary-400 block w-full sm:text-sm border-gray-200 rounded-md text-black',
}, },
time24hr: { time24hr: {
type: Boolean, type: Boolean,
default: false, default: false,
}, },
isRequired: {
type: Boolean,
default: false,
},
showExtraOptions: {
type: Boolean,
default: false,
},
sourceDate: {
type: [String, Date],
default: () => new Date(),
}
}) })
const emit = defineEmits(['update:modelValue']) const emit = defineEmits(['update:modelValue'])
const slots = useSlots() const slots = useSlots()
const companyStore = useCompanyStore() const companyStore = useCompanyStore()
const { global } = window.i18n
const vCalendar = ref(null) let config = reactive({
altInput: true,
enableTime: props.enableTime,
time_24hr: props.time24hr,
})
const date = computed({
get: () => props.modelValue,
set: (value) => {
emit('update:modelValue', value)
},
})
const carbonFormat = computed(() => {
return companyStore.selectedCompanySettings?.carbon_date_format
})
const hasIconSlot = computed(() => { const hasIconSlot = computed(() => {
return !!slots.icon return !!slots.icon
@ -186,6 +135,7 @@ const inputInvalidClass = computed(() => {
if (props.invalid) { if (props.invalid) {
return 'border-red-400 ring-red-400 focus:ring-red-400 focus:border-red-400' return 'border-red-400 ring-red-400 focus:ring-red-400 focus:border-red-400'
} }
return '' return ''
}) })
@ -193,97 +143,35 @@ const inputDisabledClass = computed(() => {
if (props.disabled) { if (props.disabled) {
return 'border border-solid rounded-md outline-none input-field box-border-2 base-date-picker-input placeholder-gray-400 bg-gray-200 text-gray-600 border-gray-200' return 'border border-solid rounded-md outline-none input-field box-border-2 base-date-picker-input placeholder-gray-400 bg-gray-200 text-gray-600 border-gray-200'
} }
return '' return ''
}) })
// to convert YYYY-MM-DD | YYYY-MM-DD HH:mm format function onClickDp(params) {
function convertYMDFormat(date) { dp.value.fp.open()
let format = props.enableTime ? 'YYYY-MM-DD HH:mm' : 'YYYY-MM-DD'
return date ? moment(date).format(format) : date
} }
const date = computed({
get: () => props.modelValue,
set: (value) => {
emit('update:modelValue', value)
},
})
const mode = computed(() => {
return props.enableTime ? 'dateTime' : 'date'
})
const config = reactive({
type: 'string',
mask: 'YYYY-MM-DD', // Uses 'iso' if missing
//timeAdjust: '00:00:00',
})
const masks = reactive({
input: null,
inputDateTime: null,
inputDateTime24hr: null,
})
const attrs = reactive([
{
dates: new Date(),
highlight: {
fillMode: 'outline',
},
/* popover: {
label: 'Today Date',
visibility: 'hover',
}, */
},
])
const carbonFormat = computed(() => {
return companyStore.selectedCompanySettings?.moment_date_format
})
watch( watch(
() => carbonFormat, () => props.enableTime,
() => { (val) => {
if (!props.enableTime) { if (props.enableTime) {
masks.input = carbonFormat.value ? carbonFormat.value : 'DD MMM YYYY' config.enableTime = props.enableTime
config.mask = 'YYYY-MM-DD'
} else {
let timeFormat = 'HH:mm'
if (props.time24hr) {
masks.inputDateTime24hr = carbonFormat.value
? `${carbonFormat.value} ${timeFormat}`
: `DD MMM YYYY ${timeFormat}`
} else {
masks.inputDateTime = carbonFormat.value
? `${carbonFormat.value} ${timeFormat}`
: `DD MMM YYYY ${timeFormat}`
}
config.mask = `YYYY-MM-DD ${timeFormat}`
} }
}, },
{ immediate: true } { immediate: true }
) )
async function moveToDate(_date) { watch(
const calendar = vCalendar.value () => carbonFormat,
_date = _date ? _date : convertYMDFormat(new Date()) () => {
date.value = _date if (!props.enableTime) {
// await calendar.move(_date) config.altFormat = carbonFormat.value ? carbonFormat.value : 'd M Y'
calendar.hidePopover() } else {
} config.altFormat = carbonFormat.value
? `${carbonFormat.value} H:i `
async function withInDays(noOfDays) { : 'd M Y H:i'
if (!noOfDays) return false }
},
let newDate = moment(props.sourceDate).add(noOfDays, 'days').toDate() { immediate: true }
newDate = convertYMDFormat(newDate) )
moveToDate(newDate)
}
</script> </script>
<style scoped>
.extra-button {
@apply bg-primary-500 text-white text-sm font-semibold px-2 py-1 rounded hover:bg-primary-700;
}
</style>

View File

@ -1,10 +1,6 @@
<template> <template>
<div> <div>
<TabGroup <TabGroup :default-index="defaultIndex" @change="onChange">
:selected-index="selectedIndex"
:default-index="defaultIndex"
@change="onChange"
>
<TabList <TabList
:class="[ :class="[
'flex border-b border-grey-light', 'flex border-b border-grey-light',
@ -58,10 +54,6 @@ const props = defineProps({
type: Number, type: Number,
default: 0, default: 0,
}, },
selectedIndex: {
type: Number,
default: 0,
},
filter: { filter: {
type: String, type: String,
default: null, default: null,
@ -75,6 +67,6 @@ const slots = useSlots()
const tabs = computed(() => slots.default().map((tab) => tab.props)) const tabs = computed(() => slots.default().map((tab) => tab.props))
function onChange(d) { function onChange(d) {
emit('change', tabs.value[d], d) emit('change', tabs.value[d])
} }
</script> </script>

View File

@ -309,6 +309,8 @@ function changeSorting(column) {
} }
if (!usesLocalData.value) { if (!usesLocalData.value) {
if (pagination.value)
pagination.value.currentPage = 1
mapDataToRows() mapDataToRows()
} }
} }
@ -326,7 +328,10 @@ async function pageChange(page) {
await mapDataToRows() await mapDataToRows()
} }
async function refresh() { async function refresh(isPreservePage = false) {
if (pagination.value && !isPreservePage)
pagination.value.currentPage = 1
await mapDataToRows() await mapDataToRows()
} }

View File

@ -3,26 +3,23 @@
width="50" width="50"
height="50" height="50"
viewBox="0 0 50 50" viewBox="0 0 50 50"
:class="colorClass"
fill="none" fill="none"
xmlns="http://www.w3.org/2000/svg" xmlns="http://www.w3.org/2000/svg"
> >
<circle cx="25" cy="25" r="25" :class="bgColor" /> <circle cx="25" cy="25" r="25" fill="#EAF1FB" />
<path <path
d="M28.2656 23.0547C27.3021 24.0182 26.1302 24.5 24.75 24.5C23.3698 24.5 22.1849 24.0182 21.1953 23.0547C20.2318 22.0651 19.75 20.8802 19.75 19.5C19.75 18.1198 20.2318 16.9479 21.1953 15.9844C22.1849 14.9948 23.3698 14.5 24.75 14.5C26.1302 14.5 27.3021 14.9948 28.2656 15.9844C29.2552 16.9479 29.75 18.1198 29.75 19.5C29.75 20.8802 29.2552 22.0651 28.2656 23.0547ZM28.2656 25.75C29.6979 25.75 30.9219 26.2708 31.9375 27.3125C32.9792 28.3281 33.5 29.5521 33.5 30.9844V32.625C33.5 33.1458 33.3177 33.5885 32.9531 33.9531C32.5885 34.3177 32.1458 34.5 31.625 34.5H17.875C17.3542 34.5 16.9115 34.3177 16.5469 33.9531C16.1823 33.5885 16 33.1458 16 32.625V30.9844C16 29.5521 16.5078 28.3281 17.5234 27.3125C18.5651 26.2708 19.8021 25.75 21.2344 25.75H21.8984C22.8099 26.1667 23.7604 26.375 24.75 26.375C25.7396 26.375 26.6901 26.1667 27.6016 25.75H28.2656Z" d="M28.2656 23.0547C27.3021 24.0182 26.1302 24.5 24.75 24.5C23.3698 24.5 22.1849 24.0182 21.1953 23.0547C20.2318 22.0651 19.75 20.8802 19.75 19.5C19.75 18.1198 20.2318 16.9479 21.1953 15.9844C22.1849 14.9948 23.3698 14.5 24.75 14.5C26.1302 14.5 27.3021 14.9948 28.2656 15.9844C29.2552 16.9479 29.75 18.1198 29.75 19.5C29.75 20.8802 29.2552 22.0651 28.2656 23.0547ZM28.2656 25.75C29.6979 25.75 30.9219 26.2708 31.9375 27.3125C32.9792 28.3281 33.5 29.5521 33.5 30.9844V32.625C33.5 33.1458 33.3177 33.5885 32.9531 33.9531C32.5885 34.3177 32.1458 34.5 31.625 34.5H17.875C17.3542 34.5 16.9115 34.3177 16.5469 33.9531C16.1823 33.5885 16 33.1458 16 32.625V30.9844C16 29.5521 16.5078 28.3281 17.5234 27.3125C18.5651 26.2708 19.8021 25.75 21.2344 25.75H21.8984C22.8099 26.1667 23.7604 26.375 24.75 26.375C25.7396 26.375 26.6901 26.1667 27.6016 25.75H28.2656Z"
:class="color" fill="currentColor"
/> />
</svg> </svg>
</template> </template>
<script setup> <script setup>
const props = defineProps({ const props = defineProps({
color: { colorClass: {
type: String, type: String,
default: 'fill-primary-500 dark:fill-white', default: 'text-primary-500',
},
bgColor: {
type: String,
default: 'fill-gray-100 dark:fill-primary-400',
}, },
}) })
</script> </script>

View File

@ -6,24 +6,10 @@
fill="none" fill="none"
xmlns="http://www.w3.org/2000/svg" xmlns="http://www.w3.org/2000/svg"
> >
<circle :class="bgColor" cx="25" cy="25" r="25" /> <circle cx="25" cy="25" r="25" fill="#FDE4E5" />
<path <path
:class="color"
d="M27.2031 23.6016C28.349 23.9401 29.2083 24.6562 29.7812 25.75C30.3802 26.8438 30.4714 27.9766 30.0547 29.1484C29.7422 30.0078 29.2083 30.6979 28.4531 31.2188C27.6979 31.7135 26.8516 31.974 25.9141 32V33.875C25.9141 34.0573 25.849 34.2005 25.7188 34.3047C25.6146 34.4349 25.4714 34.5 25.2891 34.5H24.0391C23.8568 34.5 23.7005 34.4349 23.5703 34.3047C23.4661 34.2005 23.4141 34.0573 23.4141 33.875V32C22.1641 32 21.0443 31.6094 20.0547 30.8281C19.8984 30.6979 19.8073 30.5417 19.7812 30.3594C19.7552 30.1771 19.8203 30.0208 19.9766 29.8906L21.3047 28.5625C21.5651 28.3281 21.8255 28.3021 22.0859 28.4844C22.4766 28.7448 22.9193 28.875 23.4141 28.875H25.9922C26.3307 28.875 26.6042 28.7708 26.8125 28.5625C27.0469 28.3281 27.1641 28.0417 27.1641 27.7031C27.1641 27.1302 26.8906 26.7656 26.3438 26.6094L22.3203 25.4375C21.4349 25.1771 20.6927 24.7083 20.0938 24.0312C19.4948 23.3542 19.1432 22.5729 19.0391 21.6875C18.9349 20.4115 19.2995 19.3177 20.1328 18.4062C20.9922 17.4688 22.0599 17 23.3359 17H23.4141V15.125C23.4141 14.9427 23.4661 14.7995 23.5703 14.6953C23.7005 14.5651 23.8568 14.5 24.0391 14.5H25.2891C25.4714 14.5 25.6146 14.5651 25.7188 14.6953C25.849 14.7995 25.9141 14.9427 25.9141 15.125V17C27.1641 17 28.2839 17.3906 29.2734 18.1719C29.4297 18.3021 29.5208 18.4583 29.5469 18.6406C29.5729 18.8229 29.5078 18.9792 29.3516 19.1094L28.0234 20.4375C27.763 20.6719 27.5026 20.6979 27.2422 20.5156C26.8516 20.2552 26.4089 20.125 25.9141 20.125H23.3359C22.9974 20.125 22.7109 20.2422 22.4766 20.4766C22.2682 20.6849 22.1641 20.9583 22.1641 21.2969C22.1641 21.5312 22.2422 21.7526 22.3984 21.9609C22.5547 22.1693 22.75 22.3125 22.9844 22.3906L27.2031 23.6016Z" d="M27.2031 23.6016C28.349 23.9401 29.2083 24.6562 29.7812 25.75C30.3802 26.8438 30.4714 27.9766 30.0547 29.1484C29.7422 30.0078 29.2083 30.6979 28.4531 31.2188C27.6979 31.7135 26.8516 31.974 25.9141 32V33.875C25.9141 34.0573 25.849 34.2005 25.7188 34.3047C25.6146 34.4349 25.4714 34.5 25.2891 34.5H24.0391C23.8568 34.5 23.7005 34.4349 23.5703 34.3047C23.4661 34.2005 23.4141 34.0573 23.4141 33.875V32C22.1641 32 21.0443 31.6094 20.0547 30.8281C19.8984 30.6979 19.8073 30.5417 19.7812 30.3594C19.7552 30.1771 19.8203 30.0208 19.9766 29.8906L21.3047 28.5625C21.5651 28.3281 21.8255 28.3021 22.0859 28.4844C22.4766 28.7448 22.9193 28.875 23.4141 28.875H25.9922C26.3307 28.875 26.6042 28.7708 26.8125 28.5625C27.0469 28.3281 27.1641 28.0417 27.1641 27.7031C27.1641 27.1302 26.8906 26.7656 26.3438 26.6094L22.3203 25.4375C21.4349 25.1771 20.6927 24.7083 20.0938 24.0312C19.4948 23.3542 19.1432 22.5729 19.0391 21.6875C18.9349 20.4115 19.2995 19.3177 20.1328 18.4062C20.9922 17.4688 22.0599 17 23.3359 17H23.4141V15.125C23.4141 14.9427 23.4661 14.7995 23.5703 14.6953C23.7005 14.5651 23.8568 14.5 24.0391 14.5H25.2891C25.4714 14.5 25.6146 14.5651 25.7188 14.6953C25.849 14.7995 25.9141 14.9427 25.9141 15.125V17C27.1641 17 28.2839 17.3906 29.2734 18.1719C29.4297 18.3021 29.5208 18.4583 29.5469 18.6406C29.5729 18.8229 29.5078 18.9792 29.3516 19.1094L28.0234 20.4375C27.763 20.6719 27.5026 20.6979 27.2422 20.5156C26.8516 20.2552 26.4089 20.125 25.9141 20.125H23.3359C22.9974 20.125 22.7109 20.2422 22.4766 20.4766C22.2682 20.6849 22.1641 20.9583 22.1641 21.2969C22.1641 21.5312 22.2422 21.7526 22.3984 21.9609C22.5547 22.1693 22.75 22.3125 22.9844 22.3906L27.2031 23.6016Z"
fill="#FB7178" fill="#FB7178"
/> />
</svg> </svg>
</template> </template>
<script setup>
const props = defineProps({
color: {
type: String,
default: 'fill-red-400 dark:fill-white',
},
bgColor: {
type: String,
default: 'fill-red-100 dark:fill-red-400',
},
})
</script>

View File

@ -5,24 +5,21 @@
viewBox="0 0 50 50" viewBox="0 0 50 50"
fill="none" fill="none"
xmlns="http://www.w3.org/2000/svg" xmlns="http://www.w3.org/2000/svg"
:class="colorClass"
> >
<circle cx="25" cy="25" r="25" :class="bgColor" /> <circle cx="25" cy="25" r="25" fill="#EAF1FB" />
<path <path
d="M26.75 19.8125C26.75 20.0729 26.8411 20.2943 27.0234 20.4766C27.2057 20.6589 27.4271 20.75 27.6875 20.75H33V33.5625C33 33.8229 32.9089 34.0443 32.7266 34.2266C32.5443 34.4089 32.3229 34.5 32.0625 34.5H18.9375C18.6771 34.5 18.4557 34.4089 18.2734 34.2266C18.0911 34.0443 18 33.8229 18 33.5625V15.4375C18 15.1771 18.0911 14.9557 18.2734 14.7734C18.4557 14.5911 18.6771 14.5 18.9375 14.5H26.75V19.8125ZM33 19.2656V19.5H28V14.5H28.2344C28.4948 14.5 28.7161 14.5911 28.8984 14.7734L32.7266 18.6016C32.9089 18.7839 33 19.0052 33 19.2656Z" d="M26.75 19.8125C26.75 20.0729 26.8411 20.2943 27.0234 20.4766C27.2057 20.6589 27.4271 20.75 27.6875 20.75H33V33.5625C33 33.8229 32.9089 34.0443 32.7266 34.2266C32.5443 34.4089 32.3229 34.5 32.0625 34.5H18.9375C18.6771 34.5 18.4557 34.4089 18.2734 34.2266C18.0911 34.0443 18 33.8229 18 33.5625V15.4375C18 15.1771 18.0911 14.9557 18.2734 14.7734C18.4557 14.5911 18.6771 14.5 18.9375 14.5H26.75V19.8125ZM33 19.2656V19.5H28V14.5H28.2344C28.4948 14.5 28.7161 14.5911 28.8984 14.7734L32.7266 18.6016C32.9089 18.7839 33 19.0052 33 19.2656Z"
:class="color" fill="currentColor"
/> />
</svg> </svg>
</template> </template>
<script setup> <script setup>
const props = defineProps({ const props = defineProps({
color: { colorClass: {
type: String, type: String,
default: 'fill-primary-500 dark:fill-white', default: 'text-primary-500',
},
bgColor: {
type: String,
default: 'fill-gray-100 dark:fill-primary-400',
}, },
}) })
</script> </script>

View File

@ -5,24 +5,21 @@
viewBox="0 0 50 50" viewBox="0 0 50 50"
fill="none" fill="none"
xmlns="http://www.w3.org/2000/svg" xmlns="http://www.w3.org/2000/svg"
:class="colorClass"
> >
<circle cx="25" cy="25" r="25" :class="bgColor" /> <circle cx="25" cy="25" r="25" fill="#EAF1FB" />
<path <path
d="M28.25 24.5V27H20.75V24.5H28.25ZM31.7266 18.6016C31.9089 18.7839 32 19.0052 32 19.2656V19.5H27V14.5H27.2344C27.4948 14.5 27.7161 14.5911 27.8984 14.7734L31.7266 18.6016ZM25.75 19.8125C25.75 20.0729 25.8411 20.2943 26.0234 20.4766C26.2057 20.6589 26.4271 20.75 26.6875 20.75H32V33.5625C32 33.8229 31.9089 34.0443 31.7266 34.2266C31.5443 34.4089 31.3229 34.5 31.0625 34.5H17.9375C17.6771 34.5 17.4557 34.4089 17.2734 34.2266C17.0911 34.0443 17 33.8229 17 33.5625V15.4375C17 15.1771 17.0911 14.9557 17.2734 14.7734C17.4557 14.5911 17.6771 14.5 17.9375 14.5H25.75V19.8125ZM19.5 17.3125V17.9375C19.5 18.1458 19.6042 18.25 19.8125 18.25H22.9375C23.1458 18.25 23.25 18.1458 23.25 17.9375V17.3125C23.25 17.1042 23.1458 17 22.9375 17H19.8125C19.6042 17 19.5 17.1042 19.5 17.3125ZM19.5 19.8125V20.4375C19.5 20.6458 19.6042 20.75 19.8125 20.75H22.9375C23.1458 20.75 23.25 20.6458 23.25 20.4375V19.8125C23.25 19.6042 23.1458 19.5 22.9375 19.5H19.8125C19.6042 19.5 19.5 19.6042 19.5 19.8125ZM29.5 31.6875V31.0625C29.5 30.8542 29.3958 30.75 29.1875 30.75H26.0625C25.8542 30.75 25.75 30.8542 25.75 31.0625V31.6875C25.75 31.8958 25.8542 32 26.0625 32H29.1875C29.3958 32 29.5 31.8958 29.5 31.6875ZM29.5 23.875C29.5 23.6927 29.4349 23.5495 29.3047 23.4453C29.2005 23.3151 29.0573 23.25 28.875 23.25H20.125C19.9427 23.25 19.7865 23.3151 19.6562 23.4453C19.5521 23.5495 19.5 23.6927 19.5 23.875V27.625C19.5 27.8073 19.5521 27.9635 19.6562 28.0938C19.7865 28.1979 19.9427 28.25 20.125 28.25H28.875C29.0573 28.25 29.2005 28.1979 29.3047 28.0938C29.4349 27.9635 29.5 27.8073 29.5 27.625V23.875Z" d="M28.25 24.5V27H20.75V24.5H28.25ZM31.7266 18.6016C31.9089 18.7839 32 19.0052 32 19.2656V19.5H27V14.5H27.2344C27.4948 14.5 27.7161 14.5911 27.8984 14.7734L31.7266 18.6016ZM25.75 19.8125C25.75 20.0729 25.8411 20.2943 26.0234 20.4766C26.2057 20.6589 26.4271 20.75 26.6875 20.75H32V33.5625C32 33.8229 31.9089 34.0443 31.7266 34.2266C31.5443 34.4089 31.3229 34.5 31.0625 34.5H17.9375C17.6771 34.5 17.4557 34.4089 17.2734 34.2266C17.0911 34.0443 17 33.8229 17 33.5625V15.4375C17 15.1771 17.0911 14.9557 17.2734 14.7734C17.4557 14.5911 17.6771 14.5 17.9375 14.5H25.75V19.8125ZM19.5 17.3125V17.9375C19.5 18.1458 19.6042 18.25 19.8125 18.25H22.9375C23.1458 18.25 23.25 18.1458 23.25 17.9375V17.3125C23.25 17.1042 23.1458 17 22.9375 17H19.8125C19.6042 17 19.5 17.1042 19.5 17.3125ZM19.5 19.8125V20.4375C19.5 20.6458 19.6042 20.75 19.8125 20.75H22.9375C23.1458 20.75 23.25 20.6458 23.25 20.4375V19.8125C23.25 19.6042 23.1458 19.5 22.9375 19.5H19.8125C19.6042 19.5 19.5 19.6042 19.5 19.8125ZM29.5 31.6875V31.0625C29.5 30.8542 29.3958 30.75 29.1875 30.75H26.0625C25.8542 30.75 25.75 30.8542 25.75 31.0625V31.6875C25.75 31.8958 25.8542 32 26.0625 32H29.1875C29.3958 32 29.5 31.8958 29.5 31.6875ZM29.5 23.875C29.5 23.6927 29.4349 23.5495 29.3047 23.4453C29.2005 23.3151 29.0573 23.25 28.875 23.25H20.125C19.9427 23.25 19.7865 23.3151 19.6562 23.4453C19.5521 23.5495 19.5 23.6927 19.5 23.875V27.625C19.5 27.8073 19.5521 27.9635 19.6562 28.0938C19.7865 28.1979 19.9427 28.25 20.125 28.25H28.875C29.0573 28.25 29.2005 28.1979 29.3047 28.0938C29.4349 27.9635 29.5 27.8073 29.5 27.625V23.875Z"
:class="color" fill="currentColor"
/> />
</svg> </svg>
</template> </template>
<script setup> <script setup>
const props = defineProps({ const props = defineProps({
color: { colorClass: {
type: String, type: String,
default: 'fill-primary-500 dark:fill-white', default: 'text-primary-500',
},
bgColor: {
type: String,
default: 'fill-gray-100 dark:fill-primary-400',
}, },
}) })
</script> </script>

View File

@ -7,12 +7,12 @@ export function usePopper(options) {
let popper = ref(null) let popper = ref(null)
onMounted(() => { onMounted(() => {
watchEffect((onInvalidate) => { watchEffect(onInvalidate => {
if (!container.value) return if (!container.value) return
if (!activator.value) return if (!activator.value) return
let containerEl = container.value.el || container.value let containerEl = container.value.el || container.value
let activatorEl = activator.value.$el || activator.value let activatorEl = activator.value.el || activator.value
if (!(activatorEl instanceof HTMLElement)) return if (!(activatorEl instanceof HTMLElement)) return
if (!(containerEl instanceof HTMLElement)) return if (!(containerEl instanceof HTMLElement)) return

View File

@ -863,8 +863,6 @@
"company_info": { "company_info": {
"company_info": "Company info", "company_info": "Company info",
"company_name": "Company Name", "company_name": "Company Name",
"company_slug": "Company Slug",
"company_slug_help_text": "A unique URL friendly name for your company (It will appear on Customer Portal URL)",
"company_logo": "Company Logo", "company_logo": "Company Logo",
"section_description": "Information about your company that will be displayed on invoices, estimates and other documents created by Crater.", "section_description": "Information about your company that will be displayed on invoices, estimates and other documents created by Crater.",
"phone": "Phone", "phone": "Phone",
@ -1326,8 +1324,6 @@
"company_info": "Company Information", "company_info": "Company Information",
"company_info_desc": "This information will be displayed on invoices. Note that you can edit this later on settings page.", "company_info_desc": "This information will be displayed on invoices. Note that you can edit this later on settings page.",
"company_name": "Company Name", "company_name": "Company Name",
"company_slug": "Company Slug",
"company_slug_help_text": "A unique URL friendly name for your company (It will appear on Customer Portal URL)",
"company_logo": "Company Logo", "company_logo": "Company Logo",
"logo_preview": "Logo Preview", "logo_preview": "Logo Preview",
"preferences": "Company Preferences", "preferences": "Company Preferences",
@ -1458,8 +1454,7 @@
"at_least_one_ability": "Please select atleast one Permission.", "at_least_one_ability": "Please select atleast one Permission.",
"valid_driver_key": "Please enter a valid {driver} key.", "valid_driver_key": "Please enter a valid {driver} key.",
"valid_exchange_rate": "Please enter a valid exchange rate.", "valid_exchange_rate": "Please enter a valid exchange rate.",
"company_name_not_same": "Company name must match with given name.", "company_name_not_same": "Company name must match with given name."
"invalid_slug": "Invalid Slug"
}, },
"errors": { "errors": {
"starter_plan": "This feature is available on Starter plan and onwards!", "starter_plan": "This feature is available on Starter plan and onwards!",
@ -1527,13 +1522,5 @@
"pdf_bill_to": "Bill to,", "pdf_bill_to": "Bill to,",
"pdf_ship_to": "Ship to,", "pdf_ship_to": "Ship to,",
"pdf_received_from": "Received from:", "pdf_received_from": "Received from:",
"pdf_tax_label": "Tax", "pdf_tax_label": "Tax"
"date_picker": {
"same_day": "Same Day",
"within_7_days": "Within 7 Days",
"within_15_days": "Within 15 Days",
"within_30_days": "Within 30 Days",
"within_45_days": "Within 45 Days",
"within_60_days": "Within 60 Days"
}
} }

View File

@ -17,7 +17,7 @@
<meta name="csrf-token" content="{{ csrf_token() }}"> <meta name="csrf-token" content="{{ csrf_token() }}">
<!-- Module Styles --> <!-- Module Styles -->
@foreach (\Crater\Services\Module\ModuleFacade::allStyles() as $name => $path) @foreach(\Crater\Services\Module\ModuleFacade::allStyles() as $name => $path)
<link rel="stylesheet" href="/modules/styles/{{ $name }}"> <link rel="stylesheet" href="/modules/styles/{{ $name }}">
@endforeach @endforeach
@ -25,8 +25,8 @@
</head> </head>
<body <body
class="h-full overflow-hidden bg-gray-100 dark:bg-gray-900 dark:text-white font-base class="h-full overflow-hidden bg-gray-100 font-base
@if (isset($current_theme)) theme-{{ $current_theme }} @else theme-{{ get_app_setting('admin_portal_theme') ?? 'crater' }} @endif "> @if(isset($current_theme)) theme-{{ $current_theme }} @else theme-{{get_app_setting('admin_portal_theme') ?? 'crater'}} @endif ">
<!-- Module Scripts --> <!-- Module Scripts -->
@foreach (\Crater\Services\Module\ModuleFacade::allScripts() as $name => $path) @foreach (\Crater\Services\Module\ModuleFacade::allScripts() as $name => $path)
@ -38,14 +38,6 @@
@endforeach @endforeach
<script type="module"> <script type="module">
if (localStorage.theme === 'dark' || (!('theme' in localStorage) && window.matchMedia('(prefers-color-scheme: dark)').matches)) {
document.documentElement.classList.add('dark')
document.documentElement.style.setProperty('color-scheme', 'dark');
} else {
document.documentElement.classList.remove('dark')
document.documentElement.style.setProperty('color-scheme', 'light')
}
@if(isset($customer_logo)) @if(isset($customer_logo))
window.customer_logo = "/storage/{{$customer_logo}}" window.customer_logo = "/storage/{{$customer_logo}}"
@ -65,12 +57,12 @@
window.login_page_description = "{{$login_page_description}}" window.login_page_description = "{{$login_page_description}}"
@endif @endif
@if(isset($copyright_text)) @if(isset($copyright_text))
window.copyright_text = "{{$copyright_text}}" window.copyright_text = "{{$copyright_text}}"
@endif @endif
window.Crater.start() window.Crater.start()
</script> </script>

View File

@ -19,7 +19,6 @@ module.exports = {
'./resources/scripts/**/*.js', './resources/scripts/**/*.js',
'./resources/scripts/**/*.vue', './resources/scripts/**/*.vue',
], ],
darkMode: 'class',
theme: { theme: {
extend: { extend: {
colors: { colors: {

View File

@ -415,31 +415,32 @@ test('update estimate with EUR currency', function () {
$response = putJson('api/v1/estimates/'.$estimate->id, $estimate2); $response = putJson('api/v1/estimates/'.$estimate->id, $estimate2);
$estimate_assert = collect($estimate2) $this->assertDatabaseHas('estimates', [
->only([ 'id' => $estimate['id'],
'id', 'template_name' => $estimate2['template_name'],
'template_name', 'estimate_number' => $estimate2['estimate_number'],
'estimate_number', 'discount_type' => $estimate2['discount_type'],
'discount_type', 'discount_val' => $estimate2['discount_val'],
'discount_val', 'sub_total' => $estimate2['sub_total'],
'sub_total', 'discount' => $estimate2['discount'],
'discount', 'customer_id' => $estimate2['customer_id'],
'customer_id', 'total' => $estimate2['total'],
'total', 'tax' => $estimate2['tax'],
'tax' 'exchange_rate' => $estimate2['exchange_rate'],
]) 'base_discount_val' => $estimate2['base_discount_val'],
->toArray(); 'base_sub_total' => $estimate2['base_sub_total'],
'base_total' => $estimate2['base_total'],
'base_tax' => $estimate2['base_tax'],
]);
$this->assertDatabaseHas('estimates', $estimate_assert); $this->assertDatabaseHas('estimate_items', [
'estimate_id' => $estimate2['items'][0]['estimate_id'],
$estimate_item_assert = collect($estimate2['items'][0]) 'exchange_rate' => $estimate2['items'][0]['exchange_rate'],
->only([ 'base_price' => $estimate2['items'][0]['base_price'],
'estimate_id', 'base_discount_val' => $estimate2['items'][0]['base_discount_val'],
'amount' 'base_tax' => $estimate2['items'][0]['base_tax'],
]) 'base_total' => $estimate2['items'][0]['base_total'],
->toArray(); ]);
$this->assertDatabaseHas('estimate_items', $estimate_item_assert);
$response->assertStatus(200); $response->assertStatus(200);
}); });

View File

@ -37,15 +37,13 @@ test('create expense', function () {
postJson('api/v1/expenses', $expense)->assertStatus(201); postJson('api/v1/expenses', $expense)->assertStatus(201);
$expense = collect($expense) $this->assertDatabaseHas('expenses', [
->only([ 'notes' => $expense['notes'],
'notes', 'expense_category_id' => $expense['expense_category_id'],
'expense_category_id', 'amount' => $expense['amount'],
'amount' 'exchange_rate' => $expense['exchange_rate'],
]) 'base_amount' => $expense['base_amount'],
->toArray(); ]);
$this->assertDatabaseHas('expenses', $expense);
}); });
test('store validates using a form request', function () { test('store validates using a form request', function () {
@ -148,13 +146,11 @@ test('update expense with EUR currency', function () {
putJson('api/v1/expenses/'.$expense->id, $expense2)->assertOk(); putJson('api/v1/expenses/'.$expense->id, $expense2)->assertOk();
$expense2 = collect($expense2) $this->assertDatabaseHas('expenses', [
->only([ 'id' => $expense->id,
'id', 'expense_category_id' => $expense2['expense_category_id'],
'expense_category_id', 'amount' => $expense2['amount'],
'amount' 'exchange_rate' => $expense2['exchange_rate'],
]) 'base_amount' => $expense2['base_amount'],
->toArray(); ]);
$this->assertDatabaseHas('expenses', $expense2);
}); });

View File

@ -9,6 +9,7 @@ use Crater\Models\Tax;
use Crater\Models\User; use Crater\Models\User;
use Illuminate\Support\Facades\Artisan; use Illuminate\Support\Facades\Artisan;
use Laravel\Sanctum\Sanctum; use Laravel\Sanctum\Sanctum;
use function Pest\Laravel\getJson; use function Pest\Laravel\getJson;
use function Pest\Laravel\postJson; use function Pest\Laravel\postJson;
use function Pest\Laravel\putJson; use function Pest\Laravel\putJson;
@ -430,36 +431,31 @@ test('update invoice with EUR currency', function () {
putJson('api/v1/invoices/'.$invoice->id, $invoice2)->assertOk(); putJson('api/v1/invoices/'.$invoice->id, $invoice2)->assertOk();
$invoice_assert = collect($invoice2) $this->assertDatabaseHas('invoices', [
->only([ 'id' => $invoice['id'],
'invoice_number', 'invoice_number' => $invoice2['invoice_number'],
'template_name', 'sub_total' => $invoice2['sub_total'],
'sub_total', 'total' => $invoice2['total'],
'total', 'tax' => $invoice2['tax'],
'tax', 'discount' => $invoice2['discount'],
'discount', 'customer_id' => $invoice2['customer_id'],
'customer_id', 'template_name' => $invoice2['template_name'],
]) 'exchange_rate' => $invoice2['exchange_rate'],
->toArray(); 'base_total' => $invoice2['base_total'],
]);
$this->assertDatabaseHas('invoices', $invoice_assert); $this->assertDatabaseHas('invoice_items', [
'invoice_id' => $invoice2['items'][0]['invoice_id'],
'item_id' => $invoice2['items'][0]['item_id'],
'name' => $invoice2['items'][0]['name'],
'exchange_rate' => $invoice2['items'][0]['exchange_rate'],
'base_price' => $invoice2['items'][0]['base_price'],
'base_total' => $invoice2['items'][0]['base_total'],
]);
$invoice_item_assert = collect($invoice2['items'][0]) $this->assertDatabaseHas('taxes', [
->only([ 'amount' => $invoice2['taxes'][0]['amount'],
'invoice_id', 'name' => $invoice2['taxes'][0]['name'],
'item_id', 'base_amount' => $invoice2['taxes'][0]['base_amount'],
'name', ]);
])
->toArray();
$this->assertDatabaseHas('invoice_items', $invoice_item_assert);
$invoice_tax_assert = collect($invoice2['taxes'][0])
->only([
'name',
'amount'
])
->toArray();
$this->assertDatabaseHas('taxes', $invoice_tax_assert);
}); });

View File

@ -11,7 +11,8 @@ RUN apt-get update && apt-get install -y \
unzip \ unzip \
libzip-dev \ libzip-dev \
libmagickwand-dev \ libmagickwand-dev \
mariadb-client mariadb-client \
npm
# Clear cache # Clear cache
RUN apt-get clean && rm -rf /var/lib/apt/lists/* RUN apt-get clean && rm -rf /var/lib/apt/lists/*
@ -45,4 +46,19 @@ RUN chmod -R 775 composer.json composer.lock \
RUN chown -R $(whoami):$(whoami) /var/log/ RUN chown -R $(whoami):$(whoami) /var/log/
RUN chmod -R 775 /var/log RUN chmod -R 775 /var/log
# Cleanup manually generated build files
RUN rm -rf /var/www/public/build
RUN npm config set user 0
RUN npm config set unsafe-perm true
# Frontend bulding
RUN sed -i 's/DB_CONNECTION=mysql/DB_CONNECTION=sqlite/g' /var/www/.env
RUN sed -i 's/DB_DATABASE=crater/DB_DATABASE=\/tmp\/crater.sqlite/g' /var/www/.env
RUN touch /tmp/crater.sqlite
RUN composer install --no-interaction --prefer-dist
RUN npm i -f
RUN npm install --save-dev sass
RUN export NODE_OPTIONS="--max-old-space-size=4096" && /usr/bin/npx vite build --target=es2020
RUN sed -i 's/DB_CONNECTION=sqlite/DB_CONNECTION=mysql/g' /var/www/.env
RUN sed -i 's/DB_DATABASE=\/tmp\/crater.sqlite/DB_DATABASE=crater/g' /var/www/.env
USER crater-user USER crater-user

View File

@ -1,7 +1,9 @@
ARG BASE_IMAGE
FROM $BASE_IMAGE as build
FROM nginx:1.17-alpine FROM nginx:1.17-alpine
RUN rm /etc/nginx/conf.d/default.conf RUN rm /etc/nginx/conf.d/default.conf
COPY ./ /var/www COPY --from=build /var/www /var/www
COPY ./uffizzi/nginx/nginx /etc/nginx/conf.d/ COPY ./uffizzi/nginx/nginx /etc/nginx/conf.d/

1985
yarn.lock

File diff suppressed because it is too large Load Diff