forked from verdnatura/salix-front
Merge branch 'dev' into 6321_negative_tickets
This commit is contained in:
commit
5b1819f7da
|
@ -7,6 +7,10 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
||||||
|
|
||||||
## [2420.01]
|
## [2420.01]
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- (Item) => Se añade la opción de añadir un comentario del motivo de hacer una foto
|
||||||
|
|
||||||
## [2418.01]
|
## [2418.01]
|
||||||
|
|
||||||
## [2416.01] - 2024-04-18
|
## [2416.01] - 2024-04-18
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
{
|
{
|
||||||
"name": "salix-front",
|
"name": "salix-front",
|
||||||
"version": "24.24.1",
|
"version": "24.24.3",
|
||||||
"description": "Salix frontend",
|
"description": "Salix frontend",
|
||||||
"productName": "Salix",
|
"productName": "Salix",
|
||||||
"author": "Verdnatura",
|
"author": "Verdnatura",
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
<script setup>
|
<script setup>
|
||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
import { computed, ref, watch } from 'vue';
|
import { computed, ref, watch } from 'vue';
|
||||||
|
import { useRouter } from 'vue-router';
|
||||||
import { useI18n } from 'vue-i18n';
|
import { useI18n } from 'vue-i18n';
|
||||||
import { useQuasar } from 'quasar';
|
import { useQuasar } from 'quasar';
|
||||||
import { useValidator } from 'src/composables/useValidator';
|
import { useValidator } from 'src/composables/useValidator';
|
||||||
|
@ -10,6 +11,7 @@ import VnConfirm from 'components/ui/VnConfirm.vue';
|
||||||
import SkeletonTable from 'components/ui/SkeletonTable.vue';
|
import SkeletonTable from 'components/ui/SkeletonTable.vue';
|
||||||
import { tMobile } from 'src/composables/tMobile';
|
import { tMobile } from 'src/composables/tMobile';
|
||||||
|
|
||||||
|
const { push } = useRouter();
|
||||||
const quasar = useQuasar();
|
const quasar = useQuasar();
|
||||||
const stateStore = useStateStore();
|
const stateStore = useStateStore();
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
|
@ -60,6 +62,11 @@ const $props = defineProps({
|
||||||
type: Function,
|
type: Function,
|
||||||
default: null,
|
default: null,
|
||||||
},
|
},
|
||||||
|
goTo: {
|
||||||
|
type: String,
|
||||||
|
default: '',
|
||||||
|
description: 'It is used for redirect on click "save and continue"',
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const isLoading = ref(false);
|
const isLoading = ref(false);
|
||||||
|
@ -128,6 +135,11 @@ async function onSubmit() {
|
||||||
await saveChanges($props.saveFn ? formData.value : null);
|
await saveChanges($props.saveFn ? formData.value : null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function onSumbitAndGo() {
|
||||||
|
await onSubmit();
|
||||||
|
push({ path: $props.goTo });
|
||||||
|
}
|
||||||
|
|
||||||
async function saveChanges(data) {
|
async function saveChanges(data) {
|
||||||
if ($props.saveFn) {
|
if ($props.saveFn) {
|
||||||
$props.saveFn(data, getChanges);
|
$props.saveFn(data, getChanges);
|
||||||
|
@ -310,7 +322,40 @@ watch(formUrl, async () => {
|
||||||
:title="t('globals.reset')"
|
:title="t('globals.reset')"
|
||||||
v-if="$props.defaultReset"
|
v-if="$props.defaultReset"
|
||||||
/>
|
/>
|
||||||
|
<QBtnDropdown
|
||||||
|
v-if="$props.goTo && $props.defaultSave"
|
||||||
|
@click="onSumbitAndGo"
|
||||||
|
:label="tMobile('globals.saveAndContinue')"
|
||||||
|
:title="t('globals.saveAndContinue')"
|
||||||
|
:disable="!hasChanges"
|
||||||
|
color="primary"
|
||||||
|
icon="save"
|
||||||
|
split
|
||||||
|
>
|
||||||
|
<QList>
|
||||||
|
<QItem
|
||||||
|
color="primary"
|
||||||
|
clickable
|
||||||
|
v-close-popup
|
||||||
|
@click="onSubmit"
|
||||||
|
:title="t('globals.save')"
|
||||||
|
>
|
||||||
|
<QItemSection>
|
||||||
|
<QItemLabel>
|
||||||
|
<QIcon
|
||||||
|
name="save"
|
||||||
|
color="white"
|
||||||
|
class="q-mr-sm"
|
||||||
|
size="sm"
|
||||||
|
/>
|
||||||
|
{{ t('globals.save').toUpperCase() }}
|
||||||
|
</QItemLabel>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
</QList>
|
||||||
|
</QBtnDropdown>
|
||||||
<QBtn
|
<QBtn
|
||||||
|
v-else-if="!$props.goTo && $props.defaultSave"
|
||||||
:label="tMobile('globals.save')"
|
:label="tMobile('globals.save')"
|
||||||
ref="saveButtonRef"
|
ref="saveButtonRef"
|
||||||
color="primary"
|
color="primary"
|
||||||
|
@ -318,7 +363,6 @@ watch(formUrl, async () => {
|
||||||
@click="onSubmit"
|
@click="onSubmit"
|
||||||
:disable="!hasChanges"
|
:disable="!hasChanges"
|
||||||
:title="t('globals.save')"
|
:title="t('globals.save')"
|
||||||
v-if="$props.defaultSave"
|
|
||||||
/>
|
/>
|
||||||
<slot name="moreAfterActions" />
|
<slot name="moreAfterActions" />
|
||||||
</QBtnGroup>
|
</QBtnGroup>
|
||||||
|
|
|
@ -24,7 +24,7 @@ const $props = defineProps({
|
||||||
default: '',
|
default: '',
|
||||||
},
|
},
|
||||||
limit: {
|
limit: {
|
||||||
type: String,
|
type: [String, Number],
|
||||||
default: '30',
|
default: '30',
|
||||||
},
|
},
|
||||||
params: {
|
params: {
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
<script setup>
|
<script setup>
|
||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
import { onMounted, onUnmounted, computed, ref, watch, nextTick } from 'vue';
|
import { onMounted, onUnmounted, computed, ref, watch, nextTick } from 'vue';
|
||||||
import { onBeforeRouteLeave } from 'vue-router';
|
import { onBeforeRouteLeave, useRouter } from 'vue-router';
|
||||||
import { useI18n } from 'vue-i18n';
|
import { useI18n } from 'vue-i18n';
|
||||||
import { useQuasar } from 'quasar';
|
import { useQuasar } from 'quasar';
|
||||||
import { useState } from 'src/composables/useState';
|
import { useState } from 'src/composables/useState';
|
||||||
|
@ -11,7 +11,9 @@ import useNotify from 'src/composables/useNotify.js';
|
||||||
import SkeletonForm from 'components/ui/SkeletonForm.vue';
|
import SkeletonForm from 'components/ui/SkeletonForm.vue';
|
||||||
import VnConfirm from './ui/VnConfirm.vue';
|
import VnConfirm from './ui/VnConfirm.vue';
|
||||||
import { tMobile } from 'src/composables/tMobile';
|
import { tMobile } from 'src/composables/tMobile';
|
||||||
|
import { useArrayData } from 'src/composables/useArrayData';
|
||||||
|
|
||||||
|
const { push } = useRouter();
|
||||||
const quasar = useQuasar();
|
const quasar = useQuasar();
|
||||||
const state = useState();
|
const state = useState();
|
||||||
const stateStore = useStateStore();
|
const stateStore = useStateStore();
|
||||||
|
@ -74,55 +76,17 @@ const $props = defineProps({
|
||||||
type: Function,
|
type: Function,
|
||||||
default: null,
|
default: null,
|
||||||
},
|
},
|
||||||
|
goTo: {
|
||||||
|
type: String,
|
||||||
|
default: '',
|
||||||
|
description: 'It is used for redirect on click "save and continue"',
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const emit = defineEmits(['onFetch', 'onDataSaved']);
|
const emit = defineEmits(['onFetch', 'onDataSaved']);
|
||||||
|
|
||||||
const componentIsRendered = ref(false);
|
const componentIsRendered = ref(false);
|
||||||
|
const arrayData = useArrayData($props.model);
|
||||||
onMounted(async () => {
|
|
||||||
originalData.value = $props.formInitialData;
|
|
||||||
nextTick(() => {
|
|
||||||
componentIsRendered.value = true;
|
|
||||||
});
|
|
||||||
|
|
||||||
// Podemos enviarle al form la estructura de data inicial sin necesidad de fetchearla
|
|
||||||
state.set($props.model, $props.formInitialData);
|
|
||||||
if ($props.autoLoad && !$props.formInitialData) {
|
|
||||||
await fetch();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Si así se desea disparamos el watcher del form después de 100ms, asi darle tiempo de que se haya cargado la data inicial
|
|
||||||
// para evitar que detecte cambios cuando es data inicial default
|
|
||||||
if ($props.observeFormChanges) {
|
|
||||||
setTimeout(() => {
|
|
||||||
startFormWatcher();
|
|
||||||
}, 100);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
onBeforeRouteLeave((to, from, next) => {
|
|
||||||
if (hasChanges.value && $props.observeFormChanges)
|
|
||||||
quasar.dialog({
|
|
||||||
component: VnConfirm,
|
|
||||||
componentProps: {
|
|
||||||
title: t('Unsaved changes will be lost'),
|
|
||||||
message: t('Are you sure exit without saving?'),
|
|
||||||
promise: () => next(),
|
|
||||||
},
|
|
||||||
});
|
|
||||||
else next();
|
|
||||||
});
|
|
||||||
|
|
||||||
onUnmounted(() => {
|
|
||||||
// Restauramos los datos originales en el store si se realizaron cambios en el formulario pero no se guardaron, evitando modificaciones erróneas.
|
|
||||||
if (hasChanges.value) {
|
|
||||||
state.set($props.model, originalData.value);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if ($props.clearStoreOnUnmount) state.unset($props.model);
|
|
||||||
});
|
|
||||||
|
|
||||||
const isLoading = ref(false);
|
const isLoading = ref(false);
|
||||||
// Si elegimos observar los cambios del form significa que inicialmente las actions estaran deshabilitadas
|
// Si elegimos observar los cambios del form significa que inicialmente las actions estaran deshabilitadas
|
||||||
const isResetting = ref(false);
|
const isResetting = ref(false);
|
||||||
|
@ -143,29 +107,72 @@ const defaultButtons = computed(() => ({
|
||||||
},
|
},
|
||||||
...$props.defaultButtons,
|
...$props.defaultButtons,
|
||||||
}));
|
}));
|
||||||
const startFormWatcher = () => {
|
|
||||||
|
onMounted(async () => {
|
||||||
|
originalData.value = JSON.parse(JSON.stringify($props.formInitialData ?? {}));
|
||||||
|
|
||||||
|
nextTick(() => (componentIsRendered.value = true));
|
||||||
|
|
||||||
|
// Podemos enviarle al form la estructura de data inicial sin necesidad de fetchearla
|
||||||
|
state.set($props.model, $props.formInitialData);
|
||||||
|
|
||||||
|
if ($props.autoLoad && !$props.formInitialData && $props.url) await fetch();
|
||||||
|
else if (arrayData.store.data) updateAndEmit(arrayData.store.data, 'onFetch');
|
||||||
|
|
||||||
|
if ($props.observeFormChanges) {
|
||||||
watch(
|
watch(
|
||||||
() => formData.value,
|
() => formData.value,
|
||||||
(val) => {
|
(newVal, oldVal) => {
|
||||||
hasChanges.value = !isResetting.value && val;
|
if (!oldVal) return;
|
||||||
|
hasChanges.value =
|
||||||
|
!isResetting.value &&
|
||||||
|
JSON.stringify(newVal) !== JSON.stringify(originalData.value);
|
||||||
isResetting.value = false;
|
isResetting.value = false;
|
||||||
},
|
},
|
||||||
{ deep: true }
|
{ deep: true }
|
||||||
);
|
);
|
||||||
};
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!$props.url)
|
||||||
|
watch(
|
||||||
|
() => arrayData.store.data,
|
||||||
|
(val) => updateAndEmit(val, 'onFetch')
|
||||||
|
);
|
||||||
|
|
||||||
|
watch(formUrl, async () => {
|
||||||
|
originalData.value = null;
|
||||||
|
reset();
|
||||||
|
await fetch();
|
||||||
|
});
|
||||||
|
|
||||||
|
onBeforeRouteLeave((to, from, next) => {
|
||||||
|
if (hasChanges.value && $props.observeFormChanges)
|
||||||
|
quasar.dialog({
|
||||||
|
component: VnConfirm,
|
||||||
|
componentProps: {
|
||||||
|
title: t('Unsaved changes will be lost'),
|
||||||
|
message: t('Are you sure exit without saving?'),
|
||||||
|
promise: () => next(),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
else next();
|
||||||
|
});
|
||||||
|
|
||||||
|
onUnmounted(() => {
|
||||||
|
// Restauramos los datos originales en el store si se realizaron cambios en el formulario pero no se guardaron, evitando modificaciones erróneas.
|
||||||
|
if (hasChanges.value) return state.set($props.model, originalData.value);
|
||||||
|
if ($props.clearStoreOnUnmount) state.unset($props.model);
|
||||||
|
});
|
||||||
|
|
||||||
async function fetch() {
|
async function fetch() {
|
||||||
try {
|
try {
|
||||||
let { data } = await axios.get($props.url, {
|
let { data } = await axios.get($props.url, {
|
||||||
params: { filter: JSON.stringify($props.filter) },
|
params: { filter: JSON.stringify($props.filter) },
|
||||||
});
|
});
|
||||||
|
|
||||||
if (Array.isArray(data)) data = data[0] ?? {};
|
if (Array.isArray(data)) data = data[0] ?? {};
|
||||||
|
|
||||||
state.set($props.model, data);
|
updateAndEmit(data, 'onFetch');
|
||||||
originalData.value = data && JSON.parse(JSON.stringify(data));
|
|
||||||
|
|
||||||
emit('onFetch', state.get($props.model));
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
state.set($props.model, {});
|
state.set($props.model, {});
|
||||||
originalData.value = {};
|
originalData.value = {};
|
||||||
|
@ -173,38 +180,39 @@ async function fetch() {
|
||||||
}
|
}
|
||||||
|
|
||||||
async function save() {
|
async function save() {
|
||||||
if ($props.observeFormChanges && !hasChanges.value) {
|
if ($props.observeFormChanges && !hasChanges.value)
|
||||||
notify('globals.noChanges', 'negative');
|
return notify('globals.noChanges', 'negative');
|
||||||
return;
|
|
||||||
}
|
|
||||||
isLoading.value = true;
|
|
||||||
|
|
||||||
|
isLoading.value = true;
|
||||||
try {
|
try {
|
||||||
const body = $props.mapper ? $props.mapper(formData.value) : formData.value;
|
const body = $props.mapper ? $props.mapper(formData.value) : formData.value;
|
||||||
|
const method = $props.urlCreate ? 'post' : 'patch';
|
||||||
|
const url =
|
||||||
|
$props.urlCreate || $props.urlUpdate || $props.url || arrayData.store.url;
|
||||||
let response;
|
let response;
|
||||||
|
|
||||||
if ($props.saveFn) response = await $props.saveFn(body);
|
if ($props.saveFn) response = await $props.saveFn(body);
|
||||||
else
|
else response = await axios[method](url, body);
|
||||||
response = await axios[$props.urlCreate ? 'post' : 'patch'](
|
|
||||||
$props.urlCreate || $props.urlUpdate || $props.url,
|
|
||||||
body
|
|
||||||
);
|
|
||||||
if ($props.urlCreate) notify('globals.dataCreated', 'positive');
|
if ($props.urlCreate) notify('globals.dataCreated', 'positive');
|
||||||
|
|
||||||
emit('onDataSaved', formData.value, response?.data);
|
|
||||||
originalData.value = JSON.parse(JSON.stringify(formData.value));
|
|
||||||
hasChanges.value = false;
|
hasChanges.value = false;
|
||||||
|
isLoading.value = false;
|
||||||
|
|
||||||
|
updateAndEmit(response?.data, 'onDataSaved');
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.error(err);
|
console.error(err);
|
||||||
notify('errors.writeRequest', 'negative');
|
notify('errors.writeRequest', 'negative');
|
||||||
}
|
}
|
||||||
isLoading.value = false;
|
}
|
||||||
|
|
||||||
|
async function saveAndGo() {
|
||||||
|
await save();
|
||||||
|
push({ path: $props.goTo });
|
||||||
}
|
}
|
||||||
|
|
||||||
function reset() {
|
function reset() {
|
||||||
state.set($props.model, originalData.value);
|
updateAndEmit(originalData.value, 'onFetch');
|
||||||
originalData.value = JSON.parse(JSON.stringify(originalData.value));
|
|
||||||
|
|
||||||
emit('onFetch', state.get($props.model));
|
|
||||||
if ($props.observeFormChanges) {
|
if ($props.observeFormChanges) {
|
||||||
hasChanges.value = false;
|
hasChanges.value = false;
|
||||||
isResetting.value = true;
|
isResetting.value = true;
|
||||||
|
@ -226,17 +234,15 @@ function filter(value, update, filterOptions) {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
watch(formUrl, async () => {
|
function updateAndEmit(val, evt) {
|
||||||
originalData.value = null;
|
state.set($props.model, val);
|
||||||
reset();
|
originalData.value = val && JSON.parse(JSON.stringify(val));
|
||||||
fetch();
|
if (!$props.url) arrayData.store.data = val;
|
||||||
});
|
|
||||||
|
|
||||||
defineExpose({
|
emit(evt, state.get($props.model));
|
||||||
save,
|
}
|
||||||
isLoading,
|
|
||||||
hasChanges,
|
defineExpose({ save, isLoading, hasChanges });
|
||||||
});
|
|
||||||
</script>
|
</script>
|
||||||
<template>
|
<template>
|
||||||
<div class="column items-center full-width">
|
<div class="column items-center full-width">
|
||||||
|
@ -273,10 +279,42 @@ defineExpose({
|
||||||
:disable="!hasChanges"
|
:disable="!hasChanges"
|
||||||
:title="t(defaultButtons.reset.label)"
|
:title="t(defaultButtons.reset.label)"
|
||||||
/>
|
/>
|
||||||
|
<QBtnDropdown
|
||||||
|
v-if="$props.goTo"
|
||||||
|
@click="saveAndGo"
|
||||||
|
:label="tMobile('globals.saveAndContinue')"
|
||||||
|
:title="t('globals.saveAndContinue')"
|
||||||
|
:disable="!hasChanges"
|
||||||
|
color="primary"
|
||||||
|
icon="save"
|
||||||
|
split
|
||||||
|
>
|
||||||
|
<QList>
|
||||||
|
<QItem
|
||||||
|
clickable
|
||||||
|
v-close-popup
|
||||||
|
@click="save"
|
||||||
|
:title="t('globals.save')"
|
||||||
|
>
|
||||||
|
<QItemSection>
|
||||||
|
<QItemLabel>
|
||||||
|
<QIcon
|
||||||
|
name="save"
|
||||||
|
color="white"
|
||||||
|
class="q-mr-sm"
|
||||||
|
size="sm"
|
||||||
|
/>
|
||||||
|
{{ t('globals.save').toUpperCase() }}
|
||||||
|
</QItemLabel>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
</QList>
|
||||||
|
</QBtnDropdown>
|
||||||
<QBtn
|
<QBtn
|
||||||
:label="tMobile(defaultButtons.save.label)"
|
v-else
|
||||||
:color="defaultButtons.save.color"
|
:label="tMobile('globals.save')"
|
||||||
:icon="defaultButtons.save.icon"
|
color="primary"
|
||||||
|
icon="save"
|
||||||
@click="save"
|
@click="save"
|
||||||
:disable="!hasChanges"
|
:disable="!hasChanges"
|
||||||
:title="t(defaultButtons.save.label)"
|
:title="t(defaultButtons.save.label)"
|
||||||
|
|
|
@ -20,6 +20,8 @@ const props = defineProps({
|
||||||
searchUrl: { type: String, default: undefined },
|
searchUrl: { type: String, default: undefined },
|
||||||
searchbarLabel: { type: String, default: '' },
|
searchbarLabel: { type: String, default: '' },
|
||||||
searchbarInfo: { type: String, default: '' },
|
searchbarInfo: { type: String, default: '' },
|
||||||
|
searchCustomRouteRedirect: { type: String, default: undefined },
|
||||||
|
searchRedirect: { type: Boolean, default: true },
|
||||||
});
|
});
|
||||||
|
|
||||||
const stateStore = useStateStore();
|
const stateStore = useStateStore();
|
||||||
|
@ -42,7 +44,7 @@ onBeforeMount(async () => {
|
||||||
if (props.baseUrl) {
|
if (props.baseUrl) {
|
||||||
onBeforeRouteUpdate(async (to, from) => {
|
onBeforeRouteUpdate(async (to, from) => {
|
||||||
if (to.params.id !== from.params.id) {
|
if (to.params.id !== from.params.id) {
|
||||||
arrayData.store.url = `${props.baseUrl}/${route.params.id}`;
|
arrayData.store.url = `${props.baseUrl}/${to.params.id}`;
|
||||||
await arrayData.fetch({ append: false });
|
await arrayData.fetch({ append: false });
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -62,6 +64,8 @@ watchEffect(() => {
|
||||||
:url="props.searchUrl"
|
:url="props.searchUrl"
|
||||||
:label="props.searchbarLabel"
|
:label="props.searchbarLabel"
|
||||||
:info="props.searchbarInfo"
|
:info="props.searchbarInfo"
|
||||||
|
:custom-route-redirect-name="searchCustomRouteRedirect"
|
||||||
|
:redirect="searchRedirect"
|
||||||
/>
|
/>
|
||||||
</slot>
|
</slot>
|
||||||
</Teleport>
|
</Teleport>
|
||||||
|
|
|
@ -78,6 +78,7 @@ async function save() {
|
||||||
const body = mapperDms(dms.value);
|
const body = mapperDms(dms.value);
|
||||||
const response = await axios.post(getUrl(), body[0], body[1]);
|
const response = await axios.post(getUrl(), body[0], body[1]);
|
||||||
emit('onDataSaved', body[1].params, response);
|
emit('onDataSaved', body[1].params, response);
|
||||||
|
return response;
|
||||||
}
|
}
|
||||||
|
|
||||||
function defaultData() {
|
function defaultData() {
|
||||||
|
|
|
@ -622,21 +622,6 @@ setLogTree();
|
||||||
</QList>
|
</QList>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<Teleport v-if="stateStore.isHeaderMounted()" to="#actions-append">
|
|
||||||
<div class="row q-gutter-x-sm">
|
|
||||||
<QBtn
|
|
||||||
flat
|
|
||||||
@click.stop="stateStore.toggleRightDrawer()"
|
|
||||||
round
|
|
||||||
dense
|
|
||||||
icon="menu"
|
|
||||||
>
|
|
||||||
<QTooltip bottom anchor="bottom right">
|
|
||||||
{{ t('globals.collapseMenu') }}
|
|
||||||
</QTooltip>
|
|
||||||
</QBtn>
|
|
||||||
</div>
|
|
||||||
</Teleport>
|
|
||||||
<QDrawer v-model="stateStore.rightDrawer" show-if-above side="right" :width="300">
|
<QDrawer v-model="stateStore.rightDrawer" show-if-above side="right" :width="300">
|
||||||
<QScrollArea class="fit text-grey-8">
|
<QScrollArea class="fit text-grey-8">
|
||||||
<QList dense>
|
<QList dense>
|
||||||
|
|
|
@ -1,16 +1,16 @@
|
||||||
<script setup>
|
<script setup>
|
||||||
const $props = defineProps({
|
defineProps({
|
||||||
url: { type: String, default: null },
|
url: { type: String, default: null },
|
||||||
text: { type: String, default: null },
|
text: { type: String, default: null },
|
||||||
icon: { type: String, default: 'open_in_new' },
|
icon: { type: String, default: 'open_in_new' },
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
<template>
|
<template>
|
||||||
<div class="titleBox">
|
<div :class="$q.screen.gt.md ? 'q-pb-lg' : 'q-pb-md'">
|
||||||
<div class="header-link">
|
<div class="header-link">
|
||||||
<a :href="$props.url" :class="$props.url ? 'link' : 'color-vn-text'">
|
<a :href="url" :class="url ? 'link' : 'color-vn-text'">
|
||||||
{{ $props.text }}
|
{{ text }}
|
||||||
<QIcon v-if="url" :name="$props.icon" />
|
<QIcon v-if="url" :name="icon" />
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -19,7 +19,4 @@ const $props = defineProps({
|
||||||
a {
|
a {
|
||||||
font-size: large;
|
font-size: large;
|
||||||
}
|
}
|
||||||
.titleBox {
|
|
||||||
padding-bottom: 2%;
|
|
||||||
}
|
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -5,6 +5,7 @@ import SkeletonDescriptor from 'components/ui/SkeletonDescriptor.vue';
|
||||||
import { useArrayData } from 'composables/useArrayData';
|
import { useArrayData } from 'composables/useArrayData';
|
||||||
import { useSummaryDialog } from 'src/composables/useSummaryDialog';
|
import { useSummaryDialog } from 'src/composables/useSummaryDialog';
|
||||||
import { useState } from 'src/composables/useState';
|
import { useState } from 'src/composables/useState';
|
||||||
|
import { useRoute } from 'vue-router';
|
||||||
|
|
||||||
const $props = defineProps({
|
const $props = defineProps({
|
||||||
url: {
|
url: {
|
||||||
|
@ -15,21 +16,21 @@ const $props = defineProps({
|
||||||
type: Object,
|
type: Object,
|
||||||
default: null,
|
default: null,
|
||||||
},
|
},
|
||||||
module: {
|
|
||||||
type: String,
|
|
||||||
required: true,
|
|
||||||
},
|
|
||||||
title: {
|
title: {
|
||||||
type: String,
|
type: String,
|
||||||
default: '',
|
default: '',
|
||||||
},
|
},
|
||||||
subtitle: {
|
subtitle: {
|
||||||
type: Number,
|
type: Number,
|
||||||
default: 0,
|
default: null,
|
||||||
},
|
},
|
||||||
dataKey: {
|
dataKey: {
|
||||||
type: String,
|
type: String,
|
||||||
default: '',
|
default: null,
|
||||||
|
},
|
||||||
|
module: {
|
||||||
|
type: String,
|
||||||
|
default: null,
|
||||||
},
|
},
|
||||||
summary: {
|
summary: {
|
||||||
type: Object,
|
type: Object,
|
||||||
|
@ -40,21 +41,27 @@ const $props = defineProps({
|
||||||
const state = useState();
|
const state = useState();
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
const { viewSummary } = useSummaryDialog();
|
const { viewSummary } = useSummaryDialog();
|
||||||
const arrayData = useArrayData($props.dataKey || $props.module, {
|
let arrayData;
|
||||||
|
let store;
|
||||||
|
let entity;
|
||||||
|
const isLoading = ref(false);
|
||||||
|
|
||||||
|
defineExpose({ getData });
|
||||||
|
|
||||||
|
onBeforeMount(async () => {
|
||||||
|
arrayData = useArrayData($props.dataKey, {
|
||||||
url: $props.url,
|
url: $props.url,
|
||||||
filter: $props.filter,
|
filter: $props.filter,
|
||||||
skip: 0,
|
skip: 0,
|
||||||
});
|
});
|
||||||
const { store } = arrayData;
|
store = arrayData.store;
|
||||||
const entity = computed(() => (Array.isArray(store.data) ? store.data[0] : store.data));
|
entity = computed(() => (Array.isArray(store.data) ? store.data[0] : store.data));
|
||||||
const isLoading = ref(false);
|
// It enables to load data only once if the module is the same as the dataKey
|
||||||
|
if ($props.dataKey !== useRoute().meta.moduleName) await getData();
|
||||||
defineExpose({
|
watch(
|
||||||
getData,
|
() => [$props.url, $props.filter],
|
||||||
});
|
async () => await getData()
|
||||||
onBeforeMount(async () => {
|
);
|
||||||
await getData();
|
|
||||||
watch($props, async () => await getData());
|
|
||||||
});
|
});
|
||||||
|
|
||||||
async function getData() {
|
async function getData() {
|
||||||
|
@ -132,7 +139,7 @@ const emit = defineEmits(['onFetch']);
|
||||||
<QItemLabel header class="ellipsis text-h5" :lines="1">
|
<QItemLabel header class="ellipsis text-h5" :lines="1">
|
||||||
<div class="title">
|
<div class="title">
|
||||||
<span v-if="$props.title" :title="$props.title">
|
<span v-if="$props.title" :title="$props.title">
|
||||||
{{ $props.title }}
|
{{ entity[title] ?? $props.title }}
|
||||||
</span>
|
</span>
|
||||||
<slot v-else name="description" :entity="entity">
|
<slot v-else name="description" :entity="entity">
|
||||||
<span :title="entity.name">
|
<span :title="entity.name">
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
<script setup>
|
<script setup>
|
||||||
import { onMounted, ref } from 'vue';
|
import { onMounted, ref, watch } from 'vue';
|
||||||
import { useQuasar } from 'quasar';
|
import { useQuasar } from 'quasar';
|
||||||
import { useArrayData } from 'composables/useArrayData';
|
import { useArrayData } from 'composables/useArrayData';
|
||||||
import VnInput from 'src/components/common/VnInput.vue';
|
import VnInput from 'src/components/common/VnInput.vue';
|
||||||
|
@ -67,11 +67,19 @@ const props = defineProps({
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const arrayData = useArrayData(props.dataKey, { ...props });
|
let arrayData = useArrayData(props.dataKey, { ...props });
|
||||||
const { store } = arrayData;
|
let store = arrayData.store;
|
||||||
const searchText = ref('');
|
const searchText = ref('');
|
||||||
const { navigate } = useRedirect();
|
const { navigate } = useRedirect();
|
||||||
|
|
||||||
|
watch(
|
||||||
|
() => props.dataKey,
|
||||||
|
(val) => {
|
||||||
|
arrayData = useArrayData(val, { ...props });
|
||||||
|
store = arrayData.store;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
const params = store.userParams;
|
const params = store.userParams;
|
||||||
if (params && params.search) {
|
if (params && params.search) {
|
||||||
|
|
|
@ -51,7 +51,7 @@
|
||||||
<glyph unicode="" glyph-name="agency" d="M789.333 264.533c-55.467 0-102.4-46.933-102.4-102.4s46.933-102.4 102.4-102.4 102.4 46.933 102.4 102.4c0 59.733-46.933 102.4-102.4 102.4zM789.333 110.933c-29.867 0-51.2 21.333-51.2 51.2s21.333 51.2 51.2 51.2 51.2-21.333 51.2-51.2c0-25.6-25.6-51.2-51.2-51.2zM251.733 264.533c-55.467 0-102.4-46.933-102.4-102.4s46.933-102.4 102.4-102.4c55.467 0 102.4 46.933 102.4 102.4 0 59.733-46.933 102.4-102.4 102.4zM251.733 110.933c-29.867 0-51.2 21.333-51.2 51.2s21.333 51.2 51.2 51.2c29.867 0 51.2-21.333 51.2-51.2 0-25.6-25.6-51.2-51.2-51.2zM1006.933 537.6l-196.267 192c-12.8 12.8-29.867 17.067-46.933 17.067h-98.133v38.4c0 25.6-21.333 51.2-51.2 51.2h-563.2c-29.867 0-51.2-21.333-51.2-51.2v-554.667c0-29.867 25.6-51.2 51.2-51.2h68.267c8.533 64 64 115.2 132.267 115.2 64 0 123.733-51.2 132.267-115.2h268.8c8.533 64 64 115.2 132.267 115.2s128-51.2 136.533-115.2h51.2c29.867 0 51.2 25.6 51.2 51.2v260.267c0 17.067-8.533 34.133-17.067 46.933zM725.333 682.667c0 4.267 4.267 8.533 8.533 8.533h34.133c0 0 4.267 0 4.267-4.267l153.6-145.067c4.267 0 0-12.8-4.267-12.8h-187.733c-8.533 0-8.533 4.267-8.533 8.533v145.067z" />
|
<glyph unicode="" glyph-name="agency" d="M789.333 264.533c-55.467 0-102.4-46.933-102.4-102.4s46.933-102.4 102.4-102.4 102.4 46.933 102.4 102.4c0 59.733-46.933 102.4-102.4 102.4zM789.333 110.933c-29.867 0-51.2 21.333-51.2 51.2s21.333 51.2 51.2 51.2 51.2-21.333 51.2-51.2c0-25.6-25.6-51.2-51.2-51.2zM251.733 264.533c-55.467 0-102.4-46.933-102.4-102.4s46.933-102.4 102.4-102.4c55.467 0 102.4 46.933 102.4 102.4 0 59.733-46.933 102.4-102.4 102.4zM251.733 110.933c-29.867 0-51.2 21.333-51.2 51.2s21.333 51.2 51.2 51.2c29.867 0 51.2-21.333 51.2-51.2 0-25.6-25.6-51.2-51.2-51.2zM1006.933 537.6l-196.267 192c-12.8 12.8-29.867 17.067-46.933 17.067h-98.133v38.4c0 25.6-21.333 51.2-51.2 51.2h-563.2c-29.867 0-51.2-21.333-51.2-51.2v-554.667c0-29.867 25.6-51.2 51.2-51.2h68.267c8.533 64 64 115.2 132.267 115.2 64 0 123.733-51.2 132.267-115.2h268.8c8.533 64 64 115.2 132.267 115.2s128-51.2 136.533-115.2h51.2c29.867 0 51.2 25.6 51.2 51.2v260.267c0 17.067-8.533 34.133-17.067 46.933zM725.333 682.667c0 4.267 4.267 8.533 8.533 8.533h34.133c0 0 4.267 0 4.267-4.267l153.6-145.067c4.267 0 0-12.8-4.267-12.8h-187.733c-8.533 0-8.533 4.267-8.533 8.533v145.067z" />
|
||||||
<glyph unicode="" glyph-name="credit" d="M921.6 849.067h-819.2c-55.467 0-102.4-42.667-102.4-98.133v-601.6c0-55.467 46.933-102.4 102.4-102.4h819.2c55.467 0 102.4 42.667 102.4 102.4v601.6c0 55.467-46.933 98.133-102.4 98.133zM921.6 145.067h-819.2v302.933h819.2v-302.933zM921.6 648.533h-819.2v102.4h819.2v-102.4z" />
|
<glyph unicode="" glyph-name="credit" d="M921.6 849.067h-819.2c-55.467 0-102.4-42.667-102.4-98.133v-601.6c0-55.467 46.933-102.4 102.4-102.4h819.2c55.467 0 102.4 42.667 102.4 102.4v601.6c0 55.467-46.933 98.133-102.4 98.133zM921.6 145.067h-819.2v302.933h819.2v-302.933zM921.6 648.533h-819.2v102.4h819.2v-102.4z" />
|
||||||
<glyph unicode="" glyph-name="albaran" d="M878.933 857.6h-217.6c-25.6 59.733-81.067 102.4-149.333 102.4s-123.733-42.667-145.067-102.4h-221.867c-55.467 0-102.4-46.933-102.4-102.4v-716.8c0-55.467 46.933-102.4 102.4-102.4h729.6c55.467 0 102.4 46.933 102.4 102.4v716.8c4.267 55.467-42.667 102.4-98.133 102.4zM512 857.6c29.867 0 51.2-21.333 51.2-51.2s-21.333-51.2-51.2-51.2c-29.867 0-51.2 21.333-51.2 51.2s21.333 51.2 51.2 51.2zM614.4 140.8h-362.667v102.4h366.933v-102.4zM772.267 345.6h-520.533v102.4h520.533v-102.4zM772.267 550.4h-520.533v102.4h520.533v-102.4z" />
|
<glyph unicode="" glyph-name="albaran" d="M878.933 857.6h-217.6c-25.6 59.733-81.067 102.4-149.333 102.4s-123.733-42.667-145.067-102.4h-221.867c-55.467 0-102.4-46.933-102.4-102.4v-716.8c0-55.467 46.933-102.4 102.4-102.4h729.6c55.467 0 102.4 46.933 102.4 102.4v716.8c4.267 55.467-42.667 102.4-98.133 102.4zM512 857.6c29.867 0 51.2-21.333 51.2-51.2s-21.333-51.2-51.2-51.2c-29.867 0-51.2 21.333-51.2 51.2s21.333 51.2 51.2 51.2zM614.4 140.8h-362.667v102.4h366.933v-102.4zM772.267 345.6h-520.533v102.4h520.533v-102.4zM772.267 550.4h-520.533v102.4h520.533v-102.4z" />
|
||||||
<glyph unicode="" glyph-name="deaulter" d="M677.973-64c-30.72 35.84-61.867 70.827-91.307 107.52-40.96 51.2-80.64 103.253-121.173 154.88-16.64 21.333-21.76 20.48-30.72-4.693-13.227-36.693-25.6-73.387-40.107-109.653-5.12-12.8-13.227-26.88-24.32-34.56-51.627-34.987-104.107-69.12-157.867-100.693-10.667-6.4-30.72-5.547-41.813 0.853-8.107 4.693-12.373 23.893-11.093 35.84 0.853 8.96 11.093 19.627 19.627 25.6 39.253 26.453 78.933 51.627 119.040 76.8 18.347 11.52 30.293 26.027 35.84 47.787 12.373 48.213 27.307 95.573 39.253 143.36 8.533 33.707 26.88 58.88 56.32 77.227 40.533 25.173 80.64 52.053 120.747 78.507 6.4 4.267 10.24 11.52 15.36 17.493-7.253 2.56-14.933 7.253-22.187 6.827-75.52-6.4-151.467-13.227-226.987-20.48-2.133 0-4.693-0.853-6.827-0.853-22.613-1.707-39.253 10.24-40.96 29.867s12.373 33.707 35.413 35.84c45.227 4.267 90.88 8.96 136.107 12.8 65.707 5.547 131.84 10.667 197.547 15.36 26.027 1.707 53.76-21.76 67.413-55.467 9.813-23.893 5.12-46.080-18.347-65.28-49.92-40.107-100.693-78.933-151.040-118.187-23.040-17.92-23.893-23.467-6.4-46.507 58.453-78.080 116.48-156.587 174.933-234.667 27.307-36.693 25.173-50.773-12.373-75.52-5.12 0-9.813 0-14.080 0zM791.893 649.813c-43.093 1.28-76.373-31.573-77.227-75.52-0.853-44.373 29.44-76.8 72.107-77.653 45.227-1.28 77.653 29.44 78.080 73.813 0.427 45.227-29.44 78.080-72.96 79.36zM671.147 737.707c0-72.107-34.133-136.107-87.467-176.64l-235.52-21.76c-72.107 36.693-122.027 111.787-122.027 198.4 0 122.88 99.84 222.293 222.72 222.293 122.453 0 222.293-99.413 222.293-222.293zM592.213 680.533l-50.347 18.347c-2.133-8.533-5.12-16.213-9.813-22.613-5.12-6.4-10.24-11.947-16.213-17.067-5.973-4.267-12.373-8.107-19.2-11.093s-13.653-4.693-20.053-5.547c-17.92-2.987-33.707-0.427-48.64 6.827s-26.88 18.347-36.693 32.853l76.373 12.373 7.253 32.427-97.28-15.787c-1.28 5.547-2.987 11.093-3.84 16.64l-0.853 4.267 99.413 16.213 7.253 32.427-106.667-17.493c0.853 9.387 2.987 17.493 6.4 26.027 3.84 8.533 8.107 16.213 14.080 23.040 5.547 6.827 12.8 12.373 21.333 17.067s17.92 8.107 28.587 9.813c6.827 1.28 13.227 1.707 20.907 1.28s14.507-1.707 21.333-3.84c6.827-2.133 13.653-5.973 20.053-10.24 5.973-4.693 11.947-11.093 17.493-18.773l38.827 37.973c-13.227 17.92-30.293 31.147-52.053 39.253-21.76 8.533-46.080 10.667-73.387 6.4-19.627-2.987-36.267-9.387-51.2-17.92-14.507-8.533-26.88-19.2-37.547-32-10.667-12.373-18.773-26.027-23.893-40.96-5.547-14.507-8.96-29.867-9.813-45.653l-21.76-3.84-7.253-32.427 29.013 4.693 0.427-2.987c1.28-6.827 2.56-12.8 4.267-18.347l-23.467-3.84-8.107-32.427 43.52 7.253c6.827-13.653 15.787-26.027 26.027-36.693 10.24-11.52 22.187-20.48 35.413-27.733 13.227-7.68 27.307-12.8 42.667-15.787s31.573-3.413 47.36-0.853c12.373 2.133 24.32 5.12 35.84 10.667s22.613 11.52 32.853 19.2c10.24 8.107 18.347 16.64 26.027 26.453 6.827 9.387 12.373 20.48 15.36 32.427z" />
|
<glyph unicode="" glyph-name="defaulter" d="M677.973-64c-30.72 35.84-61.867 70.827-91.307 107.52-40.96 51.2-80.64 103.253-121.173 154.88-16.64 21.333-21.76 20.48-30.72-4.693-13.227-36.693-25.6-73.387-40.107-109.653-5.12-12.8-13.227-26.88-24.32-34.56-51.627-34.987-104.107-69.12-157.867-100.693-10.667-6.4-30.72-5.547-41.813 0.853-8.107 4.693-12.373 23.893-11.093 35.84 0.853 8.96 11.093 19.627 19.627 25.6 39.253 26.453 78.933 51.627 119.040 76.8 18.347 11.52 30.293 26.027 35.84 47.787 12.373 48.213 27.307 95.573 39.253 143.36 8.533 33.707 26.88 58.88 56.32 77.227 40.533 25.173 80.64 52.053 120.747 78.507 6.4 4.267 10.24 11.52 15.36 17.493-7.253 2.56-14.933 7.253-22.187 6.827-75.52-6.4-151.467-13.227-226.987-20.48-2.133 0-4.693-0.853-6.827-0.853-22.613-1.707-39.253 10.24-40.96 29.867s12.373 33.707 35.413 35.84c45.227 4.267 90.88 8.96 136.107 12.8 65.707 5.547 131.84 10.667 197.547 15.36 26.027 1.707 53.76-21.76 67.413-55.467 9.813-23.893 5.12-46.080-18.347-65.28-49.92-40.107-100.693-78.933-151.040-118.187-23.040-17.92-23.893-23.467-6.4-46.507 58.453-78.080 116.48-156.587 174.933-234.667 27.307-36.693 25.173-50.773-12.373-75.52-5.12 0-9.813 0-14.080 0zM791.893 649.813c-43.093 1.28-76.373-31.573-77.227-75.52-0.853-44.373 29.44-76.8 72.107-77.653 45.227-1.28 77.653 29.44 78.080 73.813 0.427 45.227-29.44 78.080-72.96 79.36zM671.147 737.707c0-72.107-34.133-136.107-87.467-176.64l-235.52-21.76c-72.107 36.693-122.027 111.787-122.027 198.4 0 122.88 99.84 222.293 222.72 222.293 122.453 0 222.293-99.413 222.293-222.293zM592.213 680.533l-50.347 18.347c-2.133-8.533-5.12-16.213-9.813-22.613-5.12-6.4-10.24-11.947-16.213-17.067-5.973-4.267-12.373-8.107-19.2-11.093s-13.653-4.693-20.053-5.547c-17.92-2.987-33.707-0.427-48.64 6.827s-26.88 18.347-36.693 32.853l76.373 12.373 7.253 32.427-97.28-15.787c-1.28 5.547-2.987 11.093-3.84 16.64l-0.853 4.267 99.413 16.213 7.253 32.427-106.667-17.493c0.853 9.387 2.987 17.493 6.4 26.027 3.84 8.533 8.107 16.213 14.080 23.040 5.547 6.827 12.8 12.373 21.333 17.067s17.92 8.107 28.587 9.813c6.827 1.28 13.227 1.707 20.907 1.28s14.507-1.707 21.333-3.84c6.827-2.133 13.653-5.973 20.053-10.24 5.973-4.693 11.947-11.093 17.493-18.773l38.827 37.973c-13.227 17.92-30.293 31.147-52.053 39.253-21.76 8.533-46.080 10.667-73.387 6.4-19.627-2.987-36.267-9.387-51.2-17.92-14.507-8.533-26.88-19.2-37.547-32-10.667-12.373-18.773-26.027-23.893-40.96-5.547-14.507-8.96-29.867-9.813-45.653l-21.76-3.84-7.253-32.427 29.013 4.693 0.427-2.987c1.28-6.827 2.56-12.8 4.267-18.347l-23.467-3.84-8.107-32.427 43.52 7.253c6.827-13.653 15.787-26.027 26.027-36.693 10.24-11.52 22.187-20.48 35.413-27.733 13.227-7.68 27.307-12.8 42.667-15.787s31.573-3.413 47.36-0.853c12.373 2.133 24.32 5.12 35.84 10.667s22.613 11.52 32.853 19.2c10.24 8.107 18.347 16.64 26.027 26.453 6.827 9.387 12.373 20.48 15.36 32.427z" />
|
||||||
<glyph unicode="" glyph-name="deletedTicket" d="M160.672 85.696h693.248v639.776c0 0-2.016 234.528-349.696 234.528s-343.552-234.528-343.552-234.528v-639.776zM291.328 652.704h170.976v152.256h102.336v-152.256h171.008v-102.336h-171.008v-356.96h-102.336v356.96h-170.976v102.336zM64 61.056v-123.456h899.008v123.456h-899.008z" />
|
<glyph unicode="" glyph-name="deletedTicket" d="M160.672 85.696h693.248v639.776c0 0-2.016 234.528-349.696 234.528s-343.552-234.528-343.552-234.528v-639.776zM291.328 652.704h170.976v152.256h102.336v-152.256h171.008v-102.336h-171.008v-356.96h-102.336v356.96h-170.976v102.336zM64 61.056v-123.456h899.008v123.456h-899.008z" />
|
||||||
<glyph unicode="" glyph-name="deleteline" d="M354.133 192l-98.133 98.133 157.867 153.6-157.867 157.867 98.133 102.4 157.867-157.867 157.867 153.6 98.133-98.133-157.867-157.867 157.867-153.6-98.133-98.133-157.867 157.867-157.867-157.867zM780.8 507.733l-64-64 59.733-55.467h247.467v119.467h-243.2zM307.2 443.733l-64 64h-243.2v-119.467h251.733l55.467 55.467z" />
|
<glyph unicode="" glyph-name="deleteline" d="M354.133 192l-98.133 98.133 157.867 153.6-157.867 157.867 98.133 102.4 157.867-157.867 157.867 153.6 98.133-98.133-157.867-157.867 157.867-153.6-98.133-98.133-157.867 157.867-157.867-157.867zM780.8 507.733l-64-64 59.733-55.467h247.467v119.467h-243.2zM307.2 443.733l-64 64h-243.2v-119.467h251.733l55.467 55.467z" />
|
||||||
<glyph unicode="" glyph-name="delivery" d="M789.333 264.533c-55.467 0-102.4-46.933-102.4-102.4s46.933-102.4 102.4-102.4 102.4 46.933 102.4 102.4c0 59.733-46.933 102.4-102.4 102.4zM789.333 110.933c-29.867 0-51.2 21.333-51.2 51.2s21.333 51.2 51.2 51.2 51.2-21.333 51.2-51.2c0-25.6-25.6-51.2-51.2-51.2zM251.733 264.533c-55.467 0-102.4-46.933-102.4-102.4s46.933-102.4 102.4-102.4c55.467 0 102.4 46.933 102.4 102.4 0 59.733-46.933 102.4-102.4 102.4zM251.733 110.933c-29.867 0-51.2 21.333-51.2 51.2s21.333 51.2 51.2 51.2c29.867 0 51.2-21.333 51.2-51.2 0-25.6-25.6-51.2-51.2-51.2zM1006.933 537.6l-196.267 192c-12.8 12.8-29.867 17.067-46.933 17.067h-98.133v38.4c0 25.6-21.333 51.2-51.2 51.2h-563.2c-29.867 0-51.2-21.333-51.2-51.2v-554.667c0-29.867 25.6-51.2 51.2-51.2h68.267c8.533 64 64 115.2 132.267 115.2 64 0 123.733-51.2 132.267-115.2h268.8c8.533 64 64 115.2 132.267 115.2s128-51.2 136.533-115.2h51.2c29.867 0 51.2 25.6 51.2 51.2v260.267c0 17.067-8.533 34.133-17.067 46.933zM725.333 682.667c0 4.267 4.267 8.533 8.533 8.533h34.133c0 0 4.267 0 4.267-4.267l153.6-145.067c4.267 0 0-12.8-4.267-12.8h-187.733c-8.533 0-8.533 4.267-8.533 8.533v145.067zM311.467 597.333c0 46.933 29.867 85.333 59.733 93.867 4.267 0 4.267 0 8.533 0l98.133 12.8v-51.2c0-46.933-29.867-85.333-59.733-93.867-4.267 0-4.267 0-8.533 0l-98.133-17.067v55.467zM311.467 516.267l46.933 8.533c17.067 4.267 29.867-17.067 29.867-38.4l4.267-29.867-51.2-4.267c-17.067-4.267-29.867 12.8-29.867 38.4v25.6zM149.333 597.333v51.2l85.333 12.8c34.133 4.267 55.467-25.6 55.467-72.533v-51.2l-85.333-12.8c-34.133 0-59.733 29.867-55.467 72.533zM285.867 512v-38.4c0-34.133-21.333-64-42.667-68.267h-4.267l-72.533-8.533v38.4c0 34.133 21.333 64 42.667 68.267h4.267l72.533 8.533z" />
|
<glyph unicode="" glyph-name="delivery" d="M789.333 264.533c-55.467 0-102.4-46.933-102.4-102.4s46.933-102.4 102.4-102.4 102.4 46.933 102.4 102.4c0 59.733-46.933 102.4-102.4 102.4zM789.333 110.933c-29.867 0-51.2 21.333-51.2 51.2s21.333 51.2 51.2 51.2 51.2-21.333 51.2-51.2c0-25.6-25.6-51.2-51.2-51.2zM251.733 264.533c-55.467 0-102.4-46.933-102.4-102.4s46.933-102.4 102.4-102.4c55.467 0 102.4 46.933 102.4 102.4 0 59.733-46.933 102.4-102.4 102.4zM251.733 110.933c-29.867 0-51.2 21.333-51.2 51.2s21.333 51.2 51.2 51.2c29.867 0 51.2-21.333 51.2-51.2 0-25.6-25.6-51.2-51.2-51.2zM1006.933 537.6l-196.267 192c-12.8 12.8-29.867 17.067-46.933 17.067h-98.133v38.4c0 25.6-21.333 51.2-51.2 51.2h-563.2c-29.867 0-51.2-21.333-51.2-51.2v-554.667c0-29.867 25.6-51.2 51.2-51.2h68.267c8.533 64 64 115.2 132.267 115.2 64 0 123.733-51.2 132.267-115.2h268.8c8.533 64 64 115.2 132.267 115.2s128-51.2 136.533-115.2h51.2c29.867 0 51.2 25.6 51.2 51.2v260.267c0 17.067-8.533 34.133-17.067 46.933zM725.333 682.667c0 4.267 4.267 8.533 8.533 8.533h34.133c0 0 4.267 0 4.267-4.267l153.6-145.067c4.267 0 0-12.8-4.267-12.8h-187.733c-8.533 0-8.533 4.267-8.533 8.533v145.067zM311.467 597.333c0 46.933 29.867 85.333 59.733 93.867 4.267 0 4.267 0 8.533 0l98.133 12.8v-51.2c0-46.933-29.867-85.333-59.733-93.867-4.267 0-4.267 0-8.533 0l-98.133-17.067v55.467zM311.467 516.267l46.933 8.533c17.067 4.267 29.867-17.067 29.867-38.4l4.267-29.867-51.2-4.267c-17.067-4.267-29.867 12.8-29.867 38.4v25.6zM149.333 597.333v51.2l85.333 12.8c34.133 4.267 55.467-25.6 55.467-72.533v-51.2l-85.333-12.8c-34.133 0-59.733 29.867-55.467 72.533zM285.867 512v-38.4c0-34.133-21.333-64-42.667-68.267h-4.267l-72.533-8.533v38.4c0 34.133 21.333 64 42.667 68.267h4.267l72.533 8.533z" />
|
||||||
|
|
Before Width: | Height: | Size: 174 KiB After Width: | Height: | Size: 174 KiB |
File diff suppressed because one or more lines are too long
|
@ -26,6 +26,7 @@ globals:
|
||||||
create: Create
|
create: Create
|
||||||
edit: Edit
|
edit: Edit
|
||||||
save: Save
|
save: Save
|
||||||
|
saveAndContinue: Save and continue
|
||||||
remove: Remove
|
remove: Remove
|
||||||
reset: Reset
|
reset: Reset
|
||||||
close: Close
|
close: Close
|
||||||
|
@ -109,6 +110,7 @@ globals:
|
||||||
now: Now
|
now: Now
|
||||||
name: Name
|
name: Name
|
||||||
new: New
|
new: New
|
||||||
|
comment: Comment
|
||||||
errors:
|
errors:
|
||||||
statusUnauthorized: Access denied
|
statusUnauthorized: Access denied
|
||||||
statusInternalServerError: An internal server error has ocurred
|
statusInternalServerError: An internal server error has ocurred
|
||||||
|
|
|
@ -25,6 +25,7 @@ globals:
|
||||||
create: Crear
|
create: Crear
|
||||||
edit: Modificar
|
edit: Modificar
|
||||||
save: Guardar
|
save: Guardar
|
||||||
|
saveAndContinue: Guardar y continuar
|
||||||
remove: Eliminar
|
remove: Eliminar
|
||||||
reset: Restaurar
|
reset: Restaurar
|
||||||
close: Cerrar
|
close: Cerrar
|
||||||
|
@ -109,6 +110,7 @@ globals:
|
||||||
now: Ahora
|
now: Ahora
|
||||||
name: Nombre
|
name: Nombre
|
||||||
new: Nuevo
|
new: Nuevo
|
||||||
|
comment: Comentario
|
||||||
errors:
|
errors:
|
||||||
statusUnauthorized: Acceso denegado
|
statusUnauthorized: Acceso denegado
|
||||||
statusInternalServerError: Ha ocurrido un error interno del servidor
|
statusInternalServerError: Ha ocurrido un error interno del servidor
|
||||||
|
|
|
@ -9,18 +9,21 @@ import { downloadFile } from 'src/composables/downloadFile';
|
||||||
import FormModel from 'components/FormModel.vue';
|
import FormModel from 'components/FormModel.vue';
|
||||||
import VnSelect from 'src/components/common/VnSelect.vue';
|
import VnSelect from 'src/components/common/VnSelect.vue';
|
||||||
import FetchData from 'src/components/FetchData.vue';
|
import FetchData from 'src/components/FetchData.vue';
|
||||||
import VnRow from 'src/components/ui/VnRow.vue';
|
import VnRow from 'components/ui/VnRow.vue';
|
||||||
|
import VnInputDate from 'src/components/common/VnInputDate.vue';
|
||||||
|
import VnInput from 'src/components/common/VnInput.vue';
|
||||||
|
|
||||||
const quasar = useQuasar();
|
const quasar = useQuasar();
|
||||||
const route = useRoute();
|
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
|
|
||||||
const dms = ref({});
|
const dms = ref({});
|
||||||
|
const route = useRoute();
|
||||||
const editDownloadDisabled = ref(false);
|
const editDownloadDisabled = ref(false);
|
||||||
const arrayData = useArrayData('InvoiceIn');
|
const invoiceIn = computed(() => useArrayData(route.meta.moduleName).store.data);
|
||||||
const invoiceIn = computed(() => arrayData.store.data);
|
|
||||||
const userConfig = ref(null);
|
const userConfig = ref(null);
|
||||||
|
const invoiceId = computed(() => +route.params.id);
|
||||||
|
|
||||||
|
const expenses = ref([]);
|
||||||
const currencies = ref([]);
|
const currencies = ref([]);
|
||||||
const currenciesRef = ref();
|
const currenciesRef = ref();
|
||||||
const companies = ref([]);
|
const companies = ref([]);
|
||||||
|
@ -31,14 +34,11 @@ const warehouses = ref([]);
|
||||||
const warehousesRef = ref();
|
const warehousesRef = ref();
|
||||||
const allowTypesRef = ref();
|
const allowTypesRef = ref();
|
||||||
const allowedContentTypes = ref([]);
|
const allowedContentTypes = ref([]);
|
||||||
|
const sageWithholdings = ref([]);
|
||||||
const inputFileRef = ref();
|
const inputFileRef = ref();
|
||||||
const editDmsRef = ref();
|
const editDmsRef = ref();
|
||||||
const createDmsRef = ref();
|
const createDmsRef = ref();
|
||||||
|
|
||||||
const requiredFieldRule = (val) => val || t('globals.requiredField');
|
|
||||||
const dateMask = '####-##-##';
|
|
||||||
const fillMask = '_';
|
|
||||||
|
|
||||||
async function checkFileExists(dmsId) {
|
async function checkFileExists(dmsId) {
|
||||||
if (!dmsId) return;
|
if (!dmsId) return;
|
||||||
try {
|
try {
|
||||||
|
@ -173,11 +173,17 @@ async function upsert() {
|
||||||
@on-fetch="(data) => (userConfig = data)"
|
@on-fetch="(data) => (userConfig = data)"
|
||||||
auto-load
|
auto-load
|
||||||
/>
|
/>
|
||||||
|
<FetchData url="Expenses" auto-load @on-fetch="(data) => (expenses = data)" />
|
||||||
|
<FetchData
|
||||||
|
url="SageWithholdings"
|
||||||
|
auto-load
|
||||||
|
@on-fetch="(data) => (sageWithholdings = data)"
|
||||||
|
/>
|
||||||
<FormModel
|
<FormModel
|
||||||
v-if="invoiceIn"
|
model="InvoiceIn"
|
||||||
:url="`InvoiceIns/${route.params.id}`"
|
:go-to="`/invoice-in/${invoiceId}/vat`"
|
||||||
model="invoiceIn"
|
auto-load
|
||||||
:auto-load="true"
|
:url-update="`InvoiceIns/${invoiceId}/updateInvoiceIn`"
|
||||||
>
|
>
|
||||||
<template #form="{ data }">
|
<template #form="{ data }">
|
||||||
<VnRow>
|
<VnRow>
|
||||||
|
@ -201,7 +207,7 @@ async function upsert() {
|
||||||
</QItem>
|
</QItem>
|
||||||
</template>
|
</template>
|
||||||
</VnSelect>
|
</VnSelect>
|
||||||
<QInput
|
<VnInput
|
||||||
clearable
|
clearable
|
||||||
clear-icon="close"
|
clear-icon="close"
|
||||||
:label="t('Supplier ref')"
|
:label="t('Supplier ref')"
|
||||||
|
@ -209,69 +215,29 @@ async function upsert() {
|
||||||
/>
|
/>
|
||||||
</VnRow>
|
</VnRow>
|
||||||
<VnRow>
|
<VnRow>
|
||||||
<QInput
|
<VnInputDate :label="t('Expedition date')" v-model="data.issued" />
|
||||||
:label="t('Expedition date')"
|
<VnInputDate
|
||||||
v-model="data.issued"
|
|
||||||
:mask="dateMask"
|
|
||||||
>
|
|
||||||
<template #append>
|
|
||||||
<QIcon name="event" class="cursor-pointer" :fill-mask="fillMask">
|
|
||||||
<QPopupProxy
|
|
||||||
cover
|
|
||||||
transition-show="scale"
|
|
||||||
transition-hide="scale"
|
|
||||||
>
|
|
||||||
<QDate v-model="data.issued">
|
|
||||||
<div class="row items-center justify-end">
|
|
||||||
<QBtn
|
|
||||||
v-close-popup
|
|
||||||
label="Close"
|
|
||||||
color="primary"
|
|
||||||
flat
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</QDate>
|
|
||||||
</QPopupProxy>
|
|
||||||
</QIcon>
|
|
||||||
</template>
|
|
||||||
</QInput>
|
|
||||||
<QInput
|
|
||||||
:label="t('Operation date')"
|
:label="t('Operation date')"
|
||||||
v-model="data.operated"
|
v-model="data.operated"
|
||||||
:mask="dateMask"
|
|
||||||
:fill-mask="fillMask"
|
|
||||||
autofocus
|
autofocus
|
||||||
>
|
|
||||||
<template #append>
|
|
||||||
<QIcon name="event" class="cursor-pointer">
|
|
||||||
<QPopupProxy
|
|
||||||
cover
|
|
||||||
transition-show="scale"
|
|
||||||
transition-hide="scale"
|
|
||||||
>
|
|
||||||
<QDate v-model="data.operated" :mask="dateMask">
|
|
||||||
<div class="row items-center justify-end">
|
|
||||||
<QBtn
|
|
||||||
v-close-popup
|
|
||||||
label="Close"
|
|
||||||
color="primary"
|
|
||||||
flat
|
|
||||||
/>
|
/>
|
||||||
</div>
|
|
||||||
</QDate>
|
|
||||||
</QPopupProxy>
|
|
||||||
</QIcon>
|
|
||||||
</template>
|
|
||||||
</QInput>
|
|
||||||
</VnRow>
|
</VnRow>
|
||||||
<VnRow>
|
<VnRow>
|
||||||
<QInput
|
<VnSelect
|
||||||
:label="t('Undeductible VAT')"
|
:label="t('Undeductible VAT')"
|
||||||
v-model="data.deductibleExpenseFk"
|
v-model="data.deductibleExpenseFk"
|
||||||
clearable
|
:options="expenses"
|
||||||
clear-icon="close"
|
option-value="id"
|
||||||
/>
|
option-label="id"
|
||||||
<QInput
|
:filter-options="['id', 'name']"
|
||||||
|
>
|
||||||
|
<template #option="scope">
|
||||||
|
<QItem v-bind="scope.itemProps">
|
||||||
|
{{ `${scope.opt.id}: ${scope.opt.name}` }}
|
||||||
|
</QItem>
|
||||||
|
</template>
|
||||||
|
</VnSelect>
|
||||||
|
<VnInput
|
||||||
:label="t('Document')"
|
:label="t('Document')"
|
||||||
v-model="data.dmsFk"
|
v-model="data.dmsFk"
|
||||||
clearable
|
clearable
|
||||||
|
@ -316,67 +282,11 @@ async function upsert() {
|
||||||
<QTooltip>{{ t('Create document') }}</QTooltip>
|
<QTooltip>{{ t('Create document') }}</QTooltip>
|
||||||
</QBtn>
|
</QBtn>
|
||||||
</template>
|
</template>
|
||||||
</QInput>
|
</VnInput>
|
||||||
</VnRow>
|
</VnRow>
|
||||||
<VnRow>
|
<VnRow>
|
||||||
<QInput
|
<VnInputDate :label="t('Entry date')" v-model="data.bookEntried" />
|
||||||
:label="t('Entry date')"
|
<VnInputDate :label="t('Accounted date')" v-model="data.booked" />
|
||||||
v-model="data.bookEntried"
|
|
||||||
clearable
|
|
||||||
clear-icon="close"
|
|
||||||
:mask="dateMask"
|
|
||||||
:fill-mask="fillMask"
|
|
||||||
>
|
|
||||||
<template #append>
|
|
||||||
<QIcon name="event" class="cursor-pointer">
|
|
||||||
<QPopupProxy
|
|
||||||
cover
|
|
||||||
transition-show="scale"
|
|
||||||
transition-hide="scale"
|
|
||||||
>
|
|
||||||
<QDate v-model="data.bookEntried" :mask="dateMask">
|
|
||||||
<div class="row items-center justify-end">
|
|
||||||
<QBtn
|
|
||||||
v-close-popup
|
|
||||||
label="Close"
|
|
||||||
color="primary"
|
|
||||||
flat
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</QDate>
|
|
||||||
</QPopupProxy>
|
|
||||||
</QIcon>
|
|
||||||
</template>
|
|
||||||
</QInput>
|
|
||||||
<QInput
|
|
||||||
:label="t('Accounted date')"
|
|
||||||
v-model="data.booked"
|
|
||||||
clearable
|
|
||||||
clear-icon="close"
|
|
||||||
:mask="dateMask"
|
|
||||||
:fill-mask="fillMask"
|
|
||||||
>
|
|
||||||
<template #append>
|
|
||||||
<QIcon name="event" class="cursor-pointer">
|
|
||||||
<QPopupProxy
|
|
||||||
cover
|
|
||||||
transition-show="scale"
|
|
||||||
transition-hide="scale"
|
|
||||||
>
|
|
||||||
<QDate v-model="data.booked" :mask="maskDate">
|
|
||||||
<div class="row items-center justify-end">
|
|
||||||
<QBtn
|
|
||||||
v-close-popup
|
|
||||||
label="Close"
|
|
||||||
color="primary"
|
|
||||||
flat
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</QDate>
|
|
||||||
</QPopupProxy>
|
|
||||||
</QIcon>
|
|
||||||
</template>
|
|
||||||
</QInput>
|
|
||||||
</VnRow>
|
</VnRow>
|
||||||
<VnRow>
|
<VnRow>
|
||||||
<VnSelect
|
<VnSelect
|
||||||
|
@ -386,6 +296,7 @@ async function upsert() {
|
||||||
option-value="id"
|
option-value="id"
|
||||||
option-label="code"
|
option-label="code"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<VnSelect
|
<VnSelect
|
||||||
v-if="companiesRef"
|
v-if="companiesRef"
|
||||||
:label="t('Company')"
|
:label="t('Company')"
|
||||||
|
@ -395,7 +306,15 @@ async function upsert() {
|
||||||
option-label="code"
|
option-label="code"
|
||||||
/>
|
/>
|
||||||
</VnRow>
|
</VnRow>
|
||||||
<QCheckbox :label="t('invoiceIn.summary.booked')" v-model="data.isBooked" />
|
<VnRow>
|
||||||
|
<VnSelect
|
||||||
|
:label="t('invoiceIn.summary.sage')"
|
||||||
|
v-model="data.withholdingSageFk"
|
||||||
|
:options="sageWithholdings"
|
||||||
|
option-value="id"
|
||||||
|
option-label="withholding"
|
||||||
|
/>
|
||||||
|
</VnRow>
|
||||||
</template>
|
</template>
|
||||||
</FormModel>
|
</FormModel>
|
||||||
<QDialog ref="editDmsRef">
|
<QDialog ref="editDmsRef">
|
||||||
|
@ -411,7 +330,7 @@ async function upsert() {
|
||||||
</QCardSection>
|
</QCardSection>
|
||||||
<QCardSection class="q-py-none">
|
<QCardSection class="q-py-none">
|
||||||
<QItem>
|
<QItem>
|
||||||
<QInput
|
<VnInput
|
||||||
class="full-width q-pa-xs"
|
class="full-width q-pa-xs"
|
||||||
:label="t('Reference')"
|
:label="t('Reference')"
|
||||||
v-model="dms.reference"
|
v-model="dms.reference"
|
||||||
|
@ -420,45 +339,45 @@ async function upsert() {
|
||||||
/>
|
/>
|
||||||
<VnSelect
|
<VnSelect
|
||||||
class="full-width q-pa-xs"
|
class="full-width q-pa-xs"
|
||||||
:label="`${t('Company')}*`"
|
:label="t('Company')"
|
||||||
v-model="dms.companyId"
|
v-model="dms.companyId"
|
||||||
:options="companies"
|
:options="companies"
|
||||||
option-value="id"
|
option-value="id"
|
||||||
option-label="code"
|
option-label="code"
|
||||||
:rules="[requiredFieldRule]"
|
:required="true"
|
||||||
/>
|
/>
|
||||||
</QItem>
|
</QItem>
|
||||||
<QItem>
|
<QItem>
|
||||||
<VnSelect
|
<VnSelect
|
||||||
class="full-width q-pa-xs"
|
class="full-width q-pa-xs"
|
||||||
:label="`${t('Warehouse')}*`"
|
:label="t('Warehouse')"
|
||||||
v-model="dms.warehouseId"
|
v-model="dms.warehouseId"
|
||||||
:options="warehouses"
|
:options="warehouses"
|
||||||
option-value="id"
|
option-value="id"
|
||||||
option-label="name"
|
option-label="name"
|
||||||
:rules="[requiredFieldRule]"
|
:required="true"
|
||||||
/>
|
/>
|
||||||
<VnSelect
|
<VnSelect
|
||||||
class="full-width q-pa-xs"
|
class="full-width q-pa-xs"
|
||||||
:label="`${t('Type')}*`"
|
:label="t('Type')"
|
||||||
v-model="dms.dmsTypeId"
|
v-model="dms.dmsTypeId"
|
||||||
:options="dmsTypes"
|
:options="dmsTypes"
|
||||||
option-value="id"
|
option-value="id"
|
||||||
option-label="name"
|
option-label="name"
|
||||||
:rules="[requiredFieldRule]"
|
:required="true"
|
||||||
/>
|
/>
|
||||||
</QItem>
|
</QItem>
|
||||||
<QItem>
|
<QItem>
|
||||||
<QInput
|
<VnInput
|
||||||
class="full-width q-pa-xs"
|
:label="t('Description')"
|
||||||
|
v-model="dms.description"
|
||||||
|
:required="true"
|
||||||
type="textarea"
|
type="textarea"
|
||||||
|
class="full-width q-pa-xs"
|
||||||
size="lg"
|
size="lg"
|
||||||
autogrow
|
autogrow
|
||||||
:label="`${t('Description')}*`"
|
|
||||||
v-model="dms.description"
|
|
||||||
clearable
|
clearable
|
||||||
clear-icon="close"
|
clear-icon="close"
|
||||||
:rules="[(val) => val.length || t('Required field')]"
|
|
||||||
/>
|
/>
|
||||||
</QItem>
|
</QItem>
|
||||||
<QItem>
|
<QItem>
|
||||||
|
@ -522,7 +441,7 @@ async function upsert() {
|
||||||
</QCardSection>
|
</QCardSection>
|
||||||
<QCardSection class="q-pb-none">
|
<QCardSection class="q-pb-none">
|
||||||
<QItem>
|
<QItem>
|
||||||
<QInput
|
<VnInput
|
||||||
class="full-width q-pa-xs"
|
class="full-width q-pa-xs"
|
||||||
:label="t('Reference')"
|
:label="t('Reference')"
|
||||||
v-model="dms.reference"
|
v-model="dms.reference"
|
||||||
|
@ -534,7 +453,7 @@ async function upsert() {
|
||||||
:options="companies"
|
:options="companies"
|
||||||
option-value="id"
|
option-value="id"
|
||||||
option-label="code"
|
option-label="code"
|
||||||
:rules="[requiredFieldRule]"
|
:required="true"
|
||||||
/>
|
/>
|
||||||
</QItem>
|
</QItem>
|
||||||
<QItem>
|
<QItem>
|
||||||
|
@ -545,7 +464,7 @@ async function upsert() {
|
||||||
:options="warehouses"
|
:options="warehouses"
|
||||||
option-value="id"
|
option-value="id"
|
||||||
option-label="name"
|
option-label="name"
|
||||||
:rules="[requiredFieldRule]"
|
:required="true"
|
||||||
/>
|
/>
|
||||||
<VnSelect
|
<VnSelect
|
||||||
class="full-width q-pa-xs"
|
class="full-width q-pa-xs"
|
||||||
|
@ -554,11 +473,11 @@ async function upsert() {
|
||||||
:options="dmsTypes"
|
:options="dmsTypes"
|
||||||
option-value="id"
|
option-value="id"
|
||||||
option-label="name"
|
option-label="name"
|
||||||
:rules="[requiredFieldRule]"
|
:required="true"
|
||||||
/>
|
/>
|
||||||
</QItem>
|
</QItem>
|
||||||
<QItem>
|
<QItem>
|
||||||
<QInput
|
<VnInput
|
||||||
class="full-width q-pa-xs"
|
class="full-width q-pa-xs"
|
||||||
type="textarea"
|
type="textarea"
|
||||||
size="lg"
|
size="lg"
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
<script setup>
|
<script setup>
|
||||||
import { ref, computed } from 'vue';
|
import { ref, computed } from 'vue';
|
||||||
import { useRoute, useRouter } from 'vue-router';
|
import { useRouter } from 'vue-router';
|
||||||
import { useI18n } from 'vue-i18n';
|
import { useI18n } from 'vue-i18n';
|
||||||
import { useArrayData } from 'src/composables/useArrayData';
|
import { useArrayData } from 'src/composables/useArrayData';
|
||||||
import { useCapitalize } from 'src/composables/useCapitalize';
|
import { useCapitalize } from 'src/composables/useCapitalize';
|
||||||
|
@ -8,12 +8,11 @@ import CrudModel from 'src/components/CrudModel.vue';
|
||||||
import FetchData from 'src/components/FetchData.vue';
|
import FetchData from 'src/components/FetchData.vue';
|
||||||
import VnSelect from 'src/components/common/VnSelect.vue';
|
import VnSelect from 'src/components/common/VnSelect.vue';
|
||||||
|
|
||||||
const router = useRouter();
|
const { push, currentRoute } = useRouter();
|
||||||
const route = useRoute();
|
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
|
|
||||||
const invoiceId = route.params.id;
|
const invoiceId = +currentRoute.value.params.id;
|
||||||
const arrayData = useArrayData('InvoiceIn');
|
const arrayData = useArrayData(currentRoute.value.meta.moduleName);
|
||||||
const invoiceIn = computed(() => arrayData.store.data);
|
const invoiceIn = computed(() => arrayData.store.data);
|
||||||
const invoiceInCorrectionRef = ref();
|
const invoiceInCorrectionRef = ref();
|
||||||
const filter = {
|
const filter = {
|
||||||
|
@ -74,7 +73,7 @@ const rowsSelected = ref([]);
|
||||||
|
|
||||||
const requiredFieldRule = (val) => val || t('globals.requiredField');
|
const requiredFieldRule = (val) => val || t('globals.requiredField');
|
||||||
|
|
||||||
const onSave = (data) => data.deletes && router.push(`/invoice-in/${invoiceId}/summary`);
|
const onSave = (data) => data.deletes && push(`/invoice-in/${invoiceId}/summary`);
|
||||||
</script>
|
</script>
|
||||||
<template>
|
<template>
|
||||||
<FetchData
|
<FetchData
|
||||||
|
|
|
@ -1,12 +1,11 @@
|
||||||
<script setup>
|
<script setup>
|
||||||
import { ref, reactive, computed, onBeforeMount, watch } from 'vue';
|
import { ref, reactive, computed, onBeforeMount } from 'vue';
|
||||||
import { useRoute, useRouter } from 'vue-router';
|
import { useRouter, onBeforeRouteLeave } from 'vue-router';
|
||||||
import { useI18n } from 'vue-i18n';
|
import { useI18n } from 'vue-i18n';
|
||||||
import { useQuasar } from 'quasar';
|
import { useQuasar } from 'quasar';
|
||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
import { toCurrency, toDate } from 'src/filters';
|
import { toCurrency, toDate } from 'src/filters';
|
||||||
import { useRole } from 'src/composables/useRole';
|
import { useRole } from 'src/composables/useRole';
|
||||||
import useCardDescription from 'src/composables/useCardDescription';
|
|
||||||
import { downloadFile } from 'src/composables/downloadFile';
|
import { downloadFile } from 'src/composables/downloadFile';
|
||||||
import { useArrayData } from 'src/composables/useArrayData';
|
import { useArrayData } from 'src/composables/useArrayData';
|
||||||
import { usePrintService } from 'composables/usePrintService';
|
import { usePrintService } from 'composables/usePrintService';
|
||||||
|
@ -17,27 +16,23 @@ import SendEmailDialog from 'components/common/SendEmailDialog.vue';
|
||||||
import VnConfirm from 'src/components/ui/VnConfirm.vue';
|
import VnConfirm from 'src/components/ui/VnConfirm.vue';
|
||||||
import VnSelect from 'src/components/common/VnSelect.vue';
|
import VnSelect from 'src/components/common/VnSelect.vue';
|
||||||
import { useCapitalize } from 'src/composables/useCapitalize';
|
import { useCapitalize } from 'src/composables/useCapitalize';
|
||||||
|
import SupplierDescriptorProxy from 'src/pages/Supplier/Card/SupplierDescriptorProxy.vue';
|
||||||
|
import InvoiceInToBook from '../InvoiceInToBook.vue';
|
||||||
|
|
||||||
const $props = defineProps({
|
const $props = defineProps({ id: { type: Number, default: null } });
|
||||||
id: {
|
|
||||||
type: Number,
|
const { push, currentRoute } = useRouter();
|
||||||
required: false,
|
|
||||||
default: null,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const route = useRoute();
|
|
||||||
const router = useRouter();
|
|
||||||
const quasar = useQuasar();
|
const quasar = useQuasar();
|
||||||
const { hasAny } = useRole();
|
const { hasAny } = useRole();
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
const { openReport, sendEmail } = usePrintService();
|
const { openReport, sendEmail } = usePrintService();
|
||||||
const arrayData = useArrayData('InvoiceIn');
|
const { store } = useArrayData(currentRoute.value.meta.moduleName);
|
||||||
|
|
||||||
const invoiceIn = computed(() => arrayData.store.data);
|
const invoiceIn = computed(() => store.data);
|
||||||
const cardDescriptorRef = ref();
|
const cardDescriptorRef = ref();
|
||||||
const correctionDialogRef = ref();
|
const correctionDialogRef = ref();
|
||||||
const entityId = computed(() => $props.id || +route.params.id);
|
const entityId = computed(() => $props.id || +currentRoute.value.params.id);
|
||||||
const totalAmount = ref();
|
const totalAmount = ref();
|
||||||
const currentAction = ref();
|
const currentAction = ref();
|
||||||
const config = ref();
|
const config = ref();
|
||||||
|
@ -45,28 +40,21 @@ const cplusRectificationTypes = ref([]);
|
||||||
const siiTypeInvoiceOuts = ref([]);
|
const siiTypeInvoiceOuts = ref([]);
|
||||||
const invoiceCorrectionTypes = ref([]);
|
const invoiceCorrectionTypes = ref([]);
|
||||||
const actions = {
|
const actions = {
|
||||||
book: {
|
unbook: {
|
||||||
title: 'Are you sure you want to book this invoice?',
|
title: t('assertAction', { action: t('unbook') }),
|
||||||
cb: checkToBook,
|
action: toUnbook,
|
||||||
action: toBook,
|
|
||||||
},
|
},
|
||||||
delete: {
|
delete: {
|
||||||
title: 'Are you sure you want to delete this invoice?',
|
title: t('assertAction', { action: t('delete') }),
|
||||||
action: deleteInvoice,
|
action: deleteInvoice,
|
||||||
},
|
},
|
||||||
clone: {
|
clone: {
|
||||||
title: 'Are you sure you want to clone this invoice?',
|
title: t('assertAction', { action: t('clone') }),
|
||||||
action: cloneInvoice,
|
action: cloneInvoice,
|
||||||
},
|
},
|
||||||
showPdf: {
|
showPdf: { cb: showPdfInvoice },
|
||||||
cb: showPdfInvoice,
|
sendPdf: { cb: sendPdfInvoiceConfirmation },
|
||||||
},
|
correct: { cb: () => correctionDialogRef.value.show() },
|
||||||
sendPdf: {
|
|
||||||
cb: sendPdfInvoiceConfirmation,
|
|
||||||
},
|
|
||||||
correct: {
|
|
||||||
cb: () => correctionDialogRef.value.show(),
|
|
||||||
},
|
|
||||||
};
|
};
|
||||||
const filter = {
|
const filter = {
|
||||||
include: [
|
include: [
|
||||||
|
@ -94,11 +82,7 @@ const filter = {
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
};
|
};
|
||||||
const data = ref(useCardDescription());
|
const invoiceInCorrection = reactive({ correcting: [], corrected: null });
|
||||||
const invoiceInCorrection = reactive({
|
|
||||||
correcting: [],
|
|
||||||
corrected: null,
|
|
||||||
});
|
|
||||||
const routes = reactive({
|
const routes = reactive({
|
||||||
getSupplier: (id) => {
|
getSupplier: (id) => {
|
||||||
return { name: 'SupplierCard', params: { id } };
|
return { name: 'SupplierCard', params: { id } };
|
||||||
|
@ -139,16 +123,17 @@ const correctionFormData = reactive({
|
||||||
});
|
});
|
||||||
const isNotFilled = computed(() => Object.values(correctionFormData).includes(null));
|
const isNotFilled = computed(() => Object.values(correctionFormData).includes(null));
|
||||||
|
|
||||||
onBeforeMount(async () => await setInvoiceCorrection(entityId.value));
|
onBeforeMount(async () => {
|
||||||
|
await setInvoiceCorrection(entityId.value);
|
||||||
|
const { data } = await axios.get(`InvoiceIns/${entityId.value}/getTotals`);
|
||||||
|
totalAmount.value = data.totalDueDay;
|
||||||
|
});
|
||||||
|
|
||||||
watch(
|
onBeforeRouteLeave(async (to, from) => {
|
||||||
() => route.params.id,
|
|
||||||
async (newId) => {
|
|
||||||
invoiceInCorrection.correcting.length = 0;
|
invoiceInCorrection.correcting.length = 0;
|
||||||
invoiceInCorrection.corrected = null;
|
invoiceInCorrection.corrected = null;
|
||||||
if (newId) await setInvoiceCorrection(entityId.value);
|
if (to.params.id !== from.params.id) await setInvoiceCorrection(entityId.value);
|
||||||
}
|
});
|
||||||
);
|
|
||||||
|
|
||||||
async function setInvoiceCorrection(id) {
|
async function setInvoiceCorrection(id) {
|
||||||
const [{ data: correctingData }, { data: correctedData }] = await Promise.all([
|
const [{ data: correctingData }, { data: correctedData }] = await Promise.all([
|
||||||
|
@ -179,17 +164,6 @@ async function setInvoiceCorrection(id) {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
async function setData(entity) {
|
|
||||||
data.value = useCardDescription(entity.supplierRef, entity.id);
|
|
||||||
const { totalDueDay } = await getTotals();
|
|
||||||
totalAmount.value = totalDueDay;
|
|
||||||
}
|
|
||||||
|
|
||||||
async function getTotals() {
|
|
||||||
const { data } = await axios.get(`InvoiceIns/${entityId.value}/getTotals`);
|
|
||||||
return data;
|
|
||||||
}
|
|
||||||
|
|
||||||
function openDialog() {
|
function openDialog() {
|
||||||
quasar.dialog({
|
quasar.dialog({
|
||||||
component: VnConfirm,
|
component: VnConfirm,
|
||||||
|
@ -200,38 +174,17 @@ function openDialog() {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
async function checkToBook() {
|
async function toUnbook() {
|
||||||
let directBooking = true;
|
const { data } = await axios.post(`InvoiceIns/${entityId.value}/toUnbook`);
|
||||||
|
const { isLinked, bookEntry, accountingEntries } = data;
|
||||||
|
|
||||||
const totals = await getTotals();
|
const type = isLinked ? 'warning' : 'positive';
|
||||||
const taxableBaseNotEqualDueDay = totals.totalDueDay != totals.totalTaxableBase;
|
const message = isLinked
|
||||||
const vatNotEqualDueDay = totals.totalDueDay != totals.totalVat;
|
? t('isLinked', { bookEntry, accountingEntries })
|
||||||
|
: t('isNotLinked', { bookEntry });
|
||||||
|
|
||||||
if (taxableBaseNotEqualDueDay && vatNotEqualDueDay) directBooking = false;
|
quasar.notify({ type, message });
|
||||||
|
if (!isLinked) store.data.isBooked = false;
|
||||||
const { data: dueDaysCount } = await axios.get('InvoiceInDueDays/count', {
|
|
||||||
where: {
|
|
||||||
invoiceInFk: entityId.value,
|
|
||||||
dueDated: { gte: Date.vnNew() },
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
if (dueDaysCount) directBooking = false;
|
|
||||||
|
|
||||||
if (!directBooking) openDialog();
|
|
||||||
else toBook();
|
|
||||||
}
|
|
||||||
|
|
||||||
async function toBook() {
|
|
||||||
await axios.post(`InvoiceIns/${entityId.value}/toBook`);
|
|
||||||
|
|
||||||
quasar.notify({
|
|
||||||
type: 'positive',
|
|
||||||
message: t('globals.dataSaved'),
|
|
||||||
});
|
|
||||||
|
|
||||||
await cardDescriptorRef.value.getData();
|
|
||||||
setTimeout(() => location.reload(), 500);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async function deleteInvoice() {
|
async function deleteInvoice() {
|
||||||
|
@ -240,7 +193,7 @@ async function deleteInvoice() {
|
||||||
type: 'positive',
|
type: 'positive',
|
||||||
message: t('Invoice deleted'),
|
message: t('Invoice deleted'),
|
||||||
});
|
});
|
||||||
router.push({ path: '/invoice-in' });
|
push({ path: '/invoice-in' });
|
||||||
}
|
}
|
||||||
|
|
||||||
async function cloneInvoice() {
|
async function cloneInvoice() {
|
||||||
|
@ -249,11 +202,9 @@ async function cloneInvoice() {
|
||||||
type: 'positive',
|
type: 'positive',
|
||||||
message: t('Invoice cloned'),
|
message: t('Invoice cloned'),
|
||||||
});
|
});
|
||||||
router.push({ path: `/invoice-in/${data.id}/summary` });
|
push({ path: `/invoice-in/${data.id}/summary` });
|
||||||
}
|
}
|
||||||
|
|
||||||
const requiredFieldRule = (val) => val || t('globals.requiredField');
|
|
||||||
|
|
||||||
const isAdministrative = () => hasAny(['administrative']);
|
const isAdministrative = () => hasAny(['administrative']);
|
||||||
|
|
||||||
const isAgricultural = () =>
|
const isAgricultural = () =>
|
||||||
|
@ -299,10 +250,9 @@ const createInvoiceInCorrection = async () => {
|
||||||
'InvoiceIns/corrective',
|
'InvoiceIns/corrective',
|
||||||
Object.assign(correctionFormData, { id: entityId.value })
|
Object.assign(correctionFormData, { id: entityId.value })
|
||||||
);
|
);
|
||||||
router.push({ path: `/invoice-in/${correctingId}/summary` });
|
push({ path: `/invoice-in/${correctingId}/summary` });
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<FetchData
|
<FetchData
|
||||||
url="InvoiceInConfigs"
|
url="InvoiceInConfigs"
|
||||||
|
@ -329,22 +279,34 @@ const createInvoiceInCorrection = async () => {
|
||||||
<CardDescriptor
|
<CardDescriptor
|
||||||
ref="cardDescriptorRef"
|
ref="cardDescriptorRef"
|
||||||
module="InvoiceIn"
|
module="InvoiceIn"
|
||||||
|
data-key="InvoiceIn"
|
||||||
:url="`InvoiceIns/${entityId}`"
|
:url="`InvoiceIns/${entityId}`"
|
||||||
:filter="filter"
|
:filter="filter"
|
||||||
:title="data.title"
|
title="supplierRef"
|
||||||
:subtitle="data.subtitle"
|
|
||||||
@on-fetch="setData"
|
|
||||||
data-key="invoiceInData"
|
|
||||||
>
|
>
|
||||||
<template #menu="{ entity }">
|
<template #menu="{ entity }">
|
||||||
|
<InvoiceInToBook>
|
||||||
|
<template #content="{ book }">
|
||||||
<QItem
|
<QItem
|
||||||
v-if="!entity.isBooked && isAdministrative()"
|
v-if="!entity?.isBooked && isAdministrative()"
|
||||||
v-ripple
|
v-ripple
|
||||||
clickable
|
clickable
|
||||||
@click="triggerMenu('book')"
|
@click="book(entityId)"
|
||||||
>
|
>
|
||||||
<QItemSection>{{ t('To book') }}</QItemSection>
|
<QItemSection>{{ t('To book') }}</QItemSection>
|
||||||
</QItem>
|
</QItem>
|
||||||
|
</template>
|
||||||
|
</InvoiceInToBook>
|
||||||
|
<QItem
|
||||||
|
v-if="entity?.isBooked && isAdministrative()"
|
||||||
|
v-ripple
|
||||||
|
clickable
|
||||||
|
@click="triggerMenu('unbook')"
|
||||||
|
>
|
||||||
|
<QItemSection>
|
||||||
|
{{ t('To unbook') }}
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
<QItem
|
<QItem
|
||||||
v-if="isAdministrative()"
|
v-if="isAdministrative()"
|
||||||
v-ripple
|
v-ripple
|
||||||
|
@ -395,25 +357,24 @@ const createInvoiceInCorrection = async () => {
|
||||||
>
|
>
|
||||||
<QItemSection>{{ t('components.smartCard.downloadFile') }}</QItemSection>
|
<QItemSection>{{ t('components.smartCard.downloadFile') }}</QItemSection>
|
||||||
</QItem>
|
</QItem>
|
||||||
<QItem
|
|
||||||
v-if="entity.dmsFk"
|
|
||||||
v-ripple
|
|
||||||
clickable
|
|
||||||
@click="downloadFile(entity.dmsFk)"
|
|
||||||
>
|
|
||||||
<QItemSection>{{ t('components.smartCard.downloadFile') }}</QItemSection>
|
|
||||||
</QItem>
|
|
||||||
</template>
|
</template>
|
||||||
<template #body="{ entity }">
|
<template #body="{ entity }">
|
||||||
<VnLv :label="t('invoiceIn.card.issued')" :value="toDate(entity.issued)" />
|
<VnLv :label="t('invoiceIn.card.issued')" :value="toDate(entity.issued)" />
|
||||||
<VnLv :label="t('invoiceIn.summary.booked')" :value="toDate(entity.booked)" />
|
<VnLv :label="t('invoiceIn.summary.booked')" :value="toDate(entity.booked)" />
|
||||||
<VnLv :label="t('invoiceIn.card.amount')" :value="toCurrency(totalAmount)" />
|
|
||||||
<VnLv
|
<VnLv
|
||||||
:label="t('invoiceIn.summary.supplier')"
|
:label="t('invoiceIn.card.amount')"
|
||||||
:value="entity.supplier?.nickname"
|
:value="toCurrency(totalAmount, entity.currency?.code)"
|
||||||
/>
|
/>
|
||||||
|
<VnLv :label="t('invoiceIn.summary.supplier')">
|
||||||
|
<template #value>
|
||||||
|
<span class="link">
|
||||||
|
{{ entity?.supplier?.nickname }}
|
||||||
|
<SupplierDescriptorProxy :id="entity?.supplierFk" />
|
||||||
|
</span>
|
||||||
</template>
|
</template>
|
||||||
<template #actions="{ entity }">
|
</VnLv>
|
||||||
|
</template>
|
||||||
|
<template #action="{ entity }">
|
||||||
<QCardActions>
|
<QCardActions>
|
||||||
<QBtn
|
<QBtn
|
||||||
size="md"
|
size="md"
|
||||||
|
@ -486,7 +447,7 @@ const createInvoiceInCorrection = async () => {
|
||||||
:options="siiTypeInvoiceOuts"
|
:options="siiTypeInvoiceOuts"
|
||||||
option-value="id"
|
option-value="id"
|
||||||
option-label="code"
|
option-label="code"
|
||||||
:rules="[requiredFieldRule]"
|
:required="true"
|
||||||
/>
|
/>
|
||||||
</QItemSection>
|
</QItemSection>
|
||||||
<QItemSection>
|
<QItemSection>
|
||||||
|
@ -496,7 +457,7 @@ const createInvoiceInCorrection = async () => {
|
||||||
:options="cplusRectificationTypes"
|
:options="cplusRectificationTypes"
|
||||||
option-value="id"
|
option-value="id"
|
||||||
option-label="description"
|
option-label="description"
|
||||||
:rules="[requiredFieldRule]"
|
:required="true"
|
||||||
/>
|
/>
|
||||||
<VnSelect
|
<VnSelect
|
||||||
:label="`${useCapitalize(t('globals.reason'))}*`"
|
:label="`${useCapitalize(t('globals.reason'))}*`"
|
||||||
|
@ -504,7 +465,7 @@ const createInvoiceInCorrection = async () => {
|
||||||
:options="invoiceCorrectionTypes"
|
:options="invoiceCorrectionTypes"
|
||||||
option-value="id"
|
option-value="id"
|
||||||
option-label="description"
|
option-label="description"
|
||||||
:rules="[requiredFieldRule]"
|
:required="true"
|
||||||
/>
|
/>
|
||||||
</QItemSection>
|
</QItemSection>
|
||||||
</QItem>
|
</QItem>
|
||||||
|
@ -544,11 +505,18 @@ const createInvoiceInCorrection = async () => {
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
<i18n>
|
<i18n>
|
||||||
|
en:
|
||||||
|
isNotLinked: The entry {bookEntry} has been deleted with {accountingEntries} entries
|
||||||
|
isLinked: The entry {bookEntry} has been linked to Sage. Please contact administration for further information
|
||||||
|
assertAction: Are you sure you want to {action} this invoice?
|
||||||
es:
|
es:
|
||||||
|
book: asentar
|
||||||
|
unbook: desasentar
|
||||||
|
delete: eliminar
|
||||||
|
clone: clonar
|
||||||
To book: Contabilizar
|
To book: Contabilizar
|
||||||
Are you sure you want to book this invoice?: Estas seguro de querer asentar esta factura?
|
To unbook: Descontabilizar
|
||||||
Delete invoice: Eliminar factura
|
Delete invoice: Eliminar factura
|
||||||
Are you sure you want to delete this invoice?: Estas seguro de querer eliminar esta factura?
|
|
||||||
Invoice deleted: Factura eliminada
|
Invoice deleted: Factura eliminada
|
||||||
Clone invoice: Clonar factura
|
Clone invoice: Clonar factura
|
||||||
Invoice cloned: Factura clonada
|
Invoice cloned: Factura clonada
|
||||||
|
@ -560,4 +528,7 @@ es:
|
||||||
Rectificative invoice: Factura rectificativa
|
Rectificative invoice: Factura rectificativa
|
||||||
Original invoice: Factura origen
|
Original invoice: Factura origen
|
||||||
Entry: entrada
|
Entry: entrada
|
||||||
|
isNotLinked: Se ha eliminado el asiento nº {bookEntry} con {accountingEntries} apuntes
|
||||||
|
isLinked: El asiento {bookEntry} fue enlazado a Sage, por favor contacta con administración
|
||||||
|
assertAction: Estas seguro de querer {action} esta factura?
|
||||||
</i18n>
|
</i18n>
|
||||||
|
|
|
@ -9,24 +9,21 @@ import CrudModel from 'src/components/CrudModel.vue';
|
||||||
import FetchData from 'src/components/FetchData.vue';
|
import FetchData from 'src/components/FetchData.vue';
|
||||||
import VnSelect from 'src/components/common/VnSelect.vue';
|
import VnSelect from 'src/components/common/VnSelect.vue';
|
||||||
import VnCurrency from 'src/components/common/VnCurrency.vue';
|
import VnCurrency from 'src/components/common/VnCurrency.vue';
|
||||||
|
import { toCurrency } from 'src/filters';
|
||||||
|
|
||||||
const route = useRoute();
|
const route = useRoute();
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
const arrayData = useArrayData('InvoiceIn');
|
const arrayData = useArrayData(route.meta.moduleName);
|
||||||
const invoiceIn = computed(() => arrayData.store.data);
|
const invoiceIn = computed(() => arrayData.store.data);
|
||||||
|
|
||||||
const rowsSelected = ref([]);
|
const rowsSelected = ref([]);
|
||||||
const banks = ref([]);
|
const banks = ref([]);
|
||||||
const invoiceInFormRef = ref();
|
const invoiceInFormRef = ref();
|
||||||
const invoiceId = route.params.id;
|
const invoiceId = +route.params.id;
|
||||||
|
|
||||||
const placeholder = 'yyyy/mm/dd';
|
const placeholder = 'yyyy/mm/dd';
|
||||||
|
|
||||||
const filter = {
|
const filter = { where: { invoiceInFk: invoiceId } };
|
||||||
where: {
|
|
||||||
invoiceInFk: invoiceId,
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
const columns = computed(() => [
|
const columns = computed(() => [
|
||||||
{
|
{
|
||||||
|
@ -73,6 +70,7 @@ async function insert() {
|
||||||
await axios.post('/InvoiceInDueDays/new', { id: +invoiceId });
|
await axios.post('/InvoiceInDueDays/new', { id: +invoiceId });
|
||||||
await invoiceInFormRef.value.reload();
|
await invoiceInFormRef.value.reload();
|
||||||
}
|
}
|
||||||
|
const getTotalAmount = (rows) => rows.reduce((acc, { amount }) => acc + +amount, 0);
|
||||||
</script>
|
</script>
|
||||||
<template>
|
<template>
|
||||||
<FetchData
|
<FetchData
|
||||||
|
@ -184,6 +182,19 @@ async function insert() {
|
||||||
/>
|
/>
|
||||||
</QTd>
|
</QTd>
|
||||||
</template>
|
</template>
|
||||||
|
<template #bottom-row>
|
||||||
|
<QTr class="bg">
|
||||||
|
<QTd />
|
||||||
|
<QTd />
|
||||||
|
<QTd />
|
||||||
|
<QTd>
|
||||||
|
{{
|
||||||
|
toCurrency(getTotalAmount(rows), invoiceIn.currency.code)
|
||||||
|
}}
|
||||||
|
</QTd>
|
||||||
|
<QTd />
|
||||||
|
</QTr>
|
||||||
|
</template>
|
||||||
<template #item="props">
|
<template #item="props">
|
||||||
<div class="q-pa-xs col-xs-12 col-sm-6 grid-style-transition">
|
<div class="q-pa-xs col-xs-12 col-sm-6 grid-style-transition">
|
||||||
<QCard>
|
<QCard>
|
||||||
|
@ -294,7 +305,11 @@ async function insert() {
|
||||||
<QBtn color="primary" icon="add" size="lg" round @click="insert" />
|
<QBtn color="primary" icon="add" size="lg" round @click="insert" />
|
||||||
</QPageSticky>
|
</QPageSticky>
|
||||||
</template>
|
</template>
|
||||||
<style lang="scss" scoped></style>
|
<style lang="scss" scoped>
|
||||||
|
.bg {
|
||||||
|
background-color: var(--vn-light-gray);
|
||||||
|
}
|
||||||
|
</style>
|
||||||
<i18n>
|
<i18n>
|
||||||
es:
|
es:
|
||||||
Date: Fecha
|
Date: Fecha
|
||||||
|
|
|
@ -6,26 +6,21 @@ import { toCurrency } from 'src/filters';
|
||||||
import CrudModel from 'src/components/CrudModel.vue';
|
import CrudModel from 'src/components/CrudModel.vue';
|
||||||
import FetchData from 'src/components/FetchData.vue';
|
import FetchData from 'src/components/FetchData.vue';
|
||||||
import VnSelect from 'src/components/common/VnSelect.vue';
|
import VnSelect from 'src/components/common/VnSelect.vue';
|
||||||
import VnLv from 'src/components/ui/VnLv.vue';
|
import { useArrayData } from 'src/composables/useArrayData';
|
||||||
|
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
const route = useRoute();
|
|
||||||
|
|
||||||
|
const route = useRoute();
|
||||||
|
const currency = computed(
|
||||||
|
() => useArrayData(route.meta.moduleName).store.data?.currency?.code
|
||||||
|
);
|
||||||
const invoceInIntrastat = ref([]);
|
const invoceInIntrastat = ref([]);
|
||||||
const amountTotal = computed(() => getTotal('amount'));
|
|
||||||
const netTotal = computed(() => getTotal('net'));
|
|
||||||
const stemsTotal = computed(() => getTotal('stems'));
|
|
||||||
const rowsSelected = ref([]);
|
const rowsSelected = ref([]);
|
||||||
const countries = ref([]);
|
const countries = ref([]);
|
||||||
const intrastats = ref([]);
|
const intrastats = ref([]);
|
||||||
const invoiceInFormRef = ref();
|
const invoiceInFormRef = ref();
|
||||||
|
const invoiceInId = computed(() => +route.params.id);
|
||||||
const filter = {
|
const filter = { where: { invoiceInFk: invoiceInId.value } };
|
||||||
where: {
|
|
||||||
invoiceInFk: route.params.id,
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
const columns = computed(() => [
|
const columns = computed(() => [
|
||||||
{
|
{
|
||||||
name: 'code',
|
name: 'code',
|
||||||
|
@ -77,13 +72,8 @@ const columns = computed(() => [
|
||||||
},
|
},
|
||||||
]);
|
]);
|
||||||
|
|
||||||
function getTotal(type) {
|
const getTotal = (data, key) =>
|
||||||
if (!invoceInIntrastat.value.length) return 0.0;
|
data.reduce((acc, cur) => acc + +String(cur[key]).replace(',', '.'), 0);
|
||||||
return invoceInIntrastat.value.reduce(
|
|
||||||
(total, intrastat) => total + intrastat[type],
|
|
||||||
0.0
|
|
||||||
);
|
|
||||||
}
|
|
||||||
</script>
|
</script>
|
||||||
<template>
|
<template>
|
||||||
<FetchData
|
<FetchData
|
||||||
|
@ -99,30 +89,12 @@ function getTotal(type) {
|
||||||
@on-fetch="(data) => (intrastats = data)"
|
@on-fetch="(data) => (intrastats = data)"
|
||||||
/>
|
/>
|
||||||
<div class="invoiceIn-intrastat">
|
<div class="invoiceIn-intrastat">
|
||||||
<QCard v-if="invoceInIntrastat.length" class="full-width q-mb-md q-pa-sm">
|
|
||||||
<QItem class="justify-end">
|
|
||||||
<div>
|
|
||||||
<QItemLabel>
|
|
||||||
<VnLv
|
|
||||||
:label="t('Total amount')"
|
|
||||||
:value="toCurrency(amountTotal)"
|
|
||||||
/>
|
|
||||||
</QItemLabel>
|
|
||||||
<QItemLabel>
|
|
||||||
<VnLv :label="t('Total net')" :value="netTotal" />
|
|
||||||
</QItemLabel>
|
|
||||||
<QItemLabel>
|
|
||||||
<VnLv :label="t('Total stems')" :value="stemsTotal" />
|
|
||||||
</QItemLabel>
|
|
||||||
</div>
|
|
||||||
</QItem>
|
|
||||||
</QCard>
|
|
||||||
<CrudModel
|
<CrudModel
|
||||||
ref="invoiceInFormRef"
|
ref="invoiceInFormRef"
|
||||||
data-key="InvoiceInIntrastats"
|
data-key="InvoiceInIntrastats"
|
||||||
url="InvoiceInIntrastats"
|
url="InvoiceInIntrastats"
|
||||||
auto-load
|
:auto-load="!currency"
|
||||||
:data-required="{ invoiceInFk: route.params.id }"
|
:data-required="{ invoiceInFk: invoiceInId }"
|
||||||
:filter="filter"
|
:filter="filter"
|
||||||
v-model:selected="rowsSelected"
|
v-model:selected="rowsSelected"
|
||||||
@on-fetch="(data) => (invoceInIntrastat = data)"
|
@on-fetch="(data) => (invoceInIntrastat = data)"
|
||||||
|
@ -172,6 +144,22 @@ function getTotal(type) {
|
||||||
/>
|
/>
|
||||||
</QTd>
|
</QTd>
|
||||||
</template>
|
</template>
|
||||||
|
<template #bottom-row>
|
||||||
|
<QTr class="bg">
|
||||||
|
<QTd />
|
||||||
|
<QTd />
|
||||||
|
<QTd>
|
||||||
|
{{ toCurrency(getTotal(rows, 'amount'), currency) }}
|
||||||
|
</QTd>
|
||||||
|
<QTd>
|
||||||
|
{{ getTotal(rows, 'net') }}
|
||||||
|
</QTd>
|
||||||
|
<QTd>
|
||||||
|
{{ getTotal(rows, 'stems') }}
|
||||||
|
</QTd>
|
||||||
|
<QTd />
|
||||||
|
</QTr>
|
||||||
|
</template>
|
||||||
<template #item="props">
|
<template #item="props">
|
||||||
<div class="q-pa-xs col-xs-12 col-sm-6 grid-style-transition">
|
<div class="q-pa-xs col-xs-12 col-sm-6 grid-style-transition">
|
||||||
<QCard>
|
<QCard>
|
||||||
|
|
|
@ -3,32 +3,24 @@ import { onMounted, ref, computed } from 'vue';
|
||||||
import { useRoute } from 'vue-router';
|
import { useRoute } from 'vue-router';
|
||||||
import { useI18n } from 'vue-i18n';
|
import { useI18n } from 'vue-i18n';
|
||||||
import { toCurrency, toDate } from 'src/filters';
|
import { toCurrency, toDate } from 'src/filters';
|
||||||
|
import { useArrayData } from 'src/composables/useArrayData';
|
||||||
import { getUrl } from 'src/composables/getUrl';
|
import { getUrl } from 'src/composables/getUrl';
|
||||||
import CardSummary from 'components/ui/CardSummary.vue';
|
import CardSummary from 'components/ui/CardSummary.vue';
|
||||||
import VnLv from 'src/components/ui/VnLv.vue';
|
import VnLv from 'src/components/ui/VnLv.vue';
|
||||||
|
import SupplierDescriptorProxy from 'src/pages/Supplier/Card/SupplierDescriptorProxy.vue';
|
||||||
|
import InvoiceIntoBook from '../InvoiceInToBook.vue';
|
||||||
import VnTitle from 'src/components/common/VnTitle.vue';
|
import VnTitle from 'src/components/common/VnTitle.vue';
|
||||||
|
|
||||||
onMounted(async () => {
|
const props = defineProps({ id: { type: [Number, String], default: 0 } });
|
||||||
salixUrl.value = await getUrl('');
|
|
||||||
invoiceInUrl.value = salixUrl.value + `invoiceIn/${entityId.value}/`;
|
|
||||||
});
|
|
||||||
|
|
||||||
const route = useRoute();
|
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
|
const route = useRoute();
|
||||||
|
|
||||||
const $props = defineProps({
|
const entityId = computed(() => props.id || +route.params.id);
|
||||||
id: {
|
const invoiceIn = computed(() => useArrayData(route.meta.moduleName).store.data);
|
||||||
type: Number,
|
const currency = computed(() => invoiceIn.value?.currency?.code);
|
||||||
default: 0,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const entityId = computed(() => $props.id || route.params.id);
|
|
||||||
|
|
||||||
const salixUrl = ref();
|
|
||||||
const invoiceInUrl = ref();
|
const invoiceInUrl = ref();
|
||||||
const amountsNotMatch = ref(null);
|
const amountsNotMatch = ref(null);
|
||||||
const intrastatTotals = ref({});
|
const intrastatTotals = ref({ amount: 0, net: 0, stems: 0 });
|
||||||
|
|
||||||
const vatColumns = ref([
|
const vatColumns = ref([
|
||||||
{
|
{
|
||||||
|
@ -42,14 +34,16 @@ const vatColumns = ref([
|
||||||
name: 'landed',
|
name: 'landed',
|
||||||
label: 'invoiceIn.summary.taxableBase',
|
label: 'invoiceIn.summary.taxableBase',
|
||||||
field: (row) => row.taxableBase,
|
field: (row) => row.taxableBase,
|
||||||
format: (value) => toCurrency(value),
|
format: (value) => toCurrency(value, currency.value),
|
||||||
sortable: true,
|
sortable: true,
|
||||||
align: 'left',
|
align: 'left',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'vat',
|
name: 'vat',
|
||||||
label: 'invoiceIn.summary.sageVat',
|
label: 'invoiceIn.summary.sageVat',
|
||||||
field: (row) => row.taxTypeSage?.vat,
|
field: (row) => {
|
||||||
|
if (row.taxTypeSage) return `#${row.taxTypeSage.id} : ${row.taxTypeSage.vat}`;
|
||||||
|
},
|
||||||
format: (value) => value,
|
format: (value) => value,
|
||||||
sortable: true,
|
sortable: true,
|
||||||
align: 'left',
|
align: 'left',
|
||||||
|
@ -57,7 +51,10 @@ const vatColumns = ref([
|
||||||
{
|
{
|
||||||
name: 'transaction',
|
name: 'transaction',
|
||||||
label: 'invoiceIn.summary.sageTransaction',
|
label: 'invoiceIn.summary.sageTransaction',
|
||||||
field: (row) => row.transactionTypeSage?.transaction,
|
field: (row) => {
|
||||||
|
if (row.transactionTypeSage)
|
||||||
|
return `#${row.transactionTypeSage.id} : ${row.transactionTypeSage?.transaction}`;
|
||||||
|
},
|
||||||
format: (value) => value,
|
format: (value) => value,
|
||||||
sortable: true,
|
sortable: true,
|
||||||
align: 'left',
|
align: 'left',
|
||||||
|
@ -66,9 +63,9 @@ const vatColumns = ref([
|
||||||
name: 'rate',
|
name: 'rate',
|
||||||
label: 'invoiceIn.summary.rate',
|
label: 'invoiceIn.summary.rate',
|
||||||
field: (row) => taxRate(row.taxableBase, row.taxTypeSage?.rate),
|
field: (row) => taxRate(row.taxableBase, row.taxTypeSage?.rate),
|
||||||
format: (value) => toCurrency(value),
|
format: (value) => toCurrency(value, currency.value),
|
||||||
sortable: true,
|
sortable: true,
|
||||||
align: 'left',
|
align: 'center',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'currency',
|
name: 'currency',
|
||||||
|
@ -99,7 +96,7 @@ const dueDayColumns = ref([
|
||||||
name: 'amount',
|
name: 'amount',
|
||||||
label: 'invoiceIn.summary.amount',
|
label: 'invoiceIn.summary.amount',
|
||||||
field: (row) => row.amount,
|
field: (row) => row.amount,
|
||||||
format: (value) => toCurrency(value),
|
format: (value) => toCurrency(value, currency.value),
|
||||||
sortable: true,
|
sortable: true,
|
||||||
align: 'left',
|
align: 'left',
|
||||||
},
|
},
|
||||||
|
@ -122,13 +119,15 @@ const intrastatColumns = ref([
|
||||||
},
|
},
|
||||||
sortable: true,
|
sortable: true,
|
||||||
align: 'left',
|
align: 'left',
|
||||||
|
style: 'width: 10px',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'amount',
|
name: 'amount',
|
||||||
label: 'invoiceIn.summary.amount',
|
label: 'invoiceIn.summary.amount',
|
||||||
field: (row) => toCurrency(row.amount),
|
field: (row) => toCurrency(row.amount, currency.value),
|
||||||
sortable: true,
|
sortable: true,
|
||||||
align: 'left',
|
align: 'left',
|
||||||
|
style: 'width: 10px',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'net',
|
name: 'net',
|
||||||
|
@ -155,58 +154,55 @@ const intrastatColumns = ref([
|
||||||
},
|
},
|
||||||
]);
|
]);
|
||||||
|
|
||||||
function getAmountNotMatch(totals) {
|
onMounted(async () => {
|
||||||
return (
|
invoiceInUrl.value = `${await getUrl('')}invoiceIn/${entityId.value}/`;
|
||||||
|
});
|
||||||
|
|
||||||
|
const init = (data) => {
|
||||||
|
if (!data) return;
|
||||||
|
|
||||||
|
const { totals, invoiceInIntrastat } = data;
|
||||||
|
amountsNotMatch.value =
|
||||||
totals.totalDueDay != totals.totalTaxableBase &&
|
totals.totalDueDay != totals.totalTaxableBase &&
|
||||||
totals.totalDueDay != totals.totalVat
|
totals.totalDueDay != totals.totalVat;
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
function getIntrastatTotals(intrastat) {
|
invoiceInIntrastat.forEach((val) => {
|
||||||
const totals = {
|
intrastatTotals.value.amount += val.amount;
|
||||||
amount: intrastat.reduce((acc, cur) => acc + cur.amount, 0),
|
intrastatTotals.value.net += val.net;
|
||||||
net: intrastat.reduce((acc, cur) => acc + cur.net, 0),
|
intrastatTotals.value.stems += val.stems;
|
||||||
stems: intrastat.reduce((acc, cur) => acc + cur.stems, 0),
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
return totals;
|
const taxRate = (taxableBase = 0, rate = 0) => (rate / 100) * taxableBase;
|
||||||
}
|
|
||||||
|
|
||||||
function getTaxTotal(tax) {
|
const getTotalTax = (tax) =>
|
||||||
return tax.reduce(
|
tax.reduce((acc, cur) => acc + taxRate(cur.taxableBase, cur.taxTypeSage?.rate), 0);
|
||||||
(acc, cur) => acc + taxRate(cur.taxableBase, cur.taxTypeSage?.rate),
|
|
||||||
0
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
function setData(entity) {
|
const getLink = (param) => `#/invoice-in/${entityId.value}/${param}`;
|
||||||
if (!entity) return false;
|
|
||||||
|
|
||||||
amountsNotMatch.value = getAmountNotMatch(entity.totals);
|
|
||||||
|
|
||||||
if (entity.invoiceInIntrastat.length)
|
|
||||||
intrastatTotals.value = { ...getIntrastatTotals(entity.invoiceInIntrastat) };
|
|
||||||
}
|
|
||||||
|
|
||||||
function taxRate(taxableBase = 0, rate = 0) {
|
|
||||||
return (rate / 100) * taxableBase;
|
|
||||||
}
|
|
||||||
|
|
||||||
function getLink(param) {
|
|
||||||
return `#/invoice-in/${entityId.value}/${param}`;
|
|
||||||
}
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<CardSummary
|
<CardSummary
|
||||||
data-key="InvoiceInSummary"
|
data-key="InvoiceInSummary"
|
||||||
:url="`InvoiceIns/${entityId}/summary`"
|
:url="`InvoiceIns/${entityId}/summary`"
|
||||||
@on-fetch="(data) => setData(data)"
|
@on-fetch="(data) => init(data)"
|
||||||
>
|
>
|
||||||
<template #header="{ entity: invoiceIn }">
|
<template #header="{ entity }">
|
||||||
<div>{{ invoiceIn.id }} - {{ invoiceIn.supplier?.name }}</div>
|
<div>{{ entity.id }} - {{ entity.supplier?.name }}</div>
|
||||||
</template>
|
</template>
|
||||||
<template #body="{ entity: invoiceIn }">
|
<template #header-right v-if="!invoiceIn?.isBooked">
|
||||||
|
<InvoiceIntoBook>
|
||||||
|
<template #content="{ book }">
|
||||||
|
<QBtn
|
||||||
|
:label="t('To book')"
|
||||||
|
color="orange-11"
|
||||||
|
text-color="black"
|
||||||
|
@click="book(entityId)"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
</InvoiceIntoBook>
|
||||||
|
</template>
|
||||||
|
<template #body="{ entity }">
|
||||||
<!--Basic Data-->
|
<!--Basic Data-->
|
||||||
<QCard class="vn-one">
|
<QCard class="vn-one">
|
||||||
<QCardSection class="q-pa-none">
|
<QCardSection class="q-pa-none">
|
||||||
|
@ -217,19 +213,26 @@ function getLink(param) {
|
||||||
</QCardSection>
|
</QCardSection>
|
||||||
<VnLv
|
<VnLv
|
||||||
:label="t('invoiceIn.summary.supplier')"
|
:label="t('invoiceIn.summary.supplier')"
|
||||||
:value="invoiceIn.supplier?.name"
|
:value="entity.supplier?.name"
|
||||||
/>
|
>
|
||||||
|
<template #value>
|
||||||
|
<span class="link">
|
||||||
|
{{ entity.supplier?.name }}
|
||||||
|
<SupplierDescriptorProxy :id="entity.supplierFk" />
|
||||||
|
</span>
|
||||||
|
</template>
|
||||||
|
</VnLv>
|
||||||
<VnLv
|
<VnLv
|
||||||
:label="t('invoiceIn.summary.supplierRef')"
|
:label="t('invoiceIn.summary.supplierRef')"
|
||||||
:value="invoiceIn.supplierRef"
|
:value="entity.supplierRef"
|
||||||
/>
|
/>
|
||||||
<VnLv
|
<VnLv
|
||||||
:label="t('invoiceIn.summary.currency')"
|
:label="t('invoiceIn.summary.currency')"
|
||||||
:value="invoiceIn.currency?.code"
|
:value="entity.currency?.code"
|
||||||
/>
|
/>
|
||||||
<VnLv
|
<VnLv
|
||||||
:label="t('invoiceIn.summary.docNumber')"
|
:label="t('invoiceIn.summary.docNumber')"
|
||||||
:value="`${invoiceIn.serial}/${invoiceIn.serialNumber}`"
|
:value="`${entity.serial}/${entity.serialNumber}`"
|
||||||
/>
|
/>
|
||||||
</QCard>
|
</QCard>
|
||||||
<QCard class="vn-one">
|
<QCard class="vn-one">
|
||||||
|
@ -242,19 +245,19 @@ function getLink(param) {
|
||||||
<VnLv
|
<VnLv
|
||||||
:ellipsis-value="false"
|
:ellipsis-value="false"
|
||||||
:label="t('invoiceIn.summary.issued')"
|
:label="t('invoiceIn.summary.issued')"
|
||||||
:value="toDate(invoiceIn.issued)"
|
:value="toDate(entity.issued)"
|
||||||
/>
|
/>
|
||||||
<VnLv
|
<VnLv
|
||||||
:label="t('invoiceIn.summary.operated')"
|
:label="t('invoiceIn.summary.operated')"
|
||||||
:value="toDate(invoiceIn.operated)"
|
:value="toDate(entity.operated)"
|
||||||
/>
|
/>
|
||||||
<VnLv
|
<VnLv
|
||||||
:label="t('invoiceIn.summary.bookEntried')"
|
:label="t('invoiceIn.summary.bookEntried')"
|
||||||
:value="toDate(invoiceIn.bookEntried)"
|
:value="toDate(entity.bookEntried)"
|
||||||
/>
|
/>
|
||||||
<VnLv
|
<VnLv
|
||||||
:label="t('invoiceIn.summary.bookedDate')"
|
:label="t('invoiceIn.summary.bookedDate')"
|
||||||
:value="toDate(invoiceIn.booked)"
|
:value="toDate(entity.booked)"
|
||||||
/>
|
/>
|
||||||
</QCard>
|
</QCard>
|
||||||
<QCard class="vn-one">
|
<QCard class="vn-one">
|
||||||
|
@ -266,20 +269,19 @@ function getLink(param) {
|
||||||
</QCardSection>
|
</QCardSection>
|
||||||
<VnLv
|
<VnLv
|
||||||
:label="t('invoiceIn.summary.sage')"
|
:label="t('invoiceIn.summary.sage')"
|
||||||
:value="invoiceIn.sageWithholding?.withholding"
|
:value="entity.sageWithholding?.withholding"
|
||||||
/>
|
/>
|
||||||
<VnLv
|
<VnLv
|
||||||
:label="t('invoiceIn.summary.vat')"
|
:label="t('invoiceIn.summary.vat')"
|
||||||
:value="invoiceIn.expenseDeductible?.name"
|
:value="entity.expenseDeductible?.name"
|
||||||
/>
|
/>
|
||||||
<VnLv
|
<VnLv
|
||||||
:label="t('invoiceIn.summary.company')"
|
:label="t('invoiceIn.summary.company')"
|
||||||
:value="invoiceIn.company?.code"
|
:value="entity.company?.code"
|
||||||
/>
|
/>
|
||||||
<QCheckbox
|
<VnLv
|
||||||
:label="t('invoiceIn.summary.booked')"
|
:label="t('invoiceIn.summary.booked')"
|
||||||
v-model="invoiceIn.isBooked"
|
:value="invoiceIn?.isBooked"
|
||||||
:disable="true"
|
|
||||||
/>
|
/>
|
||||||
</QCard>
|
</QCard>
|
||||||
<QCard class="vn-one">
|
<QCard class="vn-one">
|
||||||
|
@ -290,14 +292,13 @@ function getLink(param) {
|
||||||
/>
|
/>
|
||||||
</QCardSection>
|
</QCardSection>
|
||||||
<QCardSection class="q-pa-none">
|
<QCardSection class="q-pa-none">
|
||||||
<div class="bordered q-px-sm q-mx-auto">
|
|
||||||
<VnLv
|
<VnLv
|
||||||
:label="t('invoiceIn.summary.taxableBase')"
|
:label="t('invoiceIn.summary.taxableBase')"
|
||||||
:value="toCurrency(invoiceIn.totals.totalTaxableBase)"
|
:value="toCurrency(entity.totals.totalTaxableBase, currency)"
|
||||||
/>
|
/>
|
||||||
<VnLv
|
<VnLv
|
||||||
label="Total"
|
label="Total"
|
||||||
:value="toCurrency(invoiceIn.totals.totalVat)"
|
:value="toCurrency(entity.totals.totalVat, currency)"
|
||||||
/>
|
/>
|
||||||
<VnLv :label="t('invoiceIn.summary.dueTotal')">
|
<VnLv :label="t('invoiceIn.summary.dueTotal')">
|
||||||
<template #value>
|
<template #value>
|
||||||
|
@ -311,37 +312,42 @@ function getLink(param) {
|
||||||
: t('invoiceIn.summary.dueTotal')
|
: t('invoiceIn.summary.dueTotal')
|
||||||
"
|
"
|
||||||
>
|
>
|
||||||
{{ toCurrency(invoiceIn.totals.totalDueDay) }}
|
{{ toCurrency(entity.totals.totalDueDay, currency) }}
|
||||||
</QChip>
|
</QChip>
|
||||||
</template>
|
</template>
|
||||||
</VnLv>
|
</VnLv>
|
||||||
</div>
|
|
||||||
</QCardSection>
|
</QCardSection>
|
||||||
</QCard>
|
</QCard>
|
||||||
<!--Vat-->
|
<!--Vat-->
|
||||||
<QCard v-if="invoiceIn.invoiceInTax.length">
|
<QCard v-if="entity.invoiceInTax.length" class="vat">
|
||||||
<VnTitle :url="getLink('vat')" :text="t('invoiceIn.card.vat')" />
|
<VnTitle :url="getLink('vat')" :text="t('invoiceIn.card.vat')" />
|
||||||
<QTable
|
<QTable
|
||||||
:columns="vatColumns"
|
:columns="vatColumns"
|
||||||
:rows="invoiceIn.invoiceInTax"
|
:rows="entity.invoiceInTax"
|
||||||
flat
|
flat
|
||||||
hide-pagination
|
hide-pagination
|
||||||
>
|
>
|
||||||
<template #header="props">
|
<template #header="vatProps">
|
||||||
<QTr :props="props" class="bg">
|
<QTr :props="vatProps" class="bg">
|
||||||
<QTh v-for="col in props.cols" :key="col.name" :props="props">
|
<QTh
|
||||||
|
v-for="col in vatProps.cols"
|
||||||
|
:key="col.name"
|
||||||
|
:props="vatProps"
|
||||||
|
>
|
||||||
{{ t(col.label) }}
|
{{ t(col.label) }}
|
||||||
</QTh>
|
</QTh>
|
||||||
</QTr>
|
</QTr>
|
||||||
</template>
|
</template>
|
||||||
<template #bottom-row>
|
<template #bottom-row>
|
||||||
<QTr class="bg">
|
<QTr class="bg">
|
||||||
<QTd></QTd>
|
|
||||||
<QTd>{{ toCurrency(invoiceIn.totals.totalTaxableBase) }}</QTd>
|
|
||||||
<QTd></QTd>
|
|
||||||
<QTd></QTd>
|
<QTd></QTd>
|
||||||
<QTd>{{
|
<QTd>{{
|
||||||
toCurrency(getTaxTotal(invoiceIn.invoiceInTax))
|
toCurrency(entity.totals.totalTaxableBase, currency)
|
||||||
|
}}</QTd>
|
||||||
|
<QTd></QTd>
|
||||||
|
<QTd></QTd>
|
||||||
|
<QTd class="text-center">{{
|
||||||
|
toCurrency(getTotalTax(entity.invoiceInTax, currency))
|
||||||
}}</QTd>
|
}}</QTd>
|
||||||
<QTd></QTd>
|
<QTd></QTd>
|
||||||
</QTr>
|
</QTr>
|
||||||
|
@ -349,17 +355,17 @@ function getLink(param) {
|
||||||
</QTable>
|
</QTable>
|
||||||
</QCard>
|
</QCard>
|
||||||
<!--Due Day-->
|
<!--Due Day-->
|
||||||
<QCard v-if="invoiceIn.invoiceInDueDay.length">
|
<QCard v-if="entity.invoiceInDueDay.length" class="due-day">
|
||||||
<VnTitle :url="getLink('due-day')" :text="t('invoiceIn.card.dueDay')" />
|
<VnTitle :url="getLink('due-day')" :text="t('invoiceIn.card.dueDay')" />
|
||||||
<QTable
|
<QTable :columns="dueDayColumns" :rows="entity.invoiceInDueDay" flat>
|
||||||
class="full-width"
|
<template #header="dueDayProps">
|
||||||
:columns="dueDayColumns"
|
<QTr :props="dueDayProps" class="bg">
|
||||||
:rows="invoiceIn.invoiceInDueDay"
|
<QTh
|
||||||
flat
|
table-header-style="max-width:50%"
|
||||||
|
v-for="col in dueDayProps.cols"
|
||||||
|
:key="col.name"
|
||||||
|
:props="dueDayProps"
|
||||||
>
|
>
|
||||||
<template #header="props">
|
|
||||||
<QTr :props="props" class="bg">
|
|
||||||
<QTh v-for="col in props.cols" :key="col.name" :props="props">
|
|
||||||
{{ t(col.label) }}
|
{{ t(col.label) }}
|
||||||
</QTh>
|
</QTh>
|
||||||
</QTr>
|
</QTr>
|
||||||
|
@ -368,26 +374,32 @@ function getLink(param) {
|
||||||
<QTr class="bg">
|
<QTr class="bg">
|
||||||
<QTd></QTd>
|
<QTd></QTd>
|
||||||
<QTd></QTd>
|
<QTd></QTd>
|
||||||
<QTd>{{ toCurrency(invoiceIn.totals.totalDueDay) }}</QTd>
|
<QTd>
|
||||||
|
{{ toCurrency(entity.totals.totalDueDay, currency) }}
|
||||||
|
</QTd>
|
||||||
<QTd></QTd>
|
<QTd></QTd>
|
||||||
</QTr>
|
</QTr>
|
||||||
</template>
|
</template>
|
||||||
</QTable>
|
</QTable>
|
||||||
</QCard>
|
</QCard>
|
||||||
<!--Intrastat-->
|
<!--Intrastat-->
|
||||||
<QCard v-if="invoiceIn.invoiceInIntrastat.length">
|
<QCard v-if="entity.invoiceInIntrastat.length">
|
||||||
<VnTitle
|
<VnTitle
|
||||||
:url="getLink('intrastat')"
|
:url="getLink('intrastat')"
|
||||||
:text="t('invoiceIn.card.intrastat')"
|
:text="t('invoiceIn.card.intrastat')"
|
||||||
/>
|
/>
|
||||||
<QTable
|
<QTable
|
||||||
:columns="intrastatColumns"
|
:columns="intrastatColumns"
|
||||||
:rows="invoiceIn.invoiceInIntrastat"
|
:rows="entity.invoiceInIntrastat"
|
||||||
flat
|
flat
|
||||||
>
|
>
|
||||||
<template #header="props">
|
<template #header="intrastatProps">
|
||||||
<QTr :props="props" class="bg">
|
<QTr :props="intrastatProps" class="bg">
|
||||||
<QTh v-for="col in props.cols" :key="col.name" :props="props">
|
<QTh
|
||||||
|
v-for="col in intrastatProps.cols"
|
||||||
|
:key="col.name"
|
||||||
|
:props="intrastatProps"
|
||||||
|
>
|
||||||
{{ t(col.label) }}
|
{{ t(col.label) }}
|
||||||
</QTh>
|
</QTh>
|
||||||
</QTr>
|
</QTr>
|
||||||
|
@ -395,7 +407,7 @@ function getLink(param) {
|
||||||
<template #bottom-row>
|
<template #bottom-row>
|
||||||
<QTr class="bg">
|
<QTr class="bg">
|
||||||
<QTd></QTd>
|
<QTd></QTd>
|
||||||
<QTd>{{ toCurrency(intrastatTotals.amount) }}</QTd>
|
<QTd>{{ toCurrency(intrastatTotals.amount, currency) }}</QTd>
|
||||||
<QTd>{{ intrastatTotals.net }}</QTd>
|
<QTd>{{ intrastatTotals.net }}</QTd>
|
||||||
<QTd>{{ intrastatTotals.stems }}</QTd>
|
<QTd>{{ intrastatTotals.stems }}</QTd>
|
||||||
<QTd></QTd>
|
<QTd></QTd>
|
||||||
|
@ -410,13 +422,28 @@ function getLink(param) {
|
||||||
.bg {
|
.bg {
|
||||||
background-color: var(--vn-accent-color);
|
background-color: var(--vn-accent-color);
|
||||||
}
|
}
|
||||||
.bordered {
|
@media (min-width: $breakpoint-md) {
|
||||||
border: 1px solid var(--vn-text-color);
|
.summaryBody {
|
||||||
max-width: 18em;
|
.vat {
|
||||||
|
flex: 65%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.due-day {
|
||||||
|
flex: 30%;
|
||||||
|
}
|
||||||
|
.vat,
|
||||||
|
.due-day {
|
||||||
|
.q-table th {
|
||||||
|
padding-right: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
<i18n>
|
<i18n>
|
||||||
es:
|
es:
|
||||||
Search invoice: Buscar factura recibida
|
Search invoice: Buscar factura recibida
|
||||||
You can search by invoice reference: Puedes buscar por referencia de la factura
|
You can search by invoice reference: Puedes buscar por referencia de la factura
|
||||||
|
Totals: Totales
|
||||||
|
To book: Contabilizar
|
||||||
</i18n>
|
</i18n>
|
||||||
|
|
|
@ -11,13 +11,14 @@ import VnSelect from 'src/components/common/VnSelect.vue';
|
||||||
import CrudModel from 'src/components/CrudModel.vue';
|
import CrudModel from 'src/components/CrudModel.vue';
|
||||||
import VnCurrency from 'src/components/common/VnCurrency.vue';
|
import VnCurrency from 'src/components/common/VnCurrency.vue';
|
||||||
|
|
||||||
const route = useRoute();
|
const router = useRoute();
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
const quasar = useQuasar();
|
const quasar = useQuasar();
|
||||||
|
|
||||||
const arrayData = useArrayData('InvoiceIn');
|
const arrayData = useArrayData(router.meta.moduleName);
|
||||||
const invoiceIn = computed(() => arrayData.store.data);
|
const invoiceIn = computed(() => arrayData.store.data);
|
||||||
|
const invoiceId = +router.params.id;
|
||||||
|
const currency = computed(() => invoiceIn.value?.currency?.code);
|
||||||
const expenses = ref([]);
|
const expenses = ref([]);
|
||||||
const sageTaxTypes = ref([]);
|
const sageTaxTypes = ref([]);
|
||||||
const sageTransactionTypes = ref([]);
|
const sageTransactionTypes = ref([]);
|
||||||
|
@ -55,7 +56,7 @@ const columns = computed(() => [
|
||||||
{
|
{
|
||||||
name: 'taxablebase',
|
name: 'taxablebase',
|
||||||
label: t('Taxable base'),
|
label: t('Taxable base'),
|
||||||
field: (row) => toCurrency(row.taxableBase),
|
field: (row) => toCurrency(row.taxableBase, currency.value),
|
||||||
model: 'taxableBase',
|
model: 'taxableBase',
|
||||||
sortable: true,
|
sortable: true,
|
||||||
tabIndex: 2,
|
tabIndex: 2,
|
||||||
|
@ -68,7 +69,7 @@ const columns = computed(() => [
|
||||||
options: sageTaxTypes.value,
|
options: sageTaxTypes.value,
|
||||||
model: 'taxTypeSageFk',
|
model: 'taxTypeSageFk',
|
||||||
optionValue: 'id',
|
optionValue: 'id',
|
||||||
optionLabel: 'vat',
|
optionLabel: 'id',
|
||||||
sortable: true,
|
sortable: true,
|
||||||
tabindex: 3,
|
tabindex: 3,
|
||||||
align: 'left',
|
align: 'left',
|
||||||
|
@ -80,7 +81,7 @@ const columns = computed(() => [
|
||||||
options: sageTransactionTypes.value,
|
options: sageTransactionTypes.value,
|
||||||
model: 'transactionTypeSageFk',
|
model: 'transactionTypeSageFk',
|
||||||
optionValue: 'id',
|
optionValue: 'id',
|
||||||
optionLabel: 'transaction',
|
optionLabel: 'id',
|
||||||
sortable: true,
|
sortable: true,
|
||||||
tabIndex: 4,
|
tabIndex: 4,
|
||||||
align: 'left',
|
align: 'left',
|
||||||
|
@ -90,7 +91,7 @@ const columns = computed(() => [
|
||||||
label: t('Rate'),
|
label: t('Rate'),
|
||||||
sortable: true,
|
sortable: true,
|
||||||
tabIndex: 5,
|
tabIndex: 5,
|
||||||
field: (row) => toCurrency(taxRate(row, row.taxTypeSageFk)),
|
field: (row) => toCurrency(taxRate(row, row.taxTypeSageFk), currency.value),
|
||||||
align: 'left',
|
align: 'left',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -114,7 +115,7 @@ const filter = {
|
||||||
'transactionTypeSageFk',
|
'transactionTypeSageFk',
|
||||||
],
|
],
|
||||||
where: {
|
where: {
|
||||||
invoiceInFk: route.params.id,
|
invoiceInFk: invoiceId,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -161,6 +162,9 @@ async function addExpense() {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
const getTotalTaxableBase = (rows) =>
|
||||||
|
rows.reduce((acc, { taxableBase }) => acc + +taxableBase, 0);
|
||||||
|
const getTotalRate = (rows) => rows.reduce((acc, cur) => acc + +taxRate(cur), 0);
|
||||||
</script>
|
</script>
|
||||||
<template>
|
<template>
|
||||||
<FetchData
|
<FetchData
|
||||||
|
@ -181,9 +185,10 @@ async function addExpense() {
|
||||||
data-key="InvoiceInTaxes"
|
data-key="InvoiceInTaxes"
|
||||||
url="InvoiceInTaxes"
|
url="InvoiceInTaxes"
|
||||||
:filter="filter"
|
:filter="filter"
|
||||||
:data-required="{ invoiceInFk: route.params.id }"
|
:data-required="{ invoiceInFk: invoiceId }"
|
||||||
auto-load
|
auto-load
|
||||||
v-model:selected="rowsSelected"
|
v-model:selected="rowsSelected"
|
||||||
|
:go-to="`/invoice-in/${invoiceId}/due-day`"
|
||||||
>
|
>
|
||||||
<template #body="{ rows }">
|
<template #body="{ rows }">
|
||||||
<QTable
|
<QTable
|
||||||
|
@ -303,6 +308,19 @@ async function addExpense() {
|
||||||
/>
|
/>
|
||||||
</QTd>
|
</QTd>
|
||||||
</template>
|
</template>
|
||||||
|
<template #bottom-row>
|
||||||
|
<QTr class="bg">
|
||||||
|
<QTd />
|
||||||
|
<QTd />
|
||||||
|
<QTd>
|
||||||
|
{{ toCurrency(getTotalTaxableBase(rows), currency) }}
|
||||||
|
</QTd>
|
||||||
|
<QTd />
|
||||||
|
<QTd />
|
||||||
|
<QTd> {{ toCurrency(getTotalRate(rows), currency) }}</QTd>
|
||||||
|
<QTd />
|
||||||
|
</QTr>
|
||||||
|
</template>
|
||||||
<template #item="props">
|
<template #item="props">
|
||||||
<div class="q-pa-xs col-xs-12 col-sm-6 grid-style-transition">
|
<div class="q-pa-xs col-xs-12 col-sm-6 grid-style-transition">
|
||||||
<QCard bordered flat class="q-my-xs">
|
<QCard bordered flat class="q-my-xs">
|
||||||
|
@ -391,7 +409,7 @@ async function addExpense() {
|
||||||
</VnSelect>
|
</VnSelect>
|
||||||
</QItem>
|
</QItem>
|
||||||
<QItem>
|
<QItem>
|
||||||
{{ toCurrency(taxRate(props.row)) }}
|
{{ toCurrency(taxRate(props.row), currency) }}
|
||||||
</QItem>
|
</QItem>
|
||||||
<QItem>
|
<QItem>
|
||||||
<QInput
|
<QInput
|
||||||
|
@ -458,6 +476,10 @@ async function addExpense() {
|
||||||
</QPageSticky>
|
</QPageSticky>
|
||||||
</template>
|
</template>
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
|
.bg {
|
||||||
|
background-color: var(--vn-light-gray);
|
||||||
|
}
|
||||||
|
|
||||||
@media (max-width: $breakpoint-xs) {
|
@media (max-width: $breakpoint-xs) {
|
||||||
.q-dialog {
|
.q-dialog {
|
||||||
.q-card {
|
.q-card {
|
||||||
|
|
|
@ -0,0 +1,124 @@
|
||||||
|
<script setup>
|
||||||
|
import { reactive, ref } from 'vue';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import { useRoute, useRouter } from 'vue-router';
|
||||||
|
import FormModel from 'components/FormModel.vue';
|
||||||
|
import VnRow from 'components/ui/VnRow.vue';
|
||||||
|
import VnSelect from 'src/components/common/VnSelect.vue';
|
||||||
|
import FetchData from 'components/FetchData.vue';
|
||||||
|
import VnSubToolbar from 'src/components/ui/VnSubToolbar.vue';
|
||||||
|
import VnSearchbar from 'src/components/ui/VnSearchbar.vue';
|
||||||
|
import { useStateStore } from 'stores/useStateStore';
|
||||||
|
import { useState } from 'src/composables/useState';
|
||||||
|
import VnInputDate from 'src/components/common/VnInputDate.vue';
|
||||||
|
import VnInput from 'src/components/common/VnInput.vue';
|
||||||
|
|
||||||
|
const state = useState();
|
||||||
|
const { t } = useI18n();
|
||||||
|
const route = useRoute();
|
||||||
|
const router = useRouter();
|
||||||
|
const stateStore = useStateStore();
|
||||||
|
|
||||||
|
const user = state.getUser();
|
||||||
|
const newInvoiceIn = reactive({
|
||||||
|
supplierFk: +route.query?.supplierFk || null,
|
||||||
|
supplierRef: null,
|
||||||
|
companyFk: user.value.companyFk || null,
|
||||||
|
issued: Date.vnNew(),
|
||||||
|
});
|
||||||
|
const suppliersOptions = ref([]);
|
||||||
|
const companiesOptions = ref([]);
|
||||||
|
|
||||||
|
const redirectToInvoiceInBasicData = ({ id }) => {
|
||||||
|
router.push({ name: 'InvoiceInBasicData', params: { id } });
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<FetchData
|
||||||
|
url="Suppliers"
|
||||||
|
:filter="{ fields: ['id', 'nickname'] }"
|
||||||
|
order="nickname"
|
||||||
|
@on-fetch="(data) => (suppliersOptions = data)"
|
||||||
|
auto-load
|
||||||
|
/>
|
||||||
|
<FetchData
|
||||||
|
ref="companiesRef"
|
||||||
|
url="Companies"
|
||||||
|
:filter="{ fields: ['id', 'code'] }"
|
||||||
|
order="code"
|
||||||
|
@on-fetch="(data) => (companiesOptions = data)"
|
||||||
|
auto-load
|
||||||
|
/>
|
||||||
|
<template v-if="stateStore.isHeaderMounted()">
|
||||||
|
<Teleport to="#searchbar">
|
||||||
|
<VnSearchbar
|
||||||
|
custom-route-redirect-name="InvoiceInSummary"
|
||||||
|
data-key="InvoiceInSummary"
|
||||||
|
/>
|
||||||
|
</Teleport>
|
||||||
|
</template>
|
||||||
|
<QPage>
|
||||||
|
<VnSubToolbar />
|
||||||
|
<FormModel
|
||||||
|
url-create="InvoiceIns"
|
||||||
|
model="InvoiceIn"
|
||||||
|
:form-initial-data="newInvoiceIn"
|
||||||
|
@on-data-saved="redirectToInvoiceInBasicData"
|
||||||
|
>
|
||||||
|
<template #form="{ data, validate }">
|
||||||
|
<VnRow>
|
||||||
|
<VnSelect
|
||||||
|
:label="t('Supplier')"
|
||||||
|
v-model="data.supplierFk"
|
||||||
|
:options="suppliersOptions"
|
||||||
|
option-value="id"
|
||||||
|
option-label="nickname"
|
||||||
|
hide-selected
|
||||||
|
:required="true"
|
||||||
|
:rules="validate('entry.supplierFk')"
|
||||||
|
>
|
||||||
|
<template #option="scope">
|
||||||
|
<QItem v-bind="scope.itemProps">
|
||||||
|
<QItemSection>
|
||||||
|
<QItemLabel>{{ scope.opt?.nickname }}</QItemLabel>
|
||||||
|
<QItemLabel caption>
|
||||||
|
#{{ scope.opt?.id }}
|
||||||
|
</QItemLabel>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
</template>
|
||||||
|
</VnSelect>
|
||||||
|
<VnInput
|
||||||
|
:label="t('invoiceIn.summary.supplierRef')"
|
||||||
|
v-model="data.supplierRef"
|
||||||
|
/>
|
||||||
|
</VnRow>
|
||||||
|
<VnRow>
|
||||||
|
<VnSelect
|
||||||
|
:label="t('Company')"
|
||||||
|
v-model="data.companyFk"
|
||||||
|
:options="companiesOptions"
|
||||||
|
option-value="id"
|
||||||
|
option-label="code"
|
||||||
|
map-options
|
||||||
|
hide-selected
|
||||||
|
:required="true"
|
||||||
|
:rules="validate('invoiceIn.companyFk')"
|
||||||
|
/>
|
||||||
|
<VnInputDate
|
||||||
|
:label="t('invoiceIn.summary.issued')"
|
||||||
|
v-model="data.issued"
|
||||||
|
/>
|
||||||
|
</VnRow>
|
||||||
|
</template>
|
||||||
|
</FormModel>
|
||||||
|
</QPage>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<i18n>
|
||||||
|
es:
|
||||||
|
Supplier: Proveedor
|
||||||
|
Travel: Envío
|
||||||
|
Company: Empresa
|
||||||
|
</i18n>
|
|
@ -4,33 +4,17 @@ import { useI18n } from 'vue-i18n';
|
||||||
|
|
||||||
import VnSelect from 'components/common/VnSelect.vue';
|
import VnSelect from 'components/common/VnSelect.vue';
|
||||||
import VnFilterPanel from 'src/components/ui/VnFilterPanel.vue';
|
import VnFilterPanel from 'src/components/ui/VnFilterPanel.vue';
|
||||||
import FetchData from 'components/FetchData.vue';
|
|
||||||
import VnInput from 'src/components/common/VnInput.vue';
|
import VnInput from 'src/components/common/VnInput.vue';
|
||||||
import VnInputDate from 'components/common/VnInputDate.vue';
|
import VnInputDate from 'components/common/VnInputDate.vue';
|
||||||
import VnCurrency from 'src/components/common/VnCurrency.vue';
|
import VnCurrency from 'src/components/common/VnCurrency.vue';
|
||||||
|
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
const props = defineProps({
|
defineProps({ dataKey: { type: String, required: true } });
|
||||||
dataKey: {
|
|
||||||
type: String,
|
|
||||||
required: true,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
const suppliers = ref([]);
|
const suppliers = ref([]);
|
||||||
const suppliersRef = ref();
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<FetchData
|
<VnFilterPanel :data-key="dataKey" :search-button="true">
|
||||||
ref="suppliersRef"
|
|
||||||
url="Suppliers"
|
|
||||||
:filter="{ fields: ['id', 'nickname'] }"
|
|
||||||
order="nickname"
|
|
||||||
limit="30"
|
|
||||||
@on-fetch="(data) => (suppliers = data)"
|
|
||||||
auto-load
|
|
||||||
/>
|
|
||||||
<VnFilterPanel :data-key="props.dataKey" :search-button="true">
|
|
||||||
<template #tags="{ tag, formatFn }">
|
<template #tags="{ tag, formatFn }">
|
||||||
<div class="q-gutter-x-xs">
|
<div class="q-gutter-x-xs">
|
||||||
<strong>{{ t(`params.${tag.label}`) }}: </strong>
|
<strong>{{ t(`params.${tag.label}`) }}: </strong>
|
||||||
|
@ -38,22 +22,6 @@ const suppliersRef = ref();
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<template #body="{ params, searchFn }">
|
<template #body="{ params, searchFn }">
|
||||||
<QItem>
|
|
||||||
<QItemSection>
|
|
||||||
<VnSelect
|
|
||||||
:label="t('params.supplierFk')"
|
|
||||||
v-model="params.supplierFk"
|
|
||||||
:options="suppliers"
|
|
||||||
option-value="id"
|
|
||||||
option-label="nickname"
|
|
||||||
@input-value="suppliersRef.fetch()"
|
|
||||||
dense
|
|
||||||
outlined
|
|
||||||
rounded
|
|
||||||
>
|
|
||||||
</VnSelect>
|
|
||||||
</QItemSection>
|
|
||||||
</QItem>
|
|
||||||
<QItem>
|
<QItem>
|
||||||
<QItemSection>
|
<QItemSection>
|
||||||
<VnInput
|
<VnInput
|
||||||
|
@ -68,21 +36,11 @@ const suppliersRef = ref();
|
||||||
</VnInput>
|
</VnInput>
|
||||||
</QItemSection>
|
</QItemSection>
|
||||||
</QItem>
|
</QItem>
|
||||||
<QItem>
|
|
||||||
<QItemSection>
|
|
||||||
<VnInputDate :label="t('From')" v-model="params.from" is-outlined />
|
|
||||||
</QItemSection>
|
|
||||||
</QItem>
|
|
||||||
<QItem>
|
|
||||||
<QItemSection>
|
|
||||||
<VnInputDate :label="t('To')" v-model="params.to" is-outlined />
|
|
||||||
</QItemSection>
|
|
||||||
</QItem>
|
|
||||||
<QItem>
|
<QItem>
|
||||||
<QItemSection>
|
<QItemSection>
|
||||||
<VnInput
|
<VnInput
|
||||||
:label="t('params.serial')"
|
:label="t('params.fi')"
|
||||||
v-model="params.serial"
|
v-model="params.fi"
|
||||||
is-outlined
|
is-outlined
|
||||||
lazy-rules
|
lazy-rules
|
||||||
>
|
>
|
||||||
|
@ -92,11 +50,61 @@ const suppliersRef = ref();
|
||||||
</VnInput>
|
</VnInput>
|
||||||
</QItemSection>
|
</QItemSection>
|
||||||
</QItem>
|
</QItem>
|
||||||
|
<QItem>
|
||||||
|
<QItemSection>
|
||||||
|
<VnSelect
|
||||||
|
v-model="params.supplierFk"
|
||||||
|
url="Suppliers"
|
||||||
|
:fields="['id', 'nickname']"
|
||||||
|
:label="t('params.supplierFk')"
|
||||||
|
option-value="id"
|
||||||
|
option-label="nickname"
|
||||||
|
:options="suppliers"
|
||||||
|
dense
|
||||||
|
outlined
|
||||||
|
rounded
|
||||||
|
>
|
||||||
|
</VnSelect>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
<QItem>
|
||||||
|
<QItemSection>
|
||||||
|
<VnInput
|
||||||
|
:label="t('params.account')"
|
||||||
|
v-model="params.account"
|
||||||
|
is-outlined
|
||||||
|
lazy-rules
|
||||||
|
>
|
||||||
|
<template #prepend>
|
||||||
|
<QIcon name="person" size="sm" />
|
||||||
|
</template>
|
||||||
|
</VnInput>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
<QItem>
|
<QItem>
|
||||||
<QItemSection>
|
<QItemSection>
|
||||||
<VnCurrency v-model="params.amount" is-outlined />
|
<VnCurrency v-model="params.amount" is-outlined />
|
||||||
</QItemSection>
|
</QItemSection>
|
||||||
</QItem>
|
</QItem>
|
||||||
|
<QItem>
|
||||||
|
<QItemSection>
|
||||||
|
<VnInputDate :label="t('From')" v-model="params.from" is-outlined />
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
<QItem>
|
||||||
|
<QItemSection>
|
||||||
|
<VnInputDate :label="t('To')" v-model="params.to" is-outlined />
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
<QItem>
|
||||||
|
<QItemSection>
|
||||||
|
<VnInputDate
|
||||||
|
:label="t('Issued')"
|
||||||
|
v-model="params.issued"
|
||||||
|
is-outlined
|
||||||
|
/>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
<QItem class="q-mb-md">
|
<QItem class="q-mb-md">
|
||||||
<QItemSection>
|
<QItemSection>
|
||||||
<QCheckbox
|
<QCheckbox
|
||||||
|
@ -111,8 +119,8 @@ const suppliersRef = ref();
|
||||||
<QItem>
|
<QItem>
|
||||||
<QItemSection>
|
<QItemSection>
|
||||||
<VnInput
|
<VnInput
|
||||||
:label="t('params.fi')"
|
:label="t('params.serialNumber')"
|
||||||
v-model="params.fi"
|
v-model="params.serialNumber"
|
||||||
is-outlined
|
is-outlined
|
||||||
lazy-rules
|
lazy-rules
|
||||||
>
|
>
|
||||||
|
@ -125,8 +133,8 @@ const suppliersRef = ref();
|
||||||
<QItem>
|
<QItem>
|
||||||
<QItemSection>
|
<QItemSection>
|
||||||
<VnInput
|
<VnInput
|
||||||
:label="t('params.serialNumber')"
|
:label="t('params.serial')"
|
||||||
v-model="params.serialNumber"
|
v-model="params.serial"
|
||||||
is-outlined
|
is-outlined
|
||||||
lazy-rules
|
lazy-rules
|
||||||
>
|
>
|
||||||
|
@ -150,29 +158,6 @@ const suppliersRef = ref();
|
||||||
</VnInput>
|
</VnInput>
|
||||||
</QItemSection>
|
</QItemSection>
|
||||||
</QItem>
|
</QItem>
|
||||||
<QItem>
|
|
||||||
<QItemSection>
|
|
||||||
<VnInput
|
|
||||||
:label="t('params.account')"
|
|
||||||
v-model="params.account"
|
|
||||||
is-outlined
|
|
||||||
lazy-rules
|
|
||||||
>
|
|
||||||
<template #prepend>
|
|
||||||
<QIcon name="person" size="sm" />
|
|
||||||
</template>
|
|
||||||
</VnInput>
|
|
||||||
</QItemSection>
|
|
||||||
</QItem>
|
|
||||||
<QItem>
|
|
||||||
<QItemSection>
|
|
||||||
<VnInputDate
|
|
||||||
:label="t('Issued')"
|
|
||||||
v-model="params.issued"
|
|
||||||
is-outlined
|
|
||||||
/>
|
|
||||||
</QItemSection>
|
|
||||||
</QItem>
|
|
||||||
</QExpansionItem>
|
</QExpansionItem>
|
||||||
</template>
|
</template>
|
||||||
</VnFilterPanel>
|
</VnFilterPanel>
|
||||||
|
|
|
@ -13,6 +13,7 @@ import InvoiceInFilter from './InvoiceInFilter.vue';
|
||||||
import { getUrl } from 'src/composables/getUrl';
|
import { getUrl } from 'src/composables/getUrl';
|
||||||
import InvoiceInSummary from './Card/InvoiceInSummary.vue';
|
import InvoiceInSummary from './Card/InvoiceInSummary.vue';
|
||||||
import { useSummaryDialog } from 'src/composables/useSummaryDialog';
|
import { useSummaryDialog } from 'src/composables/useSummaryDialog';
|
||||||
|
import SupplierDescriptorProxy from 'src/pages/Supplier/Card/SupplierDescriptorProxy.vue';
|
||||||
|
|
||||||
const stateStore = useStateStore();
|
const stateStore = useStateStore();
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
@ -85,7 +86,15 @@ function navigate(id) {
|
||||||
<VnLv
|
<VnLv
|
||||||
:label="t('invoiceIn.list.supplier')"
|
:label="t('invoiceIn.list.supplier')"
|
||||||
:value="row.supplierName"
|
:value="row.supplierName"
|
||||||
/>
|
@click.stop
|
||||||
|
>
|
||||||
|
<template #value>
|
||||||
|
<span class="link">
|
||||||
|
{{ row.supplierName }}
|
||||||
|
<SupplierDescriptorProxy :id="row.supplierFk" />
|
||||||
|
</span>
|
||||||
|
</template>
|
||||||
|
</VnLv>
|
||||||
<VnLv
|
<VnLv
|
||||||
:label="t('invoiceIn.list.serialNumber')"
|
:label="t('invoiceIn.list.serialNumber')"
|
||||||
:value="row.serialNumber"
|
:value="row.serialNumber"
|
||||||
|
@ -137,13 +146,7 @@ function navigate(id) {
|
||||||
</div>
|
</div>
|
||||||
</QPage>
|
</QPage>
|
||||||
<QPageSticky position="bottom-right" :offset="[20, 20]">
|
<QPageSticky position="bottom-right" :offset="[20, 20]">
|
||||||
<QBtn
|
<QBtn color="primary" icon="add" size="lg" round :href="`/#/invoice-in/create`" />
|
||||||
color="primary"
|
|
||||||
icon="add"
|
|
||||||
size="lg"
|
|
||||||
round
|
|
||||||
:href="`${url}invoice-in/create`"
|
|
||||||
/>
|
|
||||||
</QPageSticky>
|
</QPageSticky>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,63 @@
|
||||||
|
<script setup>
|
||||||
|
import { useRoute } from 'vue-router';
|
||||||
|
import axios from 'axios';
|
||||||
|
import { useQuasar } from 'quasar';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import VnConfirm from 'src/components/ui/VnConfirm.vue';
|
||||||
|
import { useArrayData } from 'src/composables/useArrayData';
|
||||||
|
const { notify, dialog } = useQuasar();
|
||||||
|
const { t } = useI18n();
|
||||||
|
|
||||||
|
defineExpose({ checkToBook });
|
||||||
|
|
||||||
|
const { store } = useArrayData(useRoute().meta.moduleName);
|
||||||
|
|
||||||
|
async function checkToBook(id) {
|
||||||
|
let directBooking = true;
|
||||||
|
|
||||||
|
const { data: totals } = await axios.get(`InvoiceIns/${id}/getTotals`);
|
||||||
|
const taxableBaseNotEqualDueDay = totals.totalDueDay != totals.totalTaxableBase;
|
||||||
|
const vatNotEqualDueDay = totals.totalDueDay != totals.totalVat;
|
||||||
|
|
||||||
|
if (taxableBaseNotEqualDueDay && vatNotEqualDueDay) directBooking = false;
|
||||||
|
|
||||||
|
const { data: dueDaysCount } = await axios.get('InvoiceInDueDays/count', {
|
||||||
|
where: {
|
||||||
|
invoiceInFk: id,
|
||||||
|
dueDated: { gte: Date.vnNew() },
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (dueDaysCount) directBooking = false;
|
||||||
|
|
||||||
|
if (directBooking) return toBook(id);
|
||||||
|
|
||||||
|
dialog({
|
||||||
|
component: VnConfirm,
|
||||||
|
componentProps: { title: t('Are you sure you want to book this invoice?') },
|
||||||
|
}).onOk(async () => await toBook(id));
|
||||||
|
}
|
||||||
|
|
||||||
|
async function toBook(id) {
|
||||||
|
let type = 'positive';
|
||||||
|
let message = t('globals.dataSaved');
|
||||||
|
|
||||||
|
try {
|
||||||
|
await axios.post(`InvoiceIns/${id}/toBook`);
|
||||||
|
store.data.isBooked = true;
|
||||||
|
} catch (e) {
|
||||||
|
type = 'negative';
|
||||||
|
message = t('It was not able to book the invoice');
|
||||||
|
} finally {
|
||||||
|
notify({ type, message });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
<template>
|
||||||
|
<slot name="content" :book="checkToBook" />
|
||||||
|
</template>
|
||||||
|
<i18n>
|
||||||
|
es:
|
||||||
|
Are you sure you want to book this invoice?: ¿Estás seguro de querer asentar esta factura?
|
||||||
|
It was not able to book the invoice: No se pudo contabilizar la factura
|
||||||
|
</i18n>
|
|
@ -220,13 +220,20 @@ const onIntrastatCreated = (response, formData) => {
|
||||||
</QIcon>
|
</QIcon>
|
||||||
</div>
|
</div>
|
||||||
</VnRow>
|
</VnRow>
|
||||||
<VnRow class="row q-gutter-md q-mb-md">
|
<VnRow>
|
||||||
<QInput
|
<VnInput
|
||||||
:label="t('basicData.description')"
|
:label="t('basicData.description')"
|
||||||
type="textarea"
|
type="textarea"
|
||||||
v-model="data.description"
|
v-model="data.description"
|
||||||
fill-input
|
fill-input
|
||||||
/>
|
/>
|
||||||
|
<VnInput
|
||||||
|
v-show="data.isPhotoRequested"
|
||||||
|
type="textarea"
|
||||||
|
:label="t('globals.comment')"
|
||||||
|
v-model="data.photoMotivation"
|
||||||
|
fill-input
|
||||||
|
/>
|
||||||
</VnRow>
|
</VnRow>
|
||||||
</template>
|
</template>
|
||||||
</FormModel>
|
</FormModel>
|
||||||
|
|
|
@ -183,7 +183,7 @@ const getEntryQueryParams = (supplier) => {
|
||||||
<QTooltip>{{ t('Go to client') }}</QTooltip>
|
<QTooltip>{{ t('Go to client') }}</QTooltip>
|
||||||
</QBtn>
|
</QBtn>
|
||||||
<QBtn
|
<QBtn
|
||||||
:href="`${url}invoice-in/create?supplierFk=${entity.id}`"
|
:href="`#/invoice-in/create?supplierFk=${entity.id}`"
|
||||||
size="md"
|
size="md"
|
||||||
icon="vn:invoice-in-create"
|
icon="vn:invoice-in-create"
|
||||||
color="primary"
|
color="primary"
|
||||||
|
|
|
@ -27,7 +27,7 @@ const save = async (data) => {
|
||||||
const lockerId = data.id ?? originaLockerId.value;
|
const lockerId = data.id ?? originaLockerId.value;
|
||||||
const workerFk = lockerId == originaLockerId.value ? null : entityId.value;
|
const workerFk = lockerId == originaLockerId.value ? null : entityId.value;
|
||||||
|
|
||||||
await axios.patch(`Lockers/${lockerId}`, { workerFk });
|
return axios.patch(`Lockers/${lockerId}`, { workerFk });
|
||||||
};
|
};
|
||||||
|
|
||||||
const init = async (data) => {
|
const init = async (data) => {
|
||||||
|
|
|
@ -1,10 +1,10 @@
|
||||||
<script setup>
|
<script setup>
|
||||||
import VnTree from 'components/ui/VnTree.vue';
|
import WorkerDepartmentTree from './WorkerDepartmentTree.vue';
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<QPage class="column items-center q-pa-md">
|
<QPage class="column items-center q-pa-md">
|
||||||
<VnTree />
|
<WorkerDepartmentTree />
|
||||||
</QPage>
|
</QPage>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
|
|
@ -4,7 +4,7 @@ import { useI18n } from 'vue-i18n';
|
||||||
import { useState } from 'src/composables/useState';
|
import { useState } from 'src/composables/useState';
|
||||||
import { useQuasar } from 'quasar';
|
import { useQuasar } from 'quasar';
|
||||||
import DepartmentDescriptorProxy from 'src/pages/Department/Card/DepartmentDescriptorProxy.vue';
|
import DepartmentDescriptorProxy from 'src/pages/Department/Card/DepartmentDescriptorProxy.vue';
|
||||||
import CreateDepartmentChild from '../CreateDepartmentChild.vue';
|
import CreateDepartmentChild from './CreateDepartmentChild.vue';
|
||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
import useNotify from 'src/composables/useNotify.js';
|
import useNotify from 'src/composables/useNotify.js';
|
||||||
import { useRouter } from 'vue-router';
|
import { useRouter } from 'vue-router';
|
|
@ -5,38 +5,29 @@ import { computed } from 'vue';
|
||||||
|
|
||||||
import VnCard from 'components/common/VnCard.vue';
|
import VnCard from 'components/common/VnCard.vue';
|
||||||
import ZoneDescriptor from './ZoneDescriptor.vue';
|
import ZoneDescriptor from './ZoneDescriptor.vue';
|
||||||
import VnSearchbar from 'src/components/ui/VnSearchbar.vue';
|
|
||||||
|
|
||||||
import { useStateStore } from 'stores/useStateStore';
|
|
||||||
|
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
const stateStore = useStateStore();
|
|
||||||
const route = useRoute();
|
const route = useRoute();
|
||||||
|
|
||||||
const routeName = computed(() => route.name);
|
const routeName = computed(() => route.name);
|
||||||
|
const customRouteRedirectName = computed(() => {
|
||||||
|
if (routeName.value === 'ZoneLocations') return null;
|
||||||
|
return routeName.value;
|
||||||
|
});
|
||||||
const searchBarDataKeys = {
|
const searchBarDataKeys = {
|
||||||
ZoneWarehouses: 'ZoneWarehouses',
|
ZoneWarehouses: 'ZoneWarehouses',
|
||||||
ZoneSummary: 'ZoneSummary',
|
ZoneSummary: 'ZoneSummary',
|
||||||
|
ZoneLocations: 'ZoneLocations',
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
<template>
|
<template>
|
||||||
<template v-if="stateStore.isHeaderMounted()">
|
|
||||||
<Teleport to="#searchbar">
|
|
||||||
<VnSearchbar
|
|
||||||
:data-key="searchBarDataKeys[routeName]"
|
|
||||||
:custom-route-redirect-name="routeName"
|
|
||||||
:label="t('list.searchZone')"
|
|
||||||
:info="t('list.searchInfo')"
|
|
||||||
/>
|
|
||||||
</Teleport>
|
|
||||||
</template>
|
|
||||||
<VnCard
|
<VnCard
|
||||||
data-key="Zone"
|
data-key="Zone"
|
||||||
base-url="Zones"
|
|
||||||
:descriptor="ZoneDescriptor"
|
:descriptor="ZoneDescriptor"
|
||||||
searchbar-data-key="ZoneList"
|
:search-data-key="searchBarDataKeys[routeName]"
|
||||||
searchbar-url="Zones"
|
:search-custom-route-redirect="customRouteRedirectName"
|
||||||
searchbar-label="Search zones"
|
:search-redirect="!!customRouteRedirectName"
|
||||||
searchbar-info="You can search by zone reference"
|
:searchbar-label="t('list.searchZone')"
|
||||||
|
:searchbar-info="t('list.searchInfo')"
|
||||||
/>
|
/>
|
||||||
</template>
|
</template>
|
||||||
|
|
|
@ -1 +1,80 @@
|
||||||
<template>Zone Locations</template>
|
<script setup>
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import { useRoute } from 'vue-router';
|
||||||
|
|
||||||
|
import ZoneLocationsTree from './ZoneLocationsTree.vue';
|
||||||
|
|
||||||
|
import axios from 'axios';
|
||||||
|
|
||||||
|
const { t } = useI18n();
|
||||||
|
const route = useRoute();
|
||||||
|
|
||||||
|
const onSelected = async (val, node) => {
|
||||||
|
try {
|
||||||
|
if (val === null) val = undefined;
|
||||||
|
const params = { geoId: node.id, isIncluded: val };
|
||||||
|
await axios.post(`Zones/${route.params.id}/toggleIsIncluded`, params);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Error updating included', err);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<QPage class="column items-center q-pa-md">
|
||||||
|
<QCard class="full-width q-pa-md" style="max-width: 800px">
|
||||||
|
<ZoneLocationsTree :root-label="t('zoneLocations.locations')">
|
||||||
|
<template #checkbox="{ node }">
|
||||||
|
<QCheckbox
|
||||||
|
v-if="node.id"
|
||||||
|
v-model="node.selected"
|
||||||
|
:label="node.name"
|
||||||
|
@update:model-value="($event) => onSelected($event, node)"
|
||||||
|
toggle-indeterminate
|
||||||
|
color="transparent"
|
||||||
|
:class="[
|
||||||
|
'checkbox',
|
||||||
|
node.selected
|
||||||
|
? '--checked'
|
||||||
|
: node.selected == false
|
||||||
|
? '--unchecked'
|
||||||
|
: '--indeterminate',
|
||||||
|
]"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
</ZoneLocationsTree>
|
||||||
|
</QCard>
|
||||||
|
</QPage>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style lang="scss">
|
||||||
|
.checkbox {
|
||||||
|
&.--checked {
|
||||||
|
.q-checkbox__bg {
|
||||||
|
border: 1px solid $info !important;
|
||||||
|
}
|
||||||
|
.q-checkbox__svg {
|
||||||
|
color: white !important;
|
||||||
|
background-color: $info !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
&.--unchecked {
|
||||||
|
.q-checkbox__bg {
|
||||||
|
border: 1px solid $negative !important;
|
||||||
|
}
|
||||||
|
.q-checkbox__svg {
|
||||||
|
background-color: $negative !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
&.--indeterminate {
|
||||||
|
.q-checkbox__bg {
|
||||||
|
border: 1px solid $white !important;
|
||||||
|
}
|
||||||
|
.q-checkbox__svg {
|
||||||
|
color: transparent !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
|
@ -0,0 +1,172 @@
|
||||||
|
<script setup>
|
||||||
|
import { onMounted, ref, computed, watch } from 'vue';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import { useRoute } from 'vue-router';
|
||||||
|
|
||||||
|
import { useState } from 'src/composables/useState';
|
||||||
|
import axios from 'axios';
|
||||||
|
import { useArrayData } from 'composables/useArrayData';
|
||||||
|
import { onUnmounted } from 'vue';
|
||||||
|
|
||||||
|
const { t } = useI18n();
|
||||||
|
const route = useRoute();
|
||||||
|
const state = useState();
|
||||||
|
|
||||||
|
const treeRef = ref();
|
||||||
|
const expanded = ref([]);
|
||||||
|
|
||||||
|
const arrayData = useArrayData('ZoneLocations', {
|
||||||
|
url: `Zones/${route.params.id}/getLeaves`,
|
||||||
|
});
|
||||||
|
const { store } = arrayData;
|
||||||
|
const storeData = computed(() => store.data);
|
||||||
|
|
||||||
|
const nodes = ref([
|
||||||
|
{ id: null, name: t('zoneLocations.locations'), sons: true, childs: [{}] },
|
||||||
|
]);
|
||||||
|
|
||||||
|
const previousExpandedNodes = ref(new Set());
|
||||||
|
|
||||||
|
const onNodeExpanded = async (nodeKeysArray) => {
|
||||||
|
let nodeKeysSet = new Set(nodeKeysArray);
|
||||||
|
const lastNodeKey = nodeKeysArray.at(-1);
|
||||||
|
|
||||||
|
if (!nodeKeysSet.has(null)) return;
|
||||||
|
|
||||||
|
const wasExpanded = !previousExpandedNodes.value.has(lastNodeKey);
|
||||||
|
if (wasExpanded) await fetchNodeLeaves(lastNodeKey);
|
||||||
|
else {
|
||||||
|
const difference = new Set(
|
||||||
|
[...previousExpandedNodes.value].filter((x) => !nodeKeysSet.has(x))
|
||||||
|
);
|
||||||
|
const collapsedNode = Array.from(difference).pop();
|
||||||
|
const node = treeRef.value?.getNodeByKey(collapsedNode);
|
||||||
|
const allNodeIds = getNodeIds(node);
|
||||||
|
expanded.value = expanded.value.filter((id) => !allNodeIds.includes(id));
|
||||||
|
}
|
||||||
|
previousExpandedNodes.value = nodeKeysSet;
|
||||||
|
};
|
||||||
|
|
||||||
|
const formatNodeSelected = (node) => {
|
||||||
|
if (node.selected === 1) node.selected = true;
|
||||||
|
else if (node.selected === 0) node.selected = false;
|
||||||
|
|
||||||
|
if (node.childs && node.childs.length > 0) {
|
||||||
|
expanded.value.push(node.id);
|
||||||
|
|
||||||
|
node.childs.forEach((childNode) => {
|
||||||
|
formatNodeSelected(childNode);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (node.sons > 0 && !node.childs) node.childs = [{}];
|
||||||
|
};
|
||||||
|
|
||||||
|
const fetchNodeLeaves = async (nodeKey) => {
|
||||||
|
try {
|
||||||
|
const node = treeRef.value?.getNodeByKey(nodeKey);
|
||||||
|
if (!node || node.sons === 0) return;
|
||||||
|
|
||||||
|
const params = { parentId: node.id };
|
||||||
|
const response = await axios.get(`Zones/${route.params.id}/getLeaves`, {
|
||||||
|
params,
|
||||||
|
});
|
||||||
|
if (response.data) {
|
||||||
|
node.childs = response.data.map((n) => {
|
||||||
|
formatNodeSelected(n);
|
||||||
|
return n;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
state.set('Tree', node);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Error fetching department leaves', err);
|
||||||
|
throw new Error();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
function getNodeIds(node) {
|
||||||
|
let ids = [];
|
||||||
|
if (node.id) ids.push(node.id);
|
||||||
|
|
||||||
|
if (node.childs)
|
||||||
|
node.childs.forEach((child) => {
|
||||||
|
ids = ids.concat(getNodeIds(child));
|
||||||
|
});
|
||||||
|
return ids;
|
||||||
|
}
|
||||||
|
|
||||||
|
watch(storeData, async (val) => {
|
||||||
|
// Se triggerea cuando se actualiza el store.data, el cual es el resultado del fetch de la searchbar
|
||||||
|
nodes.value[0].childs = [...val];
|
||||||
|
const fetchedNodeKeys = val.flatMap(getNodeIds);
|
||||||
|
state.set('Tree', [...fetchedNodeKeys]);
|
||||||
|
|
||||||
|
if (store.userParams?.search === '') {
|
||||||
|
val.forEach((n) => {
|
||||||
|
formatNodeSelected(n);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
for (let n of state.get('Tree')) {
|
||||||
|
await fetchNodeLeaves(n);
|
||||||
|
}
|
||||||
|
expanded.value = [null, ...fetchedNodeKeys];
|
||||||
|
}
|
||||||
|
previousExpandedNodes.value = new Set(expanded.value);
|
||||||
|
});
|
||||||
|
|
||||||
|
onMounted(async () => {
|
||||||
|
if (store.userParams?.search) {
|
||||||
|
await arrayData.fetch({ append: false });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const stateTree = state.get('Tree');
|
||||||
|
const tree = stateTree ? [...state.get('Tree')] : [null];
|
||||||
|
const lastStateTree = state.get('TreeState');
|
||||||
|
if (tree) {
|
||||||
|
for (let n of tree) {
|
||||||
|
await fetchNodeLeaves(n);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (lastStateTree) {
|
||||||
|
tree.push(lastStateTree);
|
||||||
|
await fetchNodeLeaves(lastStateTree);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setTimeout(() => {
|
||||||
|
if (lastStateTree) {
|
||||||
|
document.getElementById(lastStateTree).scrollIntoView();
|
||||||
|
}
|
||||||
|
}, 1000);
|
||||||
|
|
||||||
|
previousExpandedNodes.value = new Set(expanded.value);
|
||||||
|
});
|
||||||
|
|
||||||
|
onUnmounted(() => {
|
||||||
|
state.set('Tree', undefined);
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<QTree
|
||||||
|
ref="treeRef"
|
||||||
|
:nodes="nodes"
|
||||||
|
node-key="id"
|
||||||
|
label-key="name"
|
||||||
|
children-key="childs"
|
||||||
|
v-model:expanded="expanded"
|
||||||
|
@update:expanded="onNodeExpanded($event)"
|
||||||
|
:default-expand-all="true"
|
||||||
|
>
|
||||||
|
<template #default-header="{ node }">
|
||||||
|
<div
|
||||||
|
:id="node.id"
|
||||||
|
class="qtr row justify-between full-width q-pr-md cursor-pointer"
|
||||||
|
>
|
||||||
|
<span v-if="!node.id">{{ node.name }}</span>
|
||||||
|
<slot name="checkbox" :node="node" />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</QTree>
|
||||||
|
</template>
|
|
@ -42,6 +42,8 @@ summary:
|
||||||
filterPanel:
|
filterPanel:
|
||||||
name: Name
|
name: Name
|
||||||
agencyModeFk: Agency
|
agencyModeFk: Agency
|
||||||
|
zoneLocations:
|
||||||
|
locations: Locations
|
||||||
deliveryPanel:
|
deliveryPanel:
|
||||||
pickup: Pick up
|
pickup: Pick up
|
||||||
delivery: Delivery
|
delivery: Delivery
|
||||||
|
|
|
@ -42,6 +42,8 @@ summary:
|
||||||
filterPanel:
|
filterPanel:
|
||||||
name: Nombre
|
name: Nombre
|
||||||
agencyModeFk: Agencia
|
agencyModeFk: Agencia
|
||||||
|
zoneLocations:
|
||||||
|
locations: Localizaciones
|
||||||
deliveryPanel:
|
deliveryPanel:
|
||||||
pickup: Recogida
|
pickup: Recogida
|
||||||
delivery: Entrega
|
delivery: Entrega
|
||||||
|
|
|
@ -37,6 +37,15 @@ export default {
|
||||||
},
|
},
|
||||||
component: () => import('src/pages/InvoiceIn/InvoiceInList.vue'),
|
component: () => import('src/pages/InvoiceIn/InvoiceInList.vue'),
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
path: 'create',
|
||||||
|
name: 'InvoiceInCreare',
|
||||||
|
meta: {
|
||||||
|
title: 'invoiceInCreate',
|
||||||
|
icon: 'create',
|
||||||
|
},
|
||||||
|
component: () => import('src/pages/InvoiceIn/InvoiceInCreate.vue'),
|
||||||
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|
|
@ -106,7 +106,7 @@ export default {
|
||||||
path: 'location',
|
path: 'location',
|
||||||
meta: {
|
meta: {
|
||||||
title: 'locations',
|
title: 'locations',
|
||||||
icon: 'vn:greuge',
|
icon: 'my_location',
|
||||||
},
|
},
|
||||||
component: () => import('src/pages/Zone/Card/ZoneLocations.vue'),
|
component: () => import('src/pages/Zone/Card/ZoneLocations.vue'),
|
||||||
},
|
},
|
||||||
|
|
|
@ -2,8 +2,7 @@
|
||||||
describe('InvoiceInBasicData', () => {
|
describe('InvoiceInBasicData', () => {
|
||||||
const formInputs = '.q-form > .q-card input';
|
const formInputs = '.q-form > .q-card input';
|
||||||
const firstFormSelect = '.q-card > .vn-row:nth-child(1) > .q-select';
|
const firstFormSelect = '.q-card > .vn-row:nth-child(1) > .q-select';
|
||||||
const appendBtns = '.q-form label button';
|
const documentBtns = '.q-form .q-field button';
|
||||||
const dialogAppendBtns = '.q-dialog label button';
|
|
||||||
const dialogInputs = '.q-dialog input';
|
const dialogInputs = '.q-dialog input';
|
||||||
const dialogActionBtns = '.q-card__actions button';
|
const dialogActionBtns = '.q-card__actions button';
|
||||||
|
|
||||||
|
@ -15,8 +14,7 @@ describe('InvoiceInBasicData', () => {
|
||||||
it('should edit the provideer and supplier ref', () => {
|
it('should edit the provideer and supplier ref', () => {
|
||||||
cy.selectOption(firstFormSelect, 'Bros');
|
cy.selectOption(firstFormSelect, 'Bros');
|
||||||
cy.get('[title="Reset"]').click();
|
cy.get('[title="Reset"]').click();
|
||||||
cy.get(appendBtns).eq(0).click();
|
cy.get(formInputs).eq(1).type('{selectall}4739');
|
||||||
cy.get(formInputs).eq(1).type(4739);
|
|
||||||
cy.saveCard();
|
cy.saveCard();
|
||||||
|
|
||||||
cy.get(`${firstFormSelect} input`).invoke('val').should('eq', 'Plants nick');
|
cy.get(`${firstFormSelect} input`).invoke('val').should('eq', 'Plants nick');
|
||||||
|
@ -27,22 +25,20 @@ describe('InvoiceInBasicData', () => {
|
||||||
const firtsInput = 'Ticket:65';
|
const firtsInput = 'Ticket:65';
|
||||||
const secondInput = "I don't know what posting here!";
|
const secondInput = "I don't know what posting here!";
|
||||||
|
|
||||||
cy.get(appendBtns).eq(3).click();
|
cy.get(documentBtns).eq(1).click();
|
||||||
cy.get(dialogAppendBtns).eq(0).click();
|
cy.get(dialogInputs).eq(0).type(`{selectall}${firtsInput}`);
|
||||||
cy.get(dialogInputs).eq(0).type(firtsInput);
|
cy.get('textarea').type(`{selectall}${secondInput}`);
|
||||||
cy.get(dialogAppendBtns).eq(1).click();
|
|
||||||
cy.get('textarea').type(secondInput);
|
|
||||||
cy.get(dialogActionBtns).eq(1).click();
|
cy.get(dialogActionBtns).eq(1).click();
|
||||||
|
|
||||||
cy.get(appendBtns).eq(3).click();
|
cy.get(documentBtns).eq(1).click();
|
||||||
|
|
||||||
cy.get(dialogInputs).eq(0).invoke('val').should('eq', firtsInput);
|
cy.get(dialogInputs).eq(0).invoke('val').should('eq', firtsInput);
|
||||||
cy.get('textarea').invoke('val').should('eq', secondInput);
|
cy.get('textarea').invoke('val').should('eq', secondInput);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should throw an error creating a new dms if a file is not attached', () => {
|
it('should throw an error creating a new dms if a file is not attached', () => {
|
||||||
cy.get(appendBtns).eq(2).click();
|
cy.get(formInputs).eq(5).click();
|
||||||
cy.get(appendBtns).eq(1).click();
|
cy.get(formInputs).eq(5).type('{selectall}{backspace}');
|
||||||
|
cy.get(documentBtns).eq(0).click();
|
||||||
cy.get(dialogActionBtns).eq(1).click();
|
cy.get(dialogActionBtns).eq(1).click();
|
||||||
cy.get('.q-notification__message').should(
|
cy.get('.q-notification__message').should(
|
||||||
'have.text',
|
'have.text',
|
||||||
|
|
|
@ -0,0 +1,24 @@
|
||||||
|
describe('InvoiceInDescriptor', () => {
|
||||||
|
const dialogBtns = '.q-card__actions button';
|
||||||
|
const firstDescritorOpt = '.q-menu > .q-list > :nth-child(1) > .q-item__section';
|
||||||
|
const isBookedField =
|
||||||
|
'.q-card:nth-child(3) .vn-label-value:nth-child(5) > .value > span';
|
||||||
|
|
||||||
|
it('should booking and unbooking the invoice properly', () => {
|
||||||
|
cy.login('developer');
|
||||||
|
cy.visit(`/#/invoice-in/1/summary?limit=10`);
|
||||||
|
|
||||||
|
cy.openLeftMenu();
|
||||||
|
cy.openActionsDescriptor();
|
||||||
|
cy.get(firstDescritorOpt).click();
|
||||||
|
cy.get(dialogBtns).eq(1).click();
|
||||||
|
cy.get('.fullscreen').first().click();
|
||||||
|
cy.get(isBookedField).should('have.attr', 'title', 'true');
|
||||||
|
|
||||||
|
cy.openLeftMenu();
|
||||||
|
cy.openActionsDescriptor();
|
||||||
|
cy.get(firstDescritorOpt).click();
|
||||||
|
cy.get(dialogBtns).eq(1).click();
|
||||||
|
cy.get(isBookedField).should('have.attr', 'title', 'false');
|
||||||
|
});
|
||||||
|
});
|
|
@ -18,7 +18,7 @@ describe('InvoiceInVat', () => {
|
||||||
cy.saveCard();
|
cy.saveCard();
|
||||||
cy.visit(`/#/invoice-in/1/vat`);
|
cy.visit(`/#/invoice-in/1/vat`);
|
||||||
|
|
||||||
cy.getValue(firstLineVat).should('equal', 'H.P. IVA 21% CEE');
|
cy.getValue(firstLineVat).should('equal', '8');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should add a new row', () => {
|
it('should add a new row', () => {
|
||||||
|
|
|
@ -9,15 +9,15 @@ describe('VnLog', () => {
|
||||||
cy.visit(`/#/claim/${1}/log`);
|
cy.visit(`/#/claim/${1}/log`);
|
||||||
cy.openRightMenu();
|
cy.openRightMenu();
|
||||||
});
|
});
|
||||||
|
// Se tiene que cambiar el Accept-Language a 'en', ya hay una tarea para eso #7189.
|
||||||
it('should filter by insert actions', () => {
|
xit('should filter by insert actions', () => {
|
||||||
cy.checkOption(':nth-child(7) > .q-checkbox');
|
cy.checkOption(':nth-child(7) > .q-checkbox');
|
||||||
cy.get('.q-page').click();
|
cy.get('.q-page').click();
|
||||||
cy.validateContent(chips[0], 'Document');
|
cy.validateContent(chips[0], 'Document');
|
||||||
cy.validateContent(chips[1], 'Beginning');
|
cy.validateContent(chips[1], 'Beginning');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should filter by entity', () => {
|
xit('should filter by entity', () => {
|
||||||
cy.selectOption('.q-drawer--right .q-item > .q-select', 'Claim');
|
cy.selectOption('.q-drawer--right .q-item > .q-select', 'Claim');
|
||||||
cy.get('.q-page').click();
|
cy.get('.q-page').click();
|
||||||
cy.validateContent(chips[0], 'Claim');
|
cy.validateContent(chips[0], 'Claim');
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
describe('WorkerList', () => {
|
describe('WorkerLocker', () => {
|
||||||
const workerId = 1109;
|
const workerId = 1109;
|
||||||
const lockerCode = '2F';
|
const lockerCode = '2F';
|
||||||
const input = '.q-card input';
|
const input = '.q-card input';
|
||||||
|
|
|
@ -92,8 +92,13 @@ Cypress.Commands.add('checkOption', (selector) => {
|
||||||
|
|
||||||
// Global buttons
|
// Global buttons
|
||||||
Cypress.Commands.add('saveCard', () => {
|
Cypress.Commands.add('saveCard', () => {
|
||||||
|
const dropdownArrow = '.q-btn-dropdown__arrow-container > .q-btn__content > .q-icon';
|
||||||
|
cy.get('#st-actions').then(($el) => {
|
||||||
|
if ($el.find(dropdownArrow).length) cy.get(dropdownArrow).click();
|
||||||
|
});
|
||||||
cy.get('[title="Save"]').click();
|
cy.get('[title="Save"]').click();
|
||||||
});
|
});
|
||||||
|
|
||||||
Cypress.Commands.add('resetCard', () => {
|
Cypress.Commands.add('resetCard', () => {
|
||||||
cy.get('[title="Reset"]').click();
|
cy.get('[title="Reset"]').click();
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,34 +0,0 @@
|
||||||
import { vi, describe, expect, it, beforeAll } from 'vitest';
|
|
||||||
import { createWrapper, axios } from 'app/test/vitest/helper';
|
|
||||||
import InvoiceInBasicData from 'src/pages/InvoiceIn/Card/InvoiceInBasicData.vue';
|
|
||||||
|
|
||||||
describe('InvoiceInBasicData', () => {
|
|
||||||
let vm;
|
|
||||||
|
|
||||||
beforeAll(() => {
|
|
||||||
vm = createWrapper(InvoiceInBasicData, {
|
|
||||||
global: {
|
|
||||||
stubs: [],
|
|
||||||
mocks: {
|
|
||||||
fetch: vi.fn(),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}).vm;
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('upsert()', () => {
|
|
||||||
it('should throw an error when data is empty', async () => {
|
|
||||||
vi.spyOn(axios, 'post').mockResolvedValue({ data: [] });
|
|
||||||
vi.spyOn(vm.quasar, 'notify');
|
|
||||||
|
|
||||||
await vm.upsert();
|
|
||||||
|
|
||||||
expect(vm.quasar.notify).toHaveBeenCalledWith(
|
|
||||||
expect.objectContaining({
|
|
||||||
message: `The company can't be empty`,
|
|
||||||
type: 'negative',
|
|
||||||
})
|
|
||||||
);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
|
@ -19,13 +19,13 @@ describe('InvoiceInIntrastat', () => {
|
||||||
|
|
||||||
describe('getTotal()', () => {
|
describe('getTotal()', () => {
|
||||||
it('should correctly handle the sum', () => {
|
it('should correctly handle the sum', () => {
|
||||||
vm.invoceInIntrastat = [
|
const invoceInIntrastat = [
|
||||||
{ amount: 10, stems: 162 },
|
{ amount: 10, stems: 162 },
|
||||||
{ amount: 20, stems: 21 },
|
{ amount: 20, stems: 21 },
|
||||||
];
|
];
|
||||||
|
|
||||||
const totalAmount = vm.getTotal('amount');
|
const totalAmount = vm.getTotal(invoceInIntrastat, 'amount');
|
||||||
const totalStems = vm.getTotal('stems');
|
const totalStems = vm.getTotal(invoceInIntrastat, 'stems');
|
||||||
|
|
||||||
expect(totalAmount).toBe(10 + 20);
|
expect(totalAmount).toBe(10 + 20);
|
||||||
expect(totalStems).toBe(162 + 21);
|
expect(totalStems).toBe(162 + 21);
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import { vi, describe, expect, it, beforeAll } from 'vitest';
|
import { vi, describe, expect, it, beforeAll } from 'vitest';
|
||||||
import { createWrapper, axios } from 'app/test/vitest/helper';
|
import { createWrapper } from 'app/test/vitest/helper';
|
||||||
import InvoiceInVat from 'src/pages/InvoiceIn/Card/InvoiceInVat.vue';
|
import InvoiceInVat from 'src/pages/InvoiceIn/Card/InvoiceInVat.vue';
|
||||||
|
|
||||||
describe('InvoiceInVat', () => {
|
describe('InvoiceInVat', () => {
|
||||||
|
@ -16,41 +16,6 @@ describe('InvoiceInVat', () => {
|
||||||
}).vm;
|
}).vm;
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('addExpense()', () => {
|
|
||||||
beforeAll(() => {
|
|
||||||
vi.spyOn(axios, 'post').mockResolvedValue({ data: [] });
|
|
||||||
vi.spyOn(axios, 'get').mockResolvedValue({ data: [] });
|
|
||||||
vi.spyOn(vm.quasar, 'notify');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should throw an error when the code property is undefined', async () => {
|
|
||||||
await vm.addExpense();
|
|
||||||
|
|
||||||
expect(vm.quasar.notify).toHaveBeenCalledWith(
|
|
||||||
expect.objectContaining({
|
|
||||||
message: `The code can't be empty`,
|
|
||||||
type: 'negative',
|
|
||||||
})
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should correctly handle expense addition', async () => {
|
|
||||||
vm.newExpense = {
|
|
||||||
code: 123,
|
|
||||||
isWithheld: false,
|
|
||||||
description: 'Descripción del gasto',
|
|
||||||
};
|
|
||||||
|
|
||||||
await vm.addExpense();
|
|
||||||
expect(vm.quasar.notify).toHaveBeenCalledWith(
|
|
||||||
expect.objectContaining({
|
|
||||||
message: 'Data saved',
|
|
||||||
type: 'positive',
|
|
||||||
})
|
|
||||||
);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('taxRate()', () => {
|
describe('taxRate()', () => {
|
||||||
it('should correctly compute the tax rate', () => {
|
it('should correctly compute the tax rate', () => {
|
||||||
const invoiceInTax = { taxableBase: 100, taxTypeSageFk: 1 };
|
const invoiceInTax = { taxableBase: 100, taxTypeSageFk: 1 };
|
||||||
|
|
|
@ -24,6 +24,7 @@ vi.mock('vue-router', () => ({
|
||||||
params: {
|
params: {
|
||||||
id: 1,
|
id: 1,
|
||||||
},
|
},
|
||||||
|
meta: { moduleName: 'mockName' },
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
|
@ -31,6 +32,7 @@ vi.mock('vue-router', () => ({
|
||||||
matched: [],
|
matched: [],
|
||||||
query: {},
|
query: {},
|
||||||
params: {},
|
params: {},
|
||||||
|
meta: { moduleName: 'mockName' },
|
||||||
}),
|
}),
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue