Compare commits
32 Commits
dev
...
6683-creat
Author | SHA1 | Date |
---|---|---|
|
d943529f75 | |
|
601b8d8d64 | |
|
5527a7fea8 | |
|
49218511d9 | |
|
80d56a5468 | |
|
db3d17179e | |
|
36fa2c72e0 | |
|
0ea9381287 | |
|
a2fdd6820c | |
|
561b607163 | |
|
bcf5750619 | |
|
46826b6b95 | |
|
5690ab7ce1 | |
|
1fc0e4d958 | |
|
2bfa8368ab | |
|
56c1373b8e | |
|
c4b76b465e | |
|
105e95e68a | |
|
223890a1bc | |
|
bf6d4f0c5f | |
|
c34af4d888 | |
|
a4be25c24e | |
|
c4a95c8d4e | |
|
23c42ba33c | |
|
37cf083b7a | |
|
56ce26223f | |
|
c0c8500063 | |
|
c6b9b3646f | |
|
5847734ef7 | |
|
5619e9c2ca | |
|
70298cb96a | |
|
3c467c8c0e |
|
@ -44,7 +44,6 @@ export default defineConfig({
|
||||||
supportFile: 'test/cypress/support/index.js',
|
supportFile: 'test/cypress/support/index.js',
|
||||||
videosFolder: 'test/cypress/videos',
|
videosFolder: 'test/cypress/videos',
|
||||||
downloadsFolder: 'test/cypress/downloads',
|
downloadsFolder: 'test/cypress/downloads',
|
||||||
tmpUploadFolder: 'test/cypress/storage/tmp/dms',
|
|
||||||
video: false,
|
video: false,
|
||||||
specPattern: 'test/cypress/integration/**/*.spec.js',
|
specPattern: 'test/cypress/integration/**/*.spec.js',
|
||||||
experimentalRunAllSpecs: true,
|
experimentalRunAllSpecs: true,
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
{
|
{
|
||||||
"name": "salix-front",
|
"name": "salix-front",
|
||||||
"version": "25.18.0",
|
"version": "25.16.0",
|
||||||
"description": "Salix frontend",
|
"description": "Salix frontend",
|
||||||
"productName": "Salix",
|
"productName": "Salix",
|
||||||
"author": "Verdnatura",
|
"author": "Verdnatura",
|
||||||
|
|
|
@ -2,7 +2,6 @@
|
||||||
import { onMounted } from 'vue';
|
import { onMounted } from 'vue';
|
||||||
import { useQuasar, Dark } from 'quasar';
|
import { useQuasar, Dark } from 'quasar';
|
||||||
import { useI18n } from 'vue-i18n';
|
import { useI18n } from 'vue-i18n';
|
||||||
import VnScroll from './components/common/VnScroll.vue';
|
|
||||||
|
|
||||||
const quasar = useQuasar();
|
const quasar = useQuasar();
|
||||||
const { availableLocales, locale, fallbackLocale } = useI18n();
|
const { availableLocales, locale, fallbackLocale } = useI18n();
|
||||||
|
@ -39,7 +38,6 @@ quasar.iconMapFn = (iconName) => {
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<RouterView />
|
<RouterView />
|
||||||
<VnScroll/>
|
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style lang="scss">
|
<style lang="scss">
|
||||||
|
|
|
@ -67,7 +67,7 @@ describe('Axios boot', () => {
|
||||||
};
|
};
|
||||||
|
|
||||||
const result = onResponseError(error);
|
const result = onResponseError(error);
|
||||||
await expect(result).rejects.toEqual(expect.objectContaining(error));
|
expect(result).rejects.toEqual(expect.objectContaining(error));
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should call to the Notify plugin with a message from the response property', async () => {
|
it('should call to the Notify plugin with a message from the response property', async () => {
|
||||||
|
@ -83,7 +83,7 @@ describe('Axios boot', () => {
|
||||||
};
|
};
|
||||||
|
|
||||||
const result = onResponseError(error);
|
const result = onResponseError(error);
|
||||||
await expect(result).rejects.toEqual(expect.objectContaining(error));
|
expect(result).rejects.toEqual(expect.objectContaining(error));
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,6 +1,4 @@
|
||||||
import { boot } from 'quasar/wrappers';
|
import { boot } from 'quasar/wrappers';
|
||||||
import { date as quasarDate } from 'quasar';
|
|
||||||
const { formatDate } = quasarDate;
|
|
||||||
|
|
||||||
export default boot(() => {
|
export default boot(() => {
|
||||||
Date.vnUTC = () => {
|
Date.vnUTC = () => {
|
||||||
|
@ -27,34 +25,4 @@ export default boot(() => {
|
||||||
const date = new Date(Date.vnUTC());
|
const date = new Date(Date.vnUTC());
|
||||||
return new Date(date.getFullYear(), date.getMonth() + 1, 0);
|
return new Date(date.getFullYear(), date.getMonth() + 1, 0);
|
||||||
};
|
};
|
||||||
|
|
||||||
Date.getCurrentDateTimeFormatted = (
|
|
||||||
options = {
|
|
||||||
startOfDay: false,
|
|
||||||
endOfDay: true,
|
|
||||||
iso: true,
|
|
||||||
mask: 'DD-MM-YYYY HH:mm',
|
|
||||||
},
|
|
||||||
) => {
|
|
||||||
const date = Date.vnUTC();
|
|
||||||
if (options.startOfDay) {
|
|
||||||
date.setHours(0, 0, 0);
|
|
||||||
}
|
|
||||||
if (options.endOfDay) {
|
|
||||||
date.setHours(23, 59, 0);
|
|
||||||
}
|
|
||||||
if (options.iso) {
|
|
||||||
return date.toISOString();
|
|
||||||
}
|
|
||||||
return formatDate(date, options.mask);
|
|
||||||
};
|
|
||||||
|
|
||||||
Date.convertToISODateTime = (dateTimeStr) => {
|
|
||||||
const [datePart, timePart] = dateTimeStr.split(' ');
|
|
||||||
const [day, month, year] = datePart.split('-');
|
|
||||||
const [hours, minutes] = timePart.split(':');
|
|
||||||
|
|
||||||
const isoDate = new Date(year, month - 1, day, hours, minutes);
|
|
||||||
return isoDate.toISOString();
|
|
||||||
};
|
|
||||||
});
|
});
|
||||||
|
|
|
@ -20,6 +20,7 @@ const postcodeFormData = reactive({
|
||||||
provinceFk: null,
|
provinceFk: null,
|
||||||
townFk: null,
|
townFk: null,
|
||||||
});
|
});
|
||||||
|
const townFilter = ref({});
|
||||||
|
|
||||||
const countriesRef = ref(false);
|
const countriesRef = ref(false);
|
||||||
const provincesOptions = ref([]);
|
const provincesOptions = ref([]);
|
||||||
|
@ -32,11 +33,11 @@ function onDataSaved(formData) {
|
||||||
newPostcode.town = town.value.name;
|
newPostcode.town = town.value.name;
|
||||||
newPostcode.townFk = town.value.id;
|
newPostcode.townFk = town.value.id;
|
||||||
const provinceObject = provincesOptions.value.find(
|
const provinceObject = provincesOptions.value.find(
|
||||||
({ id }) => id === formData.provinceFk,
|
({ id }) => id === formData.provinceFk
|
||||||
);
|
);
|
||||||
newPostcode.province = provinceObject?.name;
|
newPostcode.province = provinceObject?.name;
|
||||||
const countryObject = countriesRef.value.opts.find(
|
const countryObject = countriesRef.value.opts.find(
|
||||||
({ id }) => id === formData.countryFk,
|
({ id }) => id === formData.countryFk
|
||||||
);
|
);
|
||||||
newPostcode.country = countryObject?.name;
|
newPostcode.country = countryObject?.name;
|
||||||
emit('onDataSaved', newPostcode);
|
emit('onDataSaved', newPostcode);
|
||||||
|
@ -66,11 +67,21 @@ function setTown(newTown, data) {
|
||||||
}
|
}
|
||||||
async function onCityCreated(newTown, formData) {
|
async function onCityCreated(newTown, formData) {
|
||||||
newTown.province = provincesOptions.value.find(
|
newTown.province = provincesOptions.value.find(
|
||||||
(province) => province.id === newTown.provinceFk,
|
(province) => province.id === newTown.provinceFk
|
||||||
);
|
);
|
||||||
formData.townFk = newTown;
|
formData.townFk = newTown;
|
||||||
setTown(newTown, formData);
|
setTown(newTown, formData);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function filterTowns(name) {
|
||||||
|
if (name !== '') {
|
||||||
|
townFilter.value.where = {
|
||||||
|
name: {
|
||||||
|
like: `%${name}%`,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
|
@ -96,6 +107,7 @@ async function onCityCreated(newTown, formData) {
|
||||||
<VnSelectDialog
|
<VnSelectDialog
|
||||||
:label="t('City')"
|
:label="t('City')"
|
||||||
@update:model-value="(value) => setTown(value, data)"
|
@update:model-value="(value) => setTown(value, data)"
|
||||||
|
@filter="filterTowns"
|
||||||
:tooltip="t('Create city')"
|
:tooltip="t('Create city')"
|
||||||
v-model="data.townFk"
|
v-model="data.townFk"
|
||||||
url="Towns/location"
|
url="Towns/location"
|
||||||
|
|
|
@ -65,7 +65,7 @@ const $props = defineProps({
|
||||||
default: null,
|
default: null,
|
||||||
},
|
},
|
||||||
beforeSaveFn: {
|
beforeSaveFn: {
|
||||||
type: [String, Function],
|
type: Function,
|
||||||
default: null,
|
default: null,
|
||||||
},
|
},
|
||||||
goTo: {
|
goTo: {
|
||||||
|
|
|
@ -140,7 +140,7 @@ const updatePhotoPreview = (value) => {
|
||||||
img.onerror = () => {
|
img.onerror = () => {
|
||||||
notify(
|
notify(
|
||||||
t("This photo provider doesn't allow remote downloads"),
|
t("This photo provider doesn't allow remote downloads"),
|
||||||
'negative',
|
'negative'
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -219,7 +219,11 @@ const makeRequest = async () => {
|
||||||
color="primary"
|
color="primary"
|
||||||
class="cursor-pointer"
|
class="cursor-pointer"
|
||||||
@click="rotateLeft()"
|
@click="rotateLeft()"
|
||||||
/>
|
>
|
||||||
|
<!-- <QTooltip class="no-pointer-events">
|
||||||
|
{{ t('Rotate left') }}
|
||||||
|
</QTooltip> -->
|
||||||
|
</QIcon>
|
||||||
<div>
|
<div>
|
||||||
<div ref="photoContainerRef" />
|
<div ref="photoContainerRef" />
|
||||||
</div>
|
</div>
|
||||||
|
@ -229,7 +233,11 @@ const makeRequest = async () => {
|
||||||
color="primary"
|
color="primary"
|
||||||
class="cursor-pointer"
|
class="cursor-pointer"
|
||||||
@click="rotateRight()"
|
@click="rotateRight()"
|
||||||
/>
|
>
|
||||||
|
<!-- <QTooltip class="no-pointer-events">
|
||||||
|
{{ t('Rotate right') }}
|
||||||
|
</QTooltip> -->
|
||||||
|
</QIcon>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="column">
|
<div class="column">
|
||||||
|
@ -257,6 +265,7 @@ const makeRequest = async () => {
|
||||||
class="cursor-pointer q-mr-sm"
|
class="cursor-pointer q-mr-sm"
|
||||||
@click="openInputFile()"
|
@click="openInputFile()"
|
||||||
>
|
>
|
||||||
|
<!-- <QTooltip>{{ t('globals.selectFile') }}</QTooltip> -->
|
||||||
</QIcon>
|
</QIcon>
|
||||||
<QIcon name="info" class="cursor-pointer">
|
<QIcon name="info" class="cursor-pointer">
|
||||||
<QTooltip>{{
|
<QTooltip>{{
|
||||||
|
|
|
@ -22,6 +22,7 @@ const { validate, validations } = useValidator();
|
||||||
const { notify } = useNotify();
|
const { notify } = useNotify();
|
||||||
const route = useRoute();
|
const route = useRoute();
|
||||||
const myForm = ref(null);
|
const myForm = ref(null);
|
||||||
|
const attrs = useAttrs();
|
||||||
const $props = defineProps({
|
const $props = defineProps({
|
||||||
url: {
|
url: {
|
||||||
type: String,
|
type: String,
|
||||||
|
@ -98,12 +99,8 @@ const $props = defineProps({
|
||||||
type: Function,
|
type: Function,
|
||||||
default: () => {},
|
default: () => {},
|
||||||
},
|
},
|
||||||
preventSubmit: {
|
|
||||||
type: Boolean,
|
|
||||||
default: true,
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
const emit = defineEmits(['onFetch', 'onDataSaved', 'submit']);
|
const emit = defineEmits(['onFetch', 'onDataSaved']);
|
||||||
const modelValue = computed(
|
const modelValue = computed(
|
||||||
() => $props.model ?? `formModel_${route?.meta?.title ?? route.name}`,
|
() => $props.model ?? `formModel_${route?.meta?.title ?? route.name}`,
|
||||||
).value;
|
).value;
|
||||||
|
@ -304,7 +301,7 @@ function onBeforeSave(formData, originalData) {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
async function onKeyup(evt) {
|
async function onKeyup(evt) {
|
||||||
if (evt.key === 'Enter' && !$props.preventSubmit) {
|
if (evt.key === 'Enter' && !('prevent-submit' in attrs)) {
|
||||||
const input = evt.target;
|
const input = evt.target;
|
||||||
if (input.type == 'textarea' && evt.shiftKey) {
|
if (input.type == 'textarea' && evt.shiftKey) {
|
||||||
let { selectionStart, selectionEnd } = input;
|
let { selectionStart, selectionEnd } = input;
|
||||||
|
@ -333,7 +330,6 @@ defineExpose({
|
||||||
<template>
|
<template>
|
||||||
<div class="column items-center full-width">
|
<div class="column items-center full-width">
|
||||||
<QForm
|
<QForm
|
||||||
v-on="$attrs"
|
|
||||||
ref="myForm"
|
ref="myForm"
|
||||||
v-if="formData"
|
v-if="formData"
|
||||||
@submit.prevent="save"
|
@submit.prevent="save"
|
||||||
|
@ -410,7 +406,6 @@ defineExpose({
|
||||||
</QBtnDropdown>
|
</QBtnDropdown>
|
||||||
<QBtn
|
<QBtn
|
||||||
v-else
|
v-else
|
||||||
data-cy="saveDefaultBtn"
|
|
||||||
:label="tMobile('globals.save')"
|
:label="tMobile('globals.save')"
|
||||||
color="primary"
|
color="primary"
|
||||||
icon="save"
|
icon="save"
|
||||||
|
|
|
@ -181,7 +181,7 @@ const searchModule = () => {
|
||||||
<template>
|
<template>
|
||||||
<QList padding class="column-max-width">
|
<QList padding class="column-max-width">
|
||||||
<template v-if="$props.source === 'main'">
|
<template v-if="$props.source === 'main'">
|
||||||
<template v-if="route?.matched[1]?.name === 'Dashboard'">
|
<template v-if="$route?.matched[1]?.name === 'Dashboard'">
|
||||||
<QItem class="q-pb-md">
|
<QItem class="q-pb-md">
|
||||||
<VnInput
|
<VnInput
|
||||||
v-model="search"
|
v-model="search"
|
||||||
|
@ -262,7 +262,7 @@ const searchModule = () => {
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<template v-for="item in items" :key="item.name">
|
<template v-for="item in items" :key="item.name">
|
||||||
<template v-if="item.name === route?.matched[1]?.name">
|
<template v-if="item.name === $route?.matched[1]?.name">
|
||||||
<QItem class="header">
|
<QItem class="header">
|
||||||
<QItemSection avatar v-if="item.icon">
|
<QItemSection avatar v-if="item.icon">
|
||||||
<QIcon :name="item.icon" />
|
<QIcon :name="item.icon" />
|
||||||
|
|
|
@ -69,7 +69,7 @@ const refresh = () => window.location.reload();
|
||||||
'no-visible': !stateQuery.isLoading().value,
|
'no-visible': !stateQuery.isLoading().value,
|
||||||
}"
|
}"
|
||||||
size="sm"
|
size="sm"
|
||||||
data-cy="navBar-spinner"
|
data-cy="loading-spinner"
|
||||||
/>
|
/>
|
||||||
<QSpace />
|
<QSpace />
|
||||||
<div id="searchbar" class="searchbar"></div>
|
<div id="searchbar" class="searchbar"></div>
|
||||||
|
|
|
@ -16,7 +16,7 @@ const $props = defineProps({
|
||||||
required: true,
|
required: true,
|
||||||
},
|
},
|
||||||
searchUrl: {
|
searchUrl: {
|
||||||
type: [String, Boolean],
|
type: String,
|
||||||
default: 'table',
|
default: 'table',
|
||||||
},
|
},
|
||||||
vertical: {
|
vertical: {
|
||||||
|
|
|
@ -33,7 +33,6 @@ import VnTableOrder from 'src/components/VnTable/VnOrder.vue';
|
||||||
import VnTableFilter from './VnTableFilter.vue';
|
import VnTableFilter from './VnTableFilter.vue';
|
||||||
import { getColAlign } from 'src/composables/getColAlign';
|
import { getColAlign } from 'src/composables/getColAlign';
|
||||||
import RightMenu from '../common/RightMenu.vue';
|
import RightMenu from '../common/RightMenu.vue';
|
||||||
import VnScroll from '../common/VnScroll.vue'
|
|
||||||
|
|
||||||
const arrayData = useArrayData(useAttrs()['data-key']);
|
const arrayData = useArrayData(useAttrs()['data-key']);
|
||||||
const $props = defineProps({
|
const $props = defineProps({
|
||||||
|
@ -66,7 +65,7 @@ const $props = defineProps({
|
||||||
default: null,
|
default: null,
|
||||||
},
|
},
|
||||||
create: {
|
create: {
|
||||||
type: [Boolean, Object],
|
type: Object,
|
||||||
default: null,
|
default: null,
|
||||||
},
|
},
|
||||||
createAsDialog: {
|
createAsDialog: {
|
||||||
|
@ -169,7 +168,6 @@ const params = ref(useFilterParams($attrs['data-key']).params);
|
||||||
const orders = ref(useFilterParams($attrs['data-key']).orders);
|
const orders = ref(useFilterParams($attrs['data-key']).orders);
|
||||||
const app = inject('app');
|
const app = inject('app');
|
||||||
const tableHeight = useTableHeight();
|
const tableHeight = useTableHeight();
|
||||||
const vnScrollRef = ref(null);
|
|
||||||
|
|
||||||
const editingRow = ref(null);
|
const editingRow = ref(null);
|
||||||
const editingField = ref(null);
|
const editingField = ref(null);
|
||||||
|
@ -191,17 +189,6 @@ const tableModes = [
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
const onVirtualScroll = ({ to }) => {
|
|
||||||
handleScroll();
|
|
||||||
const virtualScrollContainer = tableRef.value?.$el?.querySelector('.q-table__middle');
|
|
||||||
if (virtualScrollContainer) {
|
|
||||||
virtualScrollContainer.dispatchEvent(new CustomEvent('scroll'));
|
|
||||||
if (vnScrollRef.value) {
|
|
||||||
vnScrollRef.value.updateScrollContainer(virtualScrollContainer);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
onBeforeMount(() => {
|
onBeforeMount(() => {
|
||||||
const urlParams = route.query[$props.searchUrl];
|
const urlParams = route.query[$props.searchUrl];
|
||||||
hasParams.value = urlParams && Object.keys(urlParams).length !== 0;
|
hasParams.value = urlParams && Object.keys(urlParams).length !== 0;
|
||||||
|
@ -340,13 +327,16 @@ function handleOnDataSaved(_) {
|
||||||
if (_.onDataSaved) _.onDataSaved({ CrudModelRef: CrudModelRef.value });
|
if (_.onDataSaved) _.onDataSaved({ CrudModelRef: CrudModelRef.value });
|
||||||
else $props.create.onDataSaved(_);
|
else $props.create.onDataSaved(_);
|
||||||
}
|
}
|
||||||
|
|
||||||
function handleScroll() {
|
function handleScroll() {
|
||||||
if ($props.crudModel.disableInfiniteScroll) return;
|
if ($props.crudModel.disableInfiniteScroll) return;
|
||||||
|
|
||||||
const tMiddle = tableRef.value.$el.querySelector('.q-table__middle');
|
const tMiddle = tableRef.value.$el.querySelector('.q-table__middle');
|
||||||
const { scrollHeight, scrollTop, clientHeight } = tMiddle;
|
const { scrollHeight, scrollTop, clientHeight } = tMiddle;
|
||||||
const isAtBottom = Math.abs(scrollHeight - scrollTop - clientHeight) <= 40;
|
const isAtBottom = Math.abs(scrollHeight - scrollTop - clientHeight) <= 40;
|
||||||
if (isAtBottom) CrudModelRef.value.vnPaginateRef.paginate();
|
if (isAtBottom) CrudModelRef.value.vnPaginateRef.paginate();
|
||||||
}
|
}
|
||||||
|
|
||||||
function handleSelection({ evt, added, rows: selectedRows }, rows) {
|
function handleSelection({ evt, added, rows: selectedRows }, rows) {
|
||||||
if (evt?.shiftKey && added) {
|
if (evt?.shiftKey && added) {
|
||||||
const rowIndex = selectedRows[0].$index;
|
const rowIndex = selectedRows[0].$index;
|
||||||
|
@ -679,9 +669,9 @@ const rowCtrlClickFunction = computed(() => {
|
||||||
ref="tableRef"
|
ref="tableRef"
|
||||||
v-bind="table"
|
v-bind="table"
|
||||||
:class="[
|
:class="[
|
||||||
'vnTable',
|
'vnTable',
|
||||||
table ? 'selection-cell' : '',
|
table ? 'selection-cell' : '',
|
||||||
$props.footer ? 'last-row-sticky' : '',
|
$props.footer ? 'last-row-sticky' : '',
|
||||||
]"
|
]"
|
||||||
wrap-cells
|
wrap-cells
|
||||||
:columns="splittedColumns.columns"
|
:columns="splittedColumns.columns"
|
||||||
|
@ -693,7 +683,7 @@ const rowCtrlClickFunction = computed(() => {
|
||||||
flat
|
flat
|
||||||
:style="isTableMode && `max-height: ${$props.tableHeight || tableHeight}`"
|
:style="isTableMode && `max-height: ${$props.tableHeight || tableHeight}`"
|
||||||
:virtual-scroll="isTableMode"
|
:virtual-scroll="isTableMode"
|
||||||
@virtual-scroll="onVirtualScroll"
|
@virtual-scroll="handleScroll"
|
||||||
@row-click="(event, row) => handleRowClick(event, row)"
|
@row-click="(event, row) => handleRowClick(event, row)"
|
||||||
@update:selected="emit('update:selected', $event)"
|
@update:selected="emit('update:selected', $event)"
|
||||||
@selection="(details) => handleSelection(details, rows)"
|
@selection="(details) => handleSelection(details, rows)"
|
||||||
|
@ -751,7 +741,6 @@ const rowCtrlClickFunction = computed(() => {
|
||||||
withFilters
|
withFilters
|
||||||
"
|
"
|
||||||
:column="col"
|
:column="col"
|
||||||
:data-cy="`column-filter-${col.name}`"
|
|
||||||
:show-title="true"
|
:show-title="true"
|
||||||
:data-key="$attrs['data-key']"
|
:data-key="$attrs['data-key']"
|
||||||
v-model="params[columnName(col)]"
|
v-model="params[columnName(col)]"
|
||||||
|
@ -1098,11 +1087,6 @@ const rowCtrlClickFunction = computed(() => {
|
||||||
</template>
|
</template>
|
||||||
</FormModelPopup>
|
</FormModelPopup>
|
||||||
</QDialog>
|
</QDialog>
|
||||||
<VnScroll
|
|
||||||
ref="vnScrollRef"
|
|
||||||
v-if="isTableMode"
|
|
||||||
:scroll-target="tableRef?.$el?.querySelector('.q-table__middle')"
|
|
||||||
/>
|
|
||||||
</template>
|
</template>
|
||||||
<i18n>
|
<i18n>
|
||||||
en:
|
en:
|
||||||
|
|
|
@ -30,7 +30,6 @@ function columnName(col) {
|
||||||
v-bind="$attrs"
|
v-bind="$attrs"
|
||||||
:search-button="true"
|
:search-button="true"
|
||||||
:disable-submit-event="true"
|
:disable-submit-event="true"
|
||||||
:data-key="$attrs['data-key']"
|
|
||||||
:search-url
|
:search-url
|
||||||
>
|
>
|
||||||
<template #body="{ params, orders, searchFn }">
|
<template #body="{ params, orders, searchFn }">
|
||||||
|
|
|
@ -58,7 +58,7 @@ async function getConfig(url, filter) {
|
||||||
const response = await axios.get(url, {
|
const response = await axios.get(url, {
|
||||||
params: { filter: filter },
|
params: { filter: filter },
|
||||||
});
|
});
|
||||||
return response?.data && response?.data?.length > 0 ? response.data[0] : null;
|
return response.data && response.data.length > 0 ? response.data[0] : null;
|
||||||
}
|
}
|
||||||
|
|
||||||
async function fetchViewConfigData() {
|
async function fetchViewConfigData() {
|
||||||
|
|
|
@ -11,9 +11,6 @@ describe('VnTable', () => {
|
||||||
propsData: {
|
propsData: {
|
||||||
columns: [],
|
columns: [],
|
||||||
},
|
},
|
||||||
attrs: {
|
|
||||||
'data-key': 'test',
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
vm = wrapper.vm;
|
vm = wrapper.vm;
|
||||||
|
|
||||||
|
|
|
@ -11,7 +11,13 @@ describe('CrudModel', () => {
|
||||||
beforeAll(() => {
|
beforeAll(() => {
|
||||||
wrapper = createWrapper(CrudModel, {
|
wrapper = createWrapper(CrudModel, {
|
||||||
global: {
|
global: {
|
||||||
stubs: ['vnPaginate', 'vue-i18n'],
|
stubs: [
|
||||||
|
'vnPaginate',
|
||||||
|
'useState',
|
||||||
|
'arrayData',
|
||||||
|
'useStateStore',
|
||||||
|
'vue-i18n',
|
||||||
|
],
|
||||||
mocks: {
|
mocks: {
|
||||||
validate: vi.fn(),
|
validate: vi.fn(),
|
||||||
},
|
},
|
||||||
|
@ -23,7 +29,7 @@ describe('CrudModel', () => {
|
||||||
dataKey: 'crudModelKey',
|
dataKey: 'crudModelKey',
|
||||||
model: 'crudModel',
|
model: 'crudModel',
|
||||||
url: 'crudModelUrl',
|
url: 'crudModelUrl',
|
||||||
saveFn: vi.fn(),
|
saveFn: '',
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
wrapper = wrapper.wrapper;
|
wrapper = wrapper.wrapper;
|
||||||
|
@ -225,7 +231,7 @@ describe('CrudModel', () => {
|
||||||
expect(vm.isLoading).toBe(false);
|
expect(vm.isLoading).toBe(false);
|
||||||
expect(vm.hasChanges).toBe(false);
|
expect(vm.hasChanges).toBe(false);
|
||||||
|
|
||||||
await wrapper.setProps({ saveFn: null });
|
await wrapper.setProps({ saveFn: '' });
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should use default url if there's not saveFn", async () => {
|
it("should use default url if there's not saveFn", async () => {
|
||||||
|
|
|
@ -170,7 +170,7 @@ describe('LeftMenu as card', () => {
|
||||||
vm = mount('card').vm;
|
vm = mount('card').vm;
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should get routes for card source', () => {
|
it('should get routes for card source', async () => {
|
||||||
vm.getRoutes();
|
vm.getRoutes();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -251,6 +251,7 @@ describe('LeftMenu as main', () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should get routes for main source', () => {
|
it('should get routes for main source', () => {
|
||||||
|
vm.props.source = 'main';
|
||||||
vm.getRoutes();
|
vm.getRoutes();
|
||||||
expect(navigation.getModules).toHaveBeenCalled();
|
expect(navigation.getModules).toHaveBeenCalled();
|
||||||
});
|
});
|
||||||
|
|
|
@ -8,8 +8,7 @@ const model = defineModel({ prop: 'modelValue' });
|
||||||
<VnInput
|
<VnInput
|
||||||
v-model="model"
|
v-model="model"
|
||||||
ref="inputRef"
|
ref="inputRef"
|
||||||
@keydown.tab="$refs.inputRef.vnInputRef.blur()"
|
@keydown.tab="model = useAccountShortToStandard($event.target.value) ?? model"
|
||||||
@blur="model = useAccountShortToStandard(model) ?? model"
|
|
||||||
@input="model = $event.target.value.replace(/[^\d.]/g, '')"
|
@input="model = $event.target.value.replace(/[^\d.]/g, '')"
|
||||||
/>
|
/>
|
||||||
</template>
|
</template>
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
<script setup>
|
<script setup>
|
||||||
const model = defineModel({ type: [String, Number], default: '' });
|
const model = defineModel({ type: [String, Number], required: true });
|
||||||
</script>
|
</script>
|
||||||
<template>
|
<template>
|
||||||
<QDate v-model="model" :today-btn="true" :options="$attrs.options" />
|
<QDate v-model="model" :today-btn="true" :options="$attrs.options" />
|
||||||
|
|
|
@ -4,7 +4,6 @@ import { useRoute } from 'vue-router';
|
||||||
import { useI18n } from 'vue-i18n';
|
import { useI18n } from 'vue-i18n';
|
||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
|
|
||||||
import useNotify from 'src/composables/useNotify.js';
|
|
||||||
import FetchData from 'components/FetchData.vue';
|
import FetchData from 'components/FetchData.vue';
|
||||||
import VnRow from 'components/ui/VnRow.vue';
|
import VnRow from 'components/ui/VnRow.vue';
|
||||||
import VnSelect from 'src/components/common/VnSelect.vue';
|
import VnSelect from 'src/components/common/VnSelect.vue';
|
||||||
|
@ -13,7 +12,6 @@ import FormModelPopup from 'components/FormModelPopup.vue';
|
||||||
|
|
||||||
const route = useRoute();
|
const route = useRoute();
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
const { notify } = useNotify();
|
|
||||||
const emit = defineEmits(['onDataSaved']);
|
const emit = defineEmits(['onDataSaved']);
|
||||||
|
|
||||||
const $props = defineProps({
|
const $props = defineProps({
|
||||||
|
@ -63,11 +61,8 @@ function onFileChange(files) {
|
||||||
|
|
||||||
function mapperDms(data) {
|
function mapperDms(data) {
|
||||||
const formData = new FormData();
|
const formData = new FormData();
|
||||||
let files = data.files;
|
const { files } = data;
|
||||||
if (files) {
|
if (files) formData.append(files?.name, files);
|
||||||
files = Array.isArray(files) ? files : [files];
|
|
||||||
files.forEach((file) => formData.append(file?.name, file));
|
|
||||||
}
|
|
||||||
|
|
||||||
const dms = {
|
const dms = {
|
||||||
hasFile: !!data.hasFile,
|
hasFile: !!data.hasFile,
|
||||||
|
@ -88,16 +83,11 @@ function getUrl() {
|
||||||
}
|
}
|
||||||
|
|
||||||
async function save() {
|
async function save() {
|
||||||
try {
|
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);
|
delete dms.value.files;
|
||||||
notify(t('globals.dataSaved'), 'positive');
|
return response;
|
||||||
delete dms.value.files;
|
|
||||||
return response;
|
|
||||||
} catch (e) {
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function defaultData() {
|
function defaultData() {
|
||||||
|
|
|
@ -13,12 +13,11 @@ import VnDms from 'src/components/common/VnDms.vue';
|
||||||
import VnConfirm from 'components/ui/VnConfirm.vue';
|
import VnConfirm from 'components/ui/VnConfirm.vue';
|
||||||
import VnInputDate from 'components/common/VnInputDate.vue';
|
import VnInputDate from 'components/common/VnInputDate.vue';
|
||||||
import { useSession } from 'src/composables/useSession';
|
import { useSession } from 'src/composables/useSession';
|
||||||
import useNotify from 'src/composables/useNotify.js';
|
import { toDate } from 'src/filters/index';
|
||||||
|
|
||||||
const route = useRoute();
|
const route = useRoute();
|
||||||
const quasar = useQuasar();
|
const quasar = useQuasar();
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
const { notify } = useNotify();
|
|
||||||
const rows = ref([]);
|
const rows = ref([]);
|
||||||
const dmsRef = ref();
|
const dmsRef = ref();
|
||||||
const formDialog = ref({});
|
const formDialog = ref({});
|
||||||
|
@ -90,6 +89,7 @@ const dmsFilter = {
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
where: { [$props.filter]: route.params.id },
|
||||||
};
|
};
|
||||||
|
|
||||||
const columns = computed(() => [
|
const columns = computed(() => [
|
||||||
|
@ -119,7 +119,7 @@ const columns = computed(() => [
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
align: 'left',
|
align: 'left',
|
||||||
field: 'type',
|
field: 'dmsType',
|
||||||
label: t('globals.type'),
|
label: t('globals.type'),
|
||||||
name: 'type',
|
name: 'type',
|
||||||
component: QInput,
|
component: QInput,
|
||||||
|
@ -128,6 +128,7 @@ const columns = computed(() => [
|
||||||
borderless: true,
|
borderless: true,
|
||||||
'model-value': prop.row.dmsType?.name,
|
'model-value': prop.row.dmsType?.name,
|
||||||
}),
|
}),
|
||||||
|
format: (row) => row.name,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
align: 'left',
|
align: 'left',
|
||||||
|
@ -173,6 +174,7 @@ const columns = computed(() => [
|
||||||
name: prop.row.worker?.user?.name.toLowerCase(),
|
name: prop.row.worker?.user?.name.toLowerCase(),
|
||||||
workerId: prop.row.worker?.id,
|
workerId: prop.row.worker?.id,
|
||||||
}),
|
}),
|
||||||
|
format: (row) => row?.user?.name,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
align: 'left',
|
align: 'left',
|
||||||
|
@ -184,6 +186,7 @@ const columns = computed(() => [
|
||||||
disable: true,
|
disable: true,
|
||||||
'model-value': prop.row.created,
|
'model-value': prop.row.created,
|
||||||
}),
|
}),
|
||||||
|
format: (row) => toDate(row),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
field: 'options',
|
field: 'options',
|
||||||
|
@ -259,16 +262,9 @@ function deleteDms(dmsFk) {
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
.onOk(async () => {
|
.onOk(async () => {
|
||||||
try {
|
await axios.post(`${$props.deleteModel ?? $props.model}/${dmsFk}/removeFile`);
|
||||||
await axios.post(
|
const index = rows.value.findIndex((row) => row.id == dmsFk);
|
||||||
`${$props.deleteModel ?? $props.model}/${dmsFk}/removeFile`,
|
rows.value.splice(index, 1);
|
||||||
);
|
|
||||||
const index = rows.value.findIndex((row) => row.id == dmsFk);
|
|
||||||
rows.value.splice(index, 1);
|
|
||||||
notify(t('globals.dataDeleted'), 'positive');
|
|
||||||
} catch (e) {
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -308,7 +304,6 @@ defineExpose({
|
||||||
:user-filter="dmsFilter"
|
:user-filter="dmsFilter"
|
||||||
search-url="dmsFilter"
|
search-url="dmsFilter"
|
||||||
:order="['dmsFk DESC']"
|
:order="['dmsFk DESC']"
|
||||||
:filter="{ where: { [$props.filter]: route.params.id } }"
|
|
||||||
auto-load
|
auto-load
|
||||||
@on-fetch="setData"
|
@on-fetch="setData"
|
||||||
>
|
>
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
<script setup>
|
<script setup>
|
||||||
import { nextTick, watch, computed, ref, useAttrs } from 'vue';
|
import { onMounted, watch, computed, ref, useAttrs } from 'vue';
|
||||||
import { date, getCssVar } from 'quasar';
|
import { date } from 'quasar';
|
||||||
import VnDate from './VnDate.vue';
|
import VnDate from './VnDate.vue';
|
||||||
import { useRequired } from 'src/composables/useRequired';
|
import { useRequired } from 'src/composables/useRequired';
|
||||||
|
|
||||||
|
@ -20,18 +20,61 @@ const $props = defineProps({
|
||||||
});
|
});
|
||||||
|
|
||||||
const vnInputDateRef = ref(null);
|
const vnInputDateRef = ref(null);
|
||||||
const errColor = getCssVar('negative');
|
|
||||||
const textColor = ref('');
|
|
||||||
|
|
||||||
const dateFormat = 'DD/MM/YYYY';
|
const dateFormat = 'DD/MM/YYYY';
|
||||||
const isPopupOpen = ref();
|
const isPopupOpen = ref();
|
||||||
const hover = ref();
|
const hover = ref();
|
||||||
|
const mask = ref();
|
||||||
|
|
||||||
const mixinRules = [requiredFieldRule, ...($attrs.rules ?? [])];
|
const mixinRules = [requiredFieldRule, ...($attrs.rules ?? [])];
|
||||||
|
|
||||||
|
const formattedDate = computed({
|
||||||
|
get() {
|
||||||
|
if (!model.value) return model.value;
|
||||||
|
return date.formatDate(new Date(model.value), dateFormat);
|
||||||
|
},
|
||||||
|
set(value) {
|
||||||
|
if (value == model.value) return;
|
||||||
|
let newDate;
|
||||||
|
if (value) {
|
||||||
|
// parse input
|
||||||
|
if (value.includes('/') && value.length >= 10) {
|
||||||
|
if (value.at(2) == '/') value = value.split('/').reverse().join('/');
|
||||||
|
value = date.formatDate(
|
||||||
|
new Date(value).toISOString(),
|
||||||
|
'YYYY-MM-DDTHH:mm:ss.SSSZ',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
const [year, month, day] = value.split('-').map((e) => parseInt(e));
|
||||||
|
newDate = new Date(year, month - 1, day);
|
||||||
|
if (model.value) {
|
||||||
|
const orgDate =
|
||||||
|
model.value instanceof Date ? model.value : new Date(model.value);
|
||||||
|
|
||||||
|
newDate.setHours(
|
||||||
|
orgDate.getHours(),
|
||||||
|
orgDate.getMinutes(),
|
||||||
|
orgDate.getSeconds(),
|
||||||
|
orgDate.getMilliseconds(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!isNaN(newDate)) model.value = newDate.toISOString();
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
const popupDate = computed(() =>
|
const popupDate = computed(() =>
|
||||||
model.value ? date.formatDate(new Date(model.value), 'YYYY/MM/DD') : model.value,
|
model.value ? date.formatDate(new Date(model.value), 'YYYY/MM/DD') : model.value,
|
||||||
);
|
);
|
||||||
|
onMounted(() => {
|
||||||
|
// fix quasar bug
|
||||||
|
mask.value = '##/##/####';
|
||||||
|
});
|
||||||
|
watch(
|
||||||
|
() => model.value,
|
||||||
|
(val) => (formattedDate.value = val),
|
||||||
|
{ immediate: true },
|
||||||
|
);
|
||||||
|
|
||||||
const styleAttrs = computed(() => {
|
const styleAttrs = computed(() => {
|
||||||
return $props.isOutlined
|
return $props.isOutlined
|
||||||
|
@ -43,139 +86,28 @@ const styleAttrs = computed(() => {
|
||||||
: {};
|
: {};
|
||||||
});
|
});
|
||||||
|
|
||||||
const inputValue = ref('');
|
|
||||||
|
|
||||||
const validateAndCleanInput = (value) => {
|
|
||||||
inputValue.value = value.replace(/[^0-9./-]/g, '');
|
|
||||||
};
|
|
||||||
|
|
||||||
const manageDate = (date) => {
|
const manageDate = (date) => {
|
||||||
inputValue.value = date.split('/').reverse().join('/');
|
formattedDate.value = date;
|
||||||
isPopupOpen.value = false;
|
isPopupOpen.value = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
watch(
|
|
||||||
() => model.value,
|
|
||||||
(nVal) => {
|
|
||||||
if (nVal) inputValue.value = date.formatDate(new Date(model.value), dateFormat);
|
|
||||||
else inputValue.value = '';
|
|
||||||
},
|
|
||||||
{ immediate: true },
|
|
||||||
);
|
|
||||||
|
|
||||||
const formatDate = () => {
|
|
||||||
let value = inputValue.value;
|
|
||||||
if (!value || value === model.value) {
|
|
||||||
textColor.value = '';
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const regex =
|
|
||||||
/^([0]?[1-9]|[12][0-9]|3[01])([./-])([0]?[1-9]|1[0-2])([./-](\d{1,4}))?$/;
|
|
||||||
if (!regex.test(value)){
|
|
||||||
textColor.value = errColor;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
value = value.replace(/[.-]/g, '/');
|
|
||||||
const parts = value.split('/');
|
|
||||||
if (parts.length < 2) {
|
|
||||||
textColor.value = errColor;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
let [day, month, year] = parts;
|
|
||||||
if (day.length === 1) day = '0' + day;
|
|
||||||
if (month.length === 1) month = '0' + month;
|
|
||||||
|
|
||||||
const currentYear = Date.vnNew().getFullYear();
|
|
||||||
if (!year) year = currentYear;
|
|
||||||
const millennium = currentYear.toString().slice(0, 1);
|
|
||||||
|
|
||||||
switch (year.length) {
|
|
||||||
case 1:
|
|
||||||
year = `${millennium}00${year}`;
|
|
||||||
break;
|
|
||||||
case 2:
|
|
||||||
year = `${millennium}0${year}`;
|
|
||||||
break;
|
|
||||||
case 3:
|
|
||||||
year = `${millennium}${year}`;
|
|
||||||
break;
|
|
||||||
case 4:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
let isoCandidate = `${year}/${month}/${day}`;
|
|
||||||
isoCandidate = date.formatDate(
|
|
||||||
new Date(isoCandidate).toISOString(),
|
|
||||||
'YYYY-MM-DDTHH:mm:ss.SSSZ',
|
|
||||||
);
|
|
||||||
const [isoYear, isoMonth, isoDay] = isoCandidate.split('-').map((e) => parseInt(e));
|
|
||||||
const parsedDate = new Date(isoYear, isoMonth - 1, isoDay);
|
|
||||||
|
|
||||||
const isValidDate =
|
|
||||||
parsedDate instanceof Date &&
|
|
||||||
!isNaN(parsedDate) &&
|
|
||||||
parsedDate.getFullYear() === parseInt(year) &&
|
|
||||||
parsedDate.getMonth() === parseInt(month) - 1 &&
|
|
||||||
parsedDate.getDate() === parseInt(day);
|
|
||||||
|
|
||||||
if (!isValidDate) {
|
|
||||||
textColor.value = errColor;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (model.value) {
|
|
||||||
const original =
|
|
||||||
model.value instanceof Date ? model.value : new Date(model.value);
|
|
||||||
parsedDate.setHours(
|
|
||||||
original.getHours(),
|
|
||||||
original.getMinutes(),
|
|
||||||
original.getSeconds(),
|
|
||||||
original.getMilliseconds(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
model.value = parsedDate.toISOString();
|
|
||||||
|
|
||||||
textColor.value = '';
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleEnter = (event) => {
|
|
||||||
formatDate();
|
|
||||||
|
|
||||||
nextTick(() => {
|
|
||||||
const newEvent = new KeyboardEvent('keydown', {
|
|
||||||
key: 'Enter',
|
|
||||||
code: 'Enter',
|
|
||||||
bubbles: true,
|
|
||||||
cancelable: true,
|
|
||||||
});
|
|
||||||
vnInputDateRef.value?.$el?.dispatchEvent(newEvent);
|
|
||||||
});
|
|
||||||
};
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div @mouseover="hover = true" @mouseleave="hover = false">
|
<div @mouseover="hover = true" @mouseleave="hover = false">
|
||||||
{{ console.log($q) }}
|
|
||||||
<QInput
|
<QInput
|
||||||
ref="vnInputDateRef"
|
ref="vnInputDateRef"
|
||||||
v-model="inputValue"
|
v-model="formattedDate"
|
||||||
class="vn-input-date"
|
class="vn-input-date"
|
||||||
|
:mask="mask"
|
||||||
placeholder="dd/mm/aaaa"
|
placeholder="dd/mm/aaaa"
|
||||||
v-bind="{ ...$attrs, ...styleAttrs }"
|
v-bind="{ ...$attrs, ...styleAttrs }"
|
||||||
:class="{ required: isRequired }"
|
:class="{ required: isRequired }"
|
||||||
:rules="mixinRules"
|
:rules="mixinRules"
|
||||||
:clearable="false"
|
:clearable="false"
|
||||||
:input-style="{color: textColor}"
|
|
||||||
@click="isPopupOpen = !isPopupOpen"
|
@click="isPopupOpen = !isPopupOpen"
|
||||||
@keydown="isPopupOpen = false"
|
@keydown="isPopupOpen = false"
|
||||||
@blur="formatDate"
|
|
||||||
@keydown.enter.prevent="handleEnter"
|
|
||||||
hide-bottom-space
|
hide-bottom-space
|
||||||
:data-cy="($attrs['data-cy'] ?? $attrs.label) + '_inputDate'"
|
:data-cy="($attrs['data-cy'] ?? $attrs.label) + '_inputDate'"
|
||||||
@update:model-value="validateAndCleanInput"
|
|
||||||
>
|
>
|
||||||
<template #append>
|
<template #append>
|
||||||
<QIcon
|
<QIcon
|
||||||
|
@ -184,12 +116,11 @@ const handleEnter = (event) => {
|
||||||
v-if="
|
v-if="
|
||||||
($attrs.clearable == undefined || $attrs.clearable) &&
|
($attrs.clearable == undefined || $attrs.clearable) &&
|
||||||
hover &&
|
hover &&
|
||||||
inputValue &&
|
model &&
|
||||||
!$attrs.disable
|
!$attrs.disable
|
||||||
"
|
"
|
||||||
@click="
|
@click="
|
||||||
vnInputDateRef.focus();
|
vnInputDateRef.focus();
|
||||||
inputValue = null;
|
|
||||||
model = null;
|
model = null;
|
||||||
isPopupOpen = false;
|
isPopupOpen = false;
|
||||||
"
|
"
|
||||||
|
|
|
@ -1,79 +0,0 @@
|
||||||
<script setup>
|
|
||||||
import { computed, useAttrs } from 'vue';
|
|
||||||
import { date } from 'quasar';
|
|
||||||
import VnDate from './VnDate.vue';
|
|
||||||
import VnTime from './VnTime.vue';
|
|
||||||
|
|
||||||
const $attrs = useAttrs();
|
|
||||||
const model = defineModel({ type: [Date, String] });
|
|
||||||
|
|
||||||
const $props = defineProps({
|
|
||||||
isOutlined: {
|
|
||||||
type: Boolean,
|
|
||||||
default: false,
|
|
||||||
},
|
|
||||||
showEvent: {
|
|
||||||
type: Boolean,
|
|
||||||
default: true,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
const styleAttrs = computed(() => {
|
|
||||||
return $props.isOutlined
|
|
||||||
? {
|
|
||||||
dense: true,
|
|
||||||
outlined: true,
|
|
||||||
rounded: true,
|
|
||||||
}
|
|
||||||
: {};
|
|
||||||
});
|
|
||||||
const mask = 'DD-MM-YYYY HH:mm';
|
|
||||||
const selectedDate = computed({
|
|
||||||
get() {
|
|
||||||
if (!model.value) return JSON.stringify(new Date(model.value));
|
|
||||||
return date.formatDate(new Date(model.value), mask);
|
|
||||||
},
|
|
||||||
set(value) {
|
|
||||||
model.value = Date.convertToISODateTime(value);
|
|
||||||
},
|
|
||||||
});
|
|
||||||
const manageDate = (date) => {
|
|
||||||
selectedDate.value = date;
|
|
||||||
};
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<div @mouseover="hover = true" @mouseleave="hover = false">
|
|
||||||
<QInput
|
|
||||||
ref="vnInputDateRef"
|
|
||||||
v-model="selectedDate"
|
|
||||||
class="vn-input-date"
|
|
||||||
placeholder="dd/mm/aaaa HH:mm"
|
|
||||||
v-bind="{ ...$attrs, ...styleAttrs }"
|
|
||||||
:clearable="false"
|
|
||||||
@click="isPopupOpen = !isPopupOpen"
|
|
||||||
@keydown="isPopupOpen = false"
|
|
||||||
hide-bottom-space
|
|
||||||
@update:model-value="manageDate"
|
|
||||||
:data-cy="$attrs.dataCy ?? $attrs.label + '_inputDateTime'"
|
|
||||||
>
|
|
||||||
<template #prepend>
|
|
||||||
<QIcon name="today" size="xs">
|
|
||||||
<QPopupProxy cover transition-show="scale" transition-hide="scale">
|
|
||||||
<VnDate :mask="mask" v-model="selectedDate" />
|
|
||||||
</QPopupProxy>
|
|
||||||
</QIcon>
|
|
||||||
</template>
|
|
||||||
<template #append>
|
|
||||||
<QIcon name="access_time" size="xs">
|
|
||||||
<QPopupProxy cover transition-show="scale" transition-hide="scale">
|
|
||||||
<VnTime format24h :mask="mask" v-model="selectedDate" />
|
|
||||||
</QPopupProxy>
|
|
||||||
</QIcon>
|
|
||||||
</template>
|
|
||||||
</QInput>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
<i18n>
|
|
||||||
es:
|
|
||||||
Open date: Abrir fecha
|
|
||||||
</i18n>
|
|
|
@ -1,100 +0,0 @@
|
||||||
<script setup>
|
|
||||||
import { ref, onMounted, onUnmounted, watch, nextTick } from 'vue';
|
|
||||||
|
|
||||||
const props = defineProps({
|
|
||||||
scrollTarget: { type: [String, Object], default: 'window' }
|
|
||||||
});
|
|
||||||
|
|
||||||
const scrollPosition = ref(0);
|
|
||||||
const showButton = ref(false);
|
|
||||||
let scrollContainer = null;
|
|
||||||
|
|
||||||
const onScroll = () => {
|
|
||||||
if (!scrollContainer) return;
|
|
||||||
scrollPosition.value =
|
|
||||||
typeof props.scrollTarget === 'object'
|
|
||||||
? scrollContainer.scrollTop
|
|
||||||
: window.scrollY;
|
|
||||||
};
|
|
||||||
|
|
||||||
watch(scrollPosition, (newValue) => {
|
|
||||||
showButton.value = newValue > 0;
|
|
||||||
});
|
|
||||||
|
|
||||||
const scrollToTop = () => {
|
|
||||||
if (scrollContainer) {
|
|
||||||
scrollContainer.scrollTo({ top: 0, behavior: 'smooth' });
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const updateScrollContainer = (container) => {
|
|
||||||
if (container) {
|
|
||||||
if (scrollContainer) {
|
|
||||||
scrollContainer.removeEventListener('scroll', onScroll);
|
|
||||||
}
|
|
||||||
scrollContainer = container;
|
|
||||||
scrollContainer.addEventListener('scroll', onScroll);
|
|
||||||
onScroll();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
defineExpose({
|
|
||||||
updateScrollContainer
|
|
||||||
});
|
|
||||||
|
|
||||||
const initScrollContainer = async () => {
|
|
||||||
await nextTick();
|
|
||||||
|
|
||||||
if (typeof props.scrollTarget === 'object') {
|
|
||||||
scrollContainer = props.scrollTarget;
|
|
||||||
} else {
|
|
||||||
scrollContainer = window;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!scrollContainer) return
|
|
||||||
scrollContainer.addEventListener('scroll', onScroll);
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
onMounted(() => {
|
|
||||||
initScrollContainer();
|
|
||||||
});
|
|
||||||
|
|
||||||
onUnmounted(() => {
|
|
||||||
if (scrollContainer) {
|
|
||||||
scrollContainer.removeEventListener('scroll', onScroll);
|
|
||||||
scrollContainer = null;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<QIcon
|
|
||||||
v-if="showButton"
|
|
||||||
color="primary"
|
|
||||||
name="keyboard_arrow_up"
|
|
||||||
class="scroll-to-top"
|
|
||||||
@click="scrollToTop"
|
|
||||||
>
|
|
||||||
<QTooltip>{{ $t('globals.scrollToTop') }}</QTooltip>
|
|
||||||
</QIcon>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<style scoped>
|
|
||||||
.scroll-to-top {
|
|
||||||
position: fixed;
|
|
||||||
top: 70px;
|
|
||||||
font-size: 65px;
|
|
||||||
left: 50%;
|
|
||||||
transform: translateX(-50%);
|
|
||||||
z-index: 1000;
|
|
||||||
transition: transform 0.2s ease-in-out;
|
|
||||||
}
|
|
||||||
|
|
||||||
.scroll-to-top:hover {
|
|
||||||
transform: translateX(-50%) scale(1.2);
|
|
||||||
cursor: pointer;
|
|
||||||
filter: brightness(0.8);
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
|
|
|
@ -54,10 +54,6 @@ const $props = defineProps({
|
||||||
type: [Array],
|
type: [Array],
|
||||||
default: () => [],
|
default: () => [],
|
||||||
},
|
},
|
||||||
filterFn: {
|
|
||||||
type: Function,
|
|
||||||
default: null,
|
|
||||||
},
|
|
||||||
exprBuilder: {
|
exprBuilder: {
|
||||||
type: Function,
|
type: Function,
|
||||||
default: null,
|
default: null,
|
||||||
|
@ -66,12 +62,16 @@ const $props = defineProps({
|
||||||
type: Boolean,
|
type: Boolean,
|
||||||
default: true,
|
default: true,
|
||||||
},
|
},
|
||||||
|
defaultFilter: {
|
||||||
|
type: Boolean,
|
||||||
|
default: true,
|
||||||
|
},
|
||||||
fields: {
|
fields: {
|
||||||
type: Array,
|
type: Array,
|
||||||
default: null,
|
default: null,
|
||||||
},
|
},
|
||||||
include: {
|
include: {
|
||||||
type: [Object, Array, String],
|
type: [Object, Array],
|
||||||
default: null,
|
default: null,
|
||||||
},
|
},
|
||||||
where: {
|
where: {
|
||||||
|
@ -79,7 +79,7 @@ const $props = defineProps({
|
||||||
default: null,
|
default: null,
|
||||||
},
|
},
|
||||||
sortBy: {
|
sortBy: {
|
||||||
type: [String, Array],
|
type: String,
|
||||||
default: null,
|
default: null,
|
||||||
},
|
},
|
||||||
limit: {
|
limit: {
|
||||||
|
@ -152,22 +152,10 @@ const value = computed({
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const arrayDataKey =
|
|
||||||
$props.dataKey ??
|
|
||||||
($props.url?.length > 0 ? $props.url : ($attrs.name ?? $attrs.label));
|
|
||||||
|
|
||||||
const arrayData = useArrayData(arrayDataKey, {
|
|
||||||
url: $props.url,
|
|
||||||
searchUrl: false,
|
|
||||||
mapKey: $attrs['map-key'],
|
|
||||||
});
|
|
||||||
|
|
||||||
const computedSortBy = computed(() => {
|
const computedSortBy = computed(() => {
|
||||||
return $props.sortBy || $props.optionLabel + ' ASC';
|
return $props.sortBy || $props.optionLabel + ' ASC';
|
||||||
});
|
});
|
||||||
|
|
||||||
const getVal = (val) => ($props.useLike ? { like: `%${val}%` } : val);
|
|
||||||
|
|
||||||
watch(options, (newValue) => {
|
watch(options, (newValue) => {
|
||||||
setOptions(newValue);
|
setOptions(newValue);
|
||||||
});
|
});
|
||||||
|
@ -186,6 +174,16 @@ onMounted(() => {
|
||||||
if ($props.focusOnMount) setTimeout(() => vnSelectRef.value.showPopup(), 300);
|
if ($props.focusOnMount) setTimeout(() => vnSelectRef.value.showPopup(), 300);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const arrayDataKey =
|
||||||
|
$props.dataKey ??
|
||||||
|
($props.url?.length > 0 ? $props.url : ($attrs.name ?? $attrs.label));
|
||||||
|
|
||||||
|
const arrayData = useArrayData(arrayDataKey, {
|
||||||
|
url: $props.url,
|
||||||
|
searchUrl: false,
|
||||||
|
mapKey: $attrs['map-key'],
|
||||||
|
});
|
||||||
|
|
||||||
function findKeyInOptions() {
|
function findKeyInOptions() {
|
||||||
if (!$props.options) return;
|
if (!$props.options) return;
|
||||||
return filter($props.modelValue, $props.options)?.length;
|
return filter($props.modelValue, $props.options)?.length;
|
||||||
|
@ -254,41 +252,43 @@ async function fetchFilter(val) {
|
||||||
}
|
}
|
||||||
|
|
||||||
async function filterHandler(val, update) {
|
async function filterHandler(val, update) {
|
||||||
|
if (isLoading.value) return update();
|
||||||
|
if (!val && lastVal.value === val) {
|
||||||
|
lastVal.value = val;
|
||||||
|
return update();
|
||||||
|
}
|
||||||
|
lastVal.value = val;
|
||||||
let newOptions;
|
let newOptions;
|
||||||
|
|
||||||
if ($props.filterFn) update($props.filterFn(val));
|
if (!$props.defaultFilter) return update();
|
||||||
else if (!val && lastVal.value === val) update();
|
if (
|
||||||
else {
|
$props.url &&
|
||||||
const makeRequest =
|
($props.limit || (!$props.limit && Object.keys(myOptions.value).length === 0))
|
||||||
($props.url && $props.limit) ||
|
) {
|
||||||
(!$props.limit && Object.keys(myOptions.value).length === 0);
|
newOptions = await fetchFilter(val);
|
||||||
newOptions = makeRequest
|
} else newOptions = filter(val, myOptionsOriginal.value);
|
||||||
? await fetchFilter(val)
|
update(
|
||||||
: filter(val, myOptionsOriginal.value);
|
() => {
|
||||||
|
if ($props.noOne && noOneText.toLowerCase().includes(val.toLowerCase()))
|
||||||
|
newOptions.unshift(noOneOpt.value);
|
||||||
|
|
||||||
update(
|
myOptions.value = newOptions;
|
||||||
() => {
|
},
|
||||||
if ($props.noOne && noOneText.toLowerCase().includes(val.toLowerCase()))
|
(ref) => {
|
||||||
newOptions.unshift(noOneOpt.value);
|
if (val !== '' && ref.options.length > 0) {
|
||||||
|
ref.setOptionIndex(-1);
|
||||||
myOptions.value = newOptions;
|
ref.moveOptionSelection(1, true);
|
||||||
},
|
}
|
||||||
(ref) => {
|
},
|
||||||
if (val !== '' && ref.options.length > 0) {
|
);
|
||||||
ref.setOptionIndex(-1);
|
|
||||||
ref.moveOptionSelection(1, true);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
lastVal.value = val;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function nullishToTrue(value) {
|
function nullishToTrue(value) {
|
||||||
return value ?? true;
|
return value ?? true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const getVal = (val) => ($props.useLike ? { like: `%${val}%` } : val);
|
||||||
|
|
||||||
async function onScroll({ to, direction, from, index }) {
|
async function onScroll({ to, direction, from, index }) {
|
||||||
const lastIndex = myOptions.value.length - 1;
|
const lastIndex = myOptions.value.length - 1;
|
||||||
|
|
||||||
|
|
|
@ -232,7 +232,7 @@ fr:
|
||||||
pt: Portugais
|
pt: Portugais
|
||||||
pt:
|
pt:
|
||||||
Send SMS: Enviar SMS
|
Send SMS: Enviar SMS
|
||||||
CustomerDefaultLanguage: Este cliente utiliza o {locale} como seu idioma padrão
|
CustomerDefaultLanguage: Este cliente utiliza o <strong>{locale}</strong> como seu idioma padrão
|
||||||
Language: Linguagem
|
Language: Linguagem
|
||||||
Phone: Móvel
|
Phone: Móvel
|
||||||
Subject: Assunto
|
Subject: Assunto
|
||||||
|
|
|
@ -12,9 +12,7 @@ describe('VnDiscount', () => {
|
||||||
price: 100,
|
price: 100,
|
||||||
quantity: 2,
|
quantity: 2,
|
||||||
discount: 10,
|
discount: 10,
|
||||||
mana: 10,
|
}
|
||||||
promise: vi.fn(),
|
|
||||||
},
|
|
||||||
}).vm;
|
}).vm;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -41,12 +41,10 @@ describe('VnDms', () => {
|
||||||
companyFk: 2,
|
companyFk: 2,
|
||||||
dmsTypeFk: 3,
|
dmsTypeFk: 3,
|
||||||
description: 'This is a test description',
|
description: 'This is a test description',
|
||||||
files: [
|
files: {
|
||||||
{
|
name: 'example.txt',
|
||||||
name: 'example.txt',
|
content: new Blob(['file content'], { type: 'text/plain' }),
|
||||||
content: new Blob(['file content'], { type: 'text/plain' }),
|
},
|
||||||
},
|
|
||||||
],
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const expectedBody = {
|
const expectedBody = {
|
||||||
|
@ -85,7 +83,7 @@ describe('VnDms', () => {
|
||||||
it('should map DMS data correctly and add file to FormData', () => {
|
it('should map DMS data correctly and add file to FormData', () => {
|
||||||
const [formData, params] = vm.mapperDms(data);
|
const [formData, params] = vm.mapperDms(data);
|
||||||
|
|
||||||
expect([formData.get('example.txt')]).toStrictEqual(data.files);
|
expect(formData.get('example.txt')).toBe(data.files);
|
||||||
expect(expectedBody).toEqual(params.params);
|
expect(expectedBody).toEqual(params.params);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -2,6 +2,7 @@ import { createWrapper } from 'app/test/vitest/helper';
|
||||||
import { vi, describe, expect, it } from 'vitest';
|
import { vi, describe, expect, it } from 'vitest';
|
||||||
import VnInput from 'src/components/common/VnInput.vue';
|
import VnInput from 'src/components/common/VnInput.vue';
|
||||||
|
|
||||||
|
|
||||||
describe('VnInput', () => {
|
describe('VnInput', () => {
|
||||||
let vm;
|
let vm;
|
||||||
let wrapper;
|
let wrapper;
|
||||||
|
@ -11,27 +12,25 @@ describe('VnInput', () => {
|
||||||
wrapper = createWrapper(VnInput, {
|
wrapper = createWrapper(VnInput, {
|
||||||
props: {
|
props: {
|
||||||
modelValue: value,
|
modelValue: value,
|
||||||
isOutlined,
|
isOutlined, emptyToNull, insertable,
|
||||||
emptyToNull,
|
maxlength: 101
|
||||||
insertable,
|
|
||||||
maxlength: 101,
|
|
||||||
},
|
},
|
||||||
attrs: {
|
attrs: {
|
||||||
label: 'test',
|
label: 'test',
|
||||||
required: true,
|
required: true,
|
||||||
maxlength: 101,
|
maxlength: 101,
|
||||||
maxLength: 10,
|
maxLength: 10,
|
||||||
'max-length': 20,
|
'max-length':20
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
wrapper = wrapper.wrapper;
|
wrapper = wrapper.wrapper;
|
||||||
vm = wrapper.vm;
|
vm = wrapper.vm;
|
||||||
input = wrapper.find('[data-cy="test_input"]');
|
input = wrapper.find('[data-cy="test_input"]');
|
||||||
}
|
};
|
||||||
|
|
||||||
describe('value', () => {
|
describe('value', () => {
|
||||||
it('should emit update:modelValue when value changes', async () => {
|
it('should emit update:modelValue when value changes', async () => {
|
||||||
generateWrapper('12345', false, false, true);
|
generateWrapper('12345', false, false, true)
|
||||||
await input.setValue('123');
|
await input.setValue('123');
|
||||||
expect(wrapper.emitted('update:modelValue')).toBeTruthy();
|
expect(wrapper.emitted('update:modelValue')).toBeTruthy();
|
||||||
expect(wrapper.emitted('update:modelValue')[0]).toEqual(['123']);
|
expect(wrapper.emitted('update:modelValue')[0]).toEqual(['123']);
|
||||||
|
@ -63,6 +62,7 @@ describe('VnInput', () => {
|
||||||
expect(wrapper.emitted('update:modelValue')).toBeUndefined();
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined();
|
||||||
const spyhandler = vi.spyOn(vm, 'handleInsertMode');
|
const spyhandler = vi.spyOn(vm, 'handleInsertMode');
|
||||||
expect(spyhandler).not.toHaveBeenCalled();
|
expect(spyhandler).not.toHaveBeenCalled();
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -71,12 +71,12 @@ describe('VnInput', () => {
|
||||||
it.skip('handleKeydown respects insertable behavior', async () => {
|
it.skip('handleKeydown respects insertable behavior', async () => {
|
||||||
const expectedValue = '12345';
|
const expectedValue = '12345';
|
||||||
generateWrapper('1234', false, false, true);
|
generateWrapper('1234', false, false, true);
|
||||||
vm.focus();
|
vm.focus()
|
||||||
await input.trigger('keydown', { key: '5' });
|
await input.trigger('keydown', { key: '5' });
|
||||||
await vm.$nextTick();
|
await vm.$nextTick();
|
||||||
expect(wrapper.emitted('update:modelValue')).toBeTruthy();
|
expect(wrapper.emitted('update:modelValue')).toBeTruthy();
|
||||||
expect(wrapper.emitted('update:modelValue')[0]).toEqual([expectedValue]);
|
expect(wrapper.emitted('update:modelValue')[0]).toEqual([expectedValue ]);
|
||||||
expect(vm.value).toBe(expectedValue);
|
expect(vm.value).toBe( expectedValue);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -5,71 +5,52 @@ import VnInputDate from 'components/common/VnInputDate.vue';
|
||||||
let vm;
|
let vm;
|
||||||
let wrapper;
|
let wrapper;
|
||||||
|
|
||||||
function generateWrapper(outlined = false, required = false) {
|
function generateWrapper(date, outlined, required) {
|
||||||
wrapper = createWrapper(VnInputDate, {
|
wrapper = createWrapper(VnInputDate, {
|
||||||
props: {
|
props: {
|
||||||
modelValue: '2000-12-31T23:00:00.000Z',
|
modelValue: date,
|
||||||
'onUpdate:modelValue': (e) => wrapper.setProps({ modelValue: e }),
|
|
||||||
},
|
},
|
||||||
attrs: {
|
attrs: {
|
||||||
isOutlined: outlined,
|
isOutlined: outlined,
|
||||||
required: required,
|
required: required
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
wrapper = wrapper.wrapper;
|
wrapper = wrapper.wrapper;
|
||||||
vm = wrapper.vm;
|
vm = wrapper.vm;
|
||||||
}
|
};
|
||||||
|
|
||||||
describe('VnInputDate', () => {
|
describe('VnInputDate', () => {
|
||||||
|
|
||||||
describe('formattedDate', () => {
|
describe('formattedDate', () => {
|
||||||
it('validateAndCleanInput should remove non-numeric characters', async () => {
|
it('formats a valid date correctly', async () => {
|
||||||
generateWrapper();
|
generateWrapper('2023-12-25', false, false);
|
||||||
vm.validateAndCleanInput('10a/1s2/2dd0a23');
|
|
||||||
await vm.$nextTick();
|
await vm.$nextTick();
|
||||||
expect(vm.inputValue).toBe('10/12/2023');
|
expect(vm.formattedDate).toBe('25/12/2023');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('manageDate should reverse the date', async () => {
|
it('updates the model value when a new date is set', async () => {
|
||||||
generateWrapper();
|
|
||||||
vm.manageDate('10/12/2023');
|
|
||||||
await vm.$nextTick();
|
|
||||||
expect(vm.inputValue).toBe('2023/12/10');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('formatDate should format the date correctly when a valid date is entered with full year', async () => {
|
|
||||||
const input = wrapper.find('input');
|
const input = wrapper.find('input');
|
||||||
await input.setValue('25.12/2002');
|
await input.setValue('31/12/2023');
|
||||||
await vm.$nextTick();
|
expect(wrapper.emitted()['update:modelValue']).toBeTruthy();
|
||||||
await vm.formatDate();
|
expect(wrapper.emitted()['update:modelValue'][0][0]).toBe('2023-12-31T00:00:00.000Z');
|
||||||
expect(vm.model).toBe('2002-12-24T23:00:00.000Z');
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should format the date correctly when a valid date is entered with short year', async () => {
|
it('should not update the model value when an invalid date is set', async () => {
|
||||||
const input = wrapper.find('input');
|
const input = wrapper.find('input');
|
||||||
await input.setValue('31.12-23');
|
await input.setValue('invalid-date');
|
||||||
await vm.$nextTick();
|
expect(wrapper.emitted()['update:modelValue'][0][0]).toBe('2023-12-31T00:00:00.000Z');
|
||||||
await vm.formatDate();
|
|
||||||
expect(vm.model).toBe('2023-12-30T23:00:00.000Z');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should format the date correctly when a valid date is entered without year', async () => {
|
|
||||||
const input = wrapper.find('input');
|
|
||||||
await input.setValue('12.03');
|
|
||||||
await vm.$nextTick();
|
|
||||||
await vm.formatDate();
|
|
||||||
expect(vm.model).toBe('2001-03-11T23:00:00.000Z');
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('styleAttrs', () => {
|
describe('styleAttrs', () => {
|
||||||
it('should return empty styleAttrs when isOutlined is false', async () => {
|
it('should return empty styleAttrs when isOutlined is false', async () => {
|
||||||
generateWrapper();
|
generateWrapper('2023-12-25', false, false);
|
||||||
await vm.$nextTick();
|
await vm.$nextTick();
|
||||||
expect(vm.styleAttrs).toEqual({});
|
expect(vm.styleAttrs).toEqual({});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should set styleAttrs when isOutlined is true', async () => {
|
it('should set styleAttrs when isOutlined is true', async () => {
|
||||||
generateWrapper(true, false);
|
generateWrapper('2023-12-25', true, false);
|
||||||
await vm.$nextTick();
|
await vm.$nextTick();
|
||||||
expect(vm.styleAttrs.outlined).toBe(true);
|
expect(vm.styleAttrs.outlined).toBe(true);
|
||||||
});
|
});
|
||||||
|
@ -77,13 +58,13 @@ describe('VnInputDate', () => {
|
||||||
|
|
||||||
describe('required', () => {
|
describe('required', () => {
|
||||||
it('should not applies required class when isRequired is false', async () => {
|
it('should not applies required class when isRequired is false', async () => {
|
||||||
generateWrapper();
|
generateWrapper('2023-12-25', false, false);
|
||||||
await vm.$nextTick();
|
await vm.$nextTick();
|
||||||
expect(wrapper.find('.vn-input-date').classes()).not.toContain('required');
|
expect(wrapper.find('.vn-input-date').classes()).not.toContain('required');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should applies required class when isRequired is true', async () => {
|
it('should applies required class when isRequired is true', async () => {
|
||||||
generateWrapper(false, true);
|
generateWrapper('2023-12-25', false, true);
|
||||||
await vm.$nextTick();
|
await vm.$nextTick();
|
||||||
expect(wrapper.find('.vn-input-date').classes()).toContain('required');
|
expect(wrapper.find('.vn-input-date').classes()).toContain('required');
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,81 +0,0 @@
|
||||||
import { createWrapper } from 'app/test/vitest/helper.js';
|
|
||||||
import { describe, it, expect, beforeAll } from 'vitest';
|
|
||||||
import VnInputDateTime from 'components/common/VnInputDateTime.vue';
|
|
||||||
import vnDateBoot from 'src/boot/vnDate';
|
|
||||||
|
|
||||||
let vm;
|
|
||||||
let wrapper;
|
|
||||||
|
|
||||||
beforeAll(() => {
|
|
||||||
// Initialize the vnDate boot
|
|
||||||
vnDateBoot();
|
|
||||||
});
|
|
||||||
function generateWrapper(date, outlined, showEvent) {
|
|
||||||
wrapper = createWrapper(VnInputDateTime, {
|
|
||||||
props: {
|
|
||||||
modelValue: date,
|
|
||||||
isOutlined: outlined,
|
|
||||||
showEvent: showEvent,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
wrapper = wrapper.wrapper;
|
|
||||||
vm = wrapper.vm;
|
|
||||||
}
|
|
||||||
|
|
||||||
describe('VnInputDateTime', () => {
|
|
||||||
describe('selectedDate', () => {
|
|
||||||
it('formats a valid datetime correctly', async () => {
|
|
||||||
generateWrapper('2023-12-25T10:30:00', false, true);
|
|
||||||
await vm.$nextTick();
|
|
||||||
expect(vm.selectedDate).toBe('25-12-2023 10:30');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('handles null date value', async () => {
|
|
||||||
generateWrapper(null, false, true);
|
|
||||||
await vm.$nextTick();
|
|
||||||
expect(vm.selectedDate).not.toBe(null);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('updates the model value when a new datetime is set', async () => {
|
|
||||||
vm.selectedDate = '31-12-2023 15:45';
|
|
||||||
await vm.$nextTick();
|
|
||||||
expect(wrapper.emitted()['update:modelValue']).toBeTruthy();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('styleAttrs', () => {
|
|
||||||
it('should return empty styleAttrs when isOutlined is false', async () => {
|
|
||||||
generateWrapper('2023-12-25T10:30:00', false, true);
|
|
||||||
await vm.$nextTick();
|
|
||||||
expect(vm.styleAttrs).toEqual({});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should set styleAttrs when isOutlined is true', async () => {
|
|
||||||
generateWrapper('2023-12-25T10:30:00', true, true);
|
|
||||||
await vm.$nextTick();
|
|
||||||
expect(vm.styleAttrs).toEqual({
|
|
||||||
dense: true,
|
|
||||||
outlined: true,
|
|
||||||
rounded: true,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('component rendering', () => {
|
|
||||||
it('should render date and time icons', async () => {
|
|
||||||
generateWrapper('2023-12-25T10:30:00', false, true);
|
|
||||||
await vm.$nextTick();
|
|
||||||
const icons = wrapper.findAllComponents({ name: 'QIcon' });
|
|
||||||
expect(icons.length).toBe(2);
|
|
||||||
expect(icons[0].props('name')).toBe('today');
|
|
||||||
expect(icons[1].props('name')).toBe('access_time');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should render popup proxies for date and time', async () => {
|
|
||||||
generateWrapper('2023-12-25T10:30:00', false, true);
|
|
||||||
await vm.$nextTick();
|
|
||||||
const popups = wrapper.findAllComponents({ name: 'QPopupProxy' });
|
|
||||||
expect(popups.length).toBe(2);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
|
@ -90,10 +90,8 @@ describe('VnLog', () => {
|
||||||
|
|
||||||
vm = createWrapper(VnLog, {
|
vm = createWrapper(VnLog, {
|
||||||
global: {
|
global: {
|
||||||
stubs: ['FetchData', 'vue-i18n'],
|
stubs: [],
|
||||||
mocks: {
|
mocks: {},
|
||||||
fetch: vi.fn(),
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
propsData: {
|
propsData: {
|
||||||
model: 'Claim',
|
model: 'Claim',
|
||||||
|
|
|
@ -26,7 +26,7 @@ describe('VnNotes', () => {
|
||||||
) {
|
) {
|
||||||
vi.spyOn(axios, 'get').mockResolvedValue({ data: [] });
|
vi.spyOn(axios, 'get').mockResolvedValue({ data: [] });
|
||||||
wrapper = createWrapper(VnNotes, {
|
wrapper = createWrapper(VnNotes, {
|
||||||
propsData: { ...defaultOptions, ...options },
|
propsData: options,
|
||||||
});
|
});
|
||||||
wrapper = wrapper.wrapper;
|
wrapper = wrapper.wrapper;
|
||||||
vm = wrapper.vm;
|
vm = wrapper.vm;
|
||||||
|
|
|
@ -44,8 +44,7 @@ onBeforeMount(async () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
// It enables to load data only once if the module is the same as the dataKey
|
// It enables to load data only once if the module is the same as the dataKey
|
||||||
if (!isSameDataKey.value || !route.params.id || $props.id !== route.params.id)
|
if (!isSameDataKey.value || !route.params.id) await getData();
|
||||||
await getData();
|
|
||||||
watch(
|
watch(
|
||||||
() => [$props.url, $props.filter],
|
() => [$props.url, $props.filter],
|
||||||
async () => {
|
async () => {
|
||||||
|
@ -59,8 +58,7 @@ async function getData() {
|
||||||
store.filter = $props.filter ?? {};
|
store.filter = $props.filter ?? {};
|
||||||
isLoading.value = true;
|
isLoading.value = true;
|
||||||
try {
|
try {
|
||||||
await arrayData.fetch({ append: false, updateRouter: false });
|
const { data } = await arrayData.fetch({ append: false, updateRouter: false });
|
||||||
const { data } = store;
|
|
||||||
state.set($props.dataKey, data);
|
state.set($props.dataKey, data);
|
||||||
emit('onFetch', data);
|
emit('onFetch', data);
|
||||||
} finally {
|
} finally {
|
||||||
|
|
|
@ -212,7 +212,6 @@ const getLocale = (label) => {
|
||||||
color="primary"
|
color="primary"
|
||||||
style="position: fixed; z-index: 1; right: 0; bottom: 0"
|
style="position: fixed; z-index: 1; right: 0; bottom: 0"
|
||||||
icon="search"
|
icon="search"
|
||||||
data-cy="vnFilterPanel_search"
|
|
||||||
@click="search()"
|
@click="search()"
|
||||||
>
|
>
|
||||||
<QTooltip bottom anchor="bottom right">
|
<QTooltip bottom anchor="bottom right">
|
||||||
|
@ -230,7 +229,6 @@ const getLocale = (label) => {
|
||||||
<QItemSection top side>
|
<QItemSection top side>
|
||||||
<QBtn
|
<QBtn
|
||||||
@click="clearFilters"
|
@click="clearFilters"
|
||||||
data-cy="clearFilters"
|
|
||||||
color="primary"
|
color="primary"
|
||||||
dense
|
dense
|
||||||
flat
|
flat
|
||||||
|
@ -294,7 +292,6 @@ const getLocale = (label) => {
|
||||||
</QList>
|
</QList>
|
||||||
</QForm>
|
</QForm>
|
||||||
<QInnerLoading
|
<QInnerLoading
|
||||||
data-cy="filterPanel-spinner"
|
|
||||||
:label="t('globals.pleaseWait')"
|
:label="t('globals.pleaseWait')"
|
||||||
:showing="isLoading"
|
:showing="isLoading"
|
||||||
color="primary"
|
color="primary"
|
||||||
|
|
|
@ -2,9 +2,7 @@
|
||||||
import { onBeforeUnmount, onMounted, ref, watch } from 'vue';
|
import { onBeforeUnmount, onMounted, ref, watch } from 'vue';
|
||||||
import { useI18n } from 'vue-i18n';
|
import { useI18n } from 'vue-i18n';
|
||||||
import { useArrayData } from 'composables/useArrayData';
|
import { useArrayData } from 'composables/useArrayData';
|
||||||
import { useAttrs } from 'vue';
|
|
||||||
|
|
||||||
const attrs = useAttrs();
|
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
|
@ -69,7 +67,7 @@ const props = defineProps({
|
||||||
default: null,
|
default: null,
|
||||||
},
|
},
|
||||||
searchUrl: {
|
searchUrl: {
|
||||||
type: [String, Boolean],
|
type: String,
|
||||||
default: null,
|
default: null,
|
||||||
},
|
},
|
||||||
disableInfiniteScroll: {
|
disableInfiniteScroll: {
|
||||||
|
@ -77,7 +75,7 @@ const props = defineProps({
|
||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
mapKey: {
|
mapKey: {
|
||||||
type: [String, Boolean],
|
type: String,
|
||||||
default: '',
|
default: '',
|
||||||
},
|
},
|
||||||
keyData: {
|
keyData: {
|
||||||
|
@ -222,7 +220,7 @@ defineExpose({
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div class="full-width" v-bind="attrs">
|
<div class="full-width">
|
||||||
<div
|
<div
|
||||||
v-if="!store.data && !store.data?.length && !isLoading"
|
v-if="!store.data && !store.data?.length && !isLoading"
|
||||||
class="info-row q-pa-md text-center"
|
class="info-row q-pa-md text-center"
|
||||||
|
|
|
@ -46,7 +46,7 @@ const props = defineProps({
|
||||||
default: null,
|
default: null,
|
||||||
},
|
},
|
||||||
order: {
|
order: {
|
||||||
type: [String, Array],
|
type: String,
|
||||||
default: '',
|
default: '',
|
||||||
},
|
},
|
||||||
limit: {
|
limit: {
|
||||||
|
|
|
@ -23,15 +23,10 @@ describe('CardSummary', () => {
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
wrapper = createWrapper(CardSummary, {
|
wrapper = createWrapper(CardSummary, {
|
||||||
global: {
|
|
||||||
mocks: {
|
|
||||||
validate: vi.fn(),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
propsData: {
|
propsData: {
|
||||||
dataKey: 'cardSummaryKey',
|
dataKey: 'cardSummaryKey',
|
||||||
url: 'cardSummaryUrl',
|
url: 'cardSummaryUrl',
|
||||||
filter: { key: 'cardFilter' },
|
filter: 'cardFilter',
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
vm = wrapper.vm;
|
vm = wrapper.vm;
|
||||||
|
@ -55,7 +50,7 @@ describe('CardSummary', () => {
|
||||||
|
|
||||||
it('should set correct props to the store', () => {
|
it('should set correct props to the store', () => {
|
||||||
expect(vm.store.url).toEqual('cardSummaryUrl');
|
expect(vm.store.url).toEqual('cardSummaryUrl');
|
||||||
expect(vm.store.filter).toEqual({ key: 'cardFilter' });
|
expect(vm.store.filter).toEqual('cardFilter');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should respond to prop changes and refetch data', async () => {
|
it('should respond to prop changes and refetch data', async () => {
|
||||||
|
|
|
@ -7,7 +7,7 @@ describe('VnSearchbar', () => {
|
||||||
let wrapper;
|
let wrapper;
|
||||||
let applyFilterSpy;
|
let applyFilterSpy;
|
||||||
const searchText = 'Bolas de madera';
|
const searchText = 'Bolas de madera';
|
||||||
const userParams = { staticKey: 'staticValue' };
|
const userParams = {staticKey: 'staticValue'};
|
||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
wrapper = createWrapper(VnSearchbar, {
|
wrapper = createWrapper(VnSearchbar, {
|
||||||
|
@ -23,9 +23,8 @@ describe('VnSearchbar', () => {
|
||||||
|
|
||||||
vm.searchText = searchText;
|
vm.searchText = searchText;
|
||||||
vm.arrayData.store.userParams = userParams;
|
vm.arrayData.store.userParams = userParams;
|
||||||
applyFilterSpy = vi
|
applyFilterSpy = vi.spyOn(vm.arrayData, 'applyFilter').mockImplementation(() => {});
|
||||||
.spyOn(vm.arrayData, 'applyFilter')
|
|
||||||
.mockImplementation(() => {});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
afterEach(() => {
|
afterEach(() => {
|
||||||
|
@ -33,9 +32,7 @@ describe('VnSearchbar', () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
it('search resets pagination and applies filter', async () => {
|
it('search resets pagination and applies filter', async () => {
|
||||||
const resetPaginationSpy = vi
|
const resetPaginationSpy = vi.spyOn(vm.arrayData, 'resetPagination').mockImplementation(() => {});
|
||||||
.spyOn(vm.arrayData, 'resetPagination')
|
|
||||||
.mockImplementation(() => {});
|
|
||||||
await vm.search();
|
await vm.search();
|
||||||
|
|
||||||
expect(resetPaginationSpy).toHaveBeenCalled();
|
expect(resetPaginationSpy).toHaveBeenCalled();
|
||||||
|
@ -51,7 +48,7 @@ describe('VnSearchbar', () => {
|
||||||
|
|
||||||
expect(applyFilterSpy).toHaveBeenCalledWith({
|
expect(applyFilterSpy).toHaveBeenCalledWith({
|
||||||
params: { staticKey: 'staticValue', search: searchText },
|
params: { staticKey: 'staticValue', search: searchText },
|
||||||
filter: { skip: 0 },
|
filter: {skip: 0},
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
import { vi, describe, expect, it, beforeAll, afterEach } from 'vitest';
|
import { vi, describe, expect, it, beforeAll, afterEach } from 'vitest';
|
||||||
|
import axios from 'axios';
|
||||||
import { createWrapper } from 'app/test/vitest/helper';
|
import { createWrapper } from 'app/test/vitest/helper';
|
||||||
import VnSms from 'src/components/ui/VnSms.vue';
|
import VnSms from 'src/components/ui/VnSms.vue';
|
||||||
|
|
||||||
|
@ -11,9 +12,6 @@ describe('VnSms', () => {
|
||||||
stubs: ['VnPaginate'],
|
stubs: ['VnPaginate'],
|
||||||
mocks: {},
|
mocks: {},
|
||||||
},
|
},
|
||||||
propsData: {
|
|
||||||
url: 'SmsUrl',
|
|
||||||
},
|
|
||||||
}).vm;
|
}).vm;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -4,8 +4,6 @@ import { useArrayData } from 'composables/useArrayData';
|
||||||
import { useRouter } from 'vue-router';
|
import { useRouter } from 'vue-router';
|
||||||
import * as vueRouter from 'vue-router';
|
import * as vueRouter from 'vue-router';
|
||||||
import { setActivePinia, createPinia } from 'pinia';
|
import { setActivePinia, createPinia } from 'pinia';
|
||||||
import { defineComponent, h } from 'vue';
|
|
||||||
import { mount } from '@vue/test-utils';
|
|
||||||
|
|
||||||
describe('useArrayData', () => {
|
describe('useArrayData', () => {
|
||||||
const filter = '{"limit":20,"skip":0}';
|
const filter = '{"limit":20,"skip":0}';
|
||||||
|
@ -45,7 +43,7 @@ describe('useArrayData', () => {
|
||||||
it('should fetch and replace url with new params', async () => {
|
it('should fetch and replace url with new params', async () => {
|
||||||
vi.spyOn(axios, 'get').mockResolvedValueOnce({ data: [] });
|
vi.spyOn(axios, 'get').mockResolvedValueOnce({ data: [] });
|
||||||
|
|
||||||
const arrayData = mountArrayData('ArrayData', {
|
const arrayData = useArrayData('ArrayData', {
|
||||||
url: 'mockUrl',
|
url: 'mockUrl',
|
||||||
searchUrl: 'params',
|
searchUrl: 'params',
|
||||||
});
|
});
|
||||||
|
@ -74,7 +72,7 @@ describe('useArrayData', () => {
|
||||||
data: [{ id: 1 }],
|
data: [{ id: 1 }],
|
||||||
});
|
});
|
||||||
|
|
||||||
const arrayData = mountArrayData('ArrayData', {
|
const arrayData = useArrayData('ArrayData', {
|
||||||
url: 'mockUrl',
|
url: 'mockUrl',
|
||||||
navigate: {},
|
navigate: {},
|
||||||
});
|
});
|
||||||
|
@ -96,7 +94,7 @@ describe('useArrayData', () => {
|
||||||
],
|
],
|
||||||
});
|
});
|
||||||
|
|
||||||
const arrayData = mountArrayData('ArrayData', {
|
const arrayData = useArrayData('ArrayData', {
|
||||||
url: 'mockUrl',
|
url: 'mockUrl',
|
||||||
oneRecord: true,
|
oneRecord: true,
|
||||||
});
|
});
|
||||||
|
@ -109,17 +107,3 @@ describe('useArrayData', () => {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
function mountArrayData(...args) {
|
|
||||||
let arrayData;
|
|
||||||
|
|
||||||
const TestComponent = defineComponent({
|
|
||||||
setup() {
|
|
||||||
arrayData = useArrayData(...args);
|
|
||||||
return () => h('div');
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const asd = mount(TestComponent);
|
|
||||||
return arrayData;
|
|
||||||
}
|
|
||||||
|
|
|
@ -64,84 +64,88 @@ describe('session', () => {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('login', () => {
|
describe(
|
||||||
const expectedUser = {
|
'login',
|
||||||
id: 999,
|
() => {
|
||||||
name: `T'Challa`,
|
const expectedUser = {
|
||||||
nickname: 'Black Panther',
|
id: 999,
|
||||||
lang: 'en',
|
name: `T'Challa`,
|
||||||
userConfig: {
|
nickname: 'Black Panther',
|
||||||
darkMode: false,
|
lang: 'en',
|
||||||
},
|
userConfig: {
|
||||||
worker: { department: { departmentFk: 155 } },
|
darkMode: false,
|
||||||
};
|
|
||||||
const rolesData = [
|
|
||||||
{
|
|
||||||
role: {
|
|
||||||
name: 'salesPerson',
|
|
||||||
},
|
},
|
||||||
},
|
worker: { department: { departmentFk: 155 } },
|
||||||
{
|
};
|
||||||
role: {
|
const rolesData = [
|
||||||
name: 'admin',
|
{
|
||||||
|
role: {
|
||||||
|
name: 'salesPerson',
|
||||||
|
},
|
||||||
},
|
},
|
||||||
},
|
{
|
||||||
];
|
role: {
|
||||||
beforeEach(() => {
|
name: 'admin',
|
||||||
vi.spyOn(axios, 'get').mockImplementation((url) => {
|
},
|
||||||
if (url === 'VnUsers/acls') return Promise.resolve({ data: [] });
|
},
|
||||||
return Promise.resolve({
|
];
|
||||||
data: { roles: rolesData, user: expectedUser },
|
beforeEach(() => {
|
||||||
|
vi.spyOn(axios, 'get').mockImplementation((url) => {
|
||||||
|
if (url === 'VnUsers/acls') return Promise.resolve({ data: [] });
|
||||||
|
return Promise.resolve({
|
||||||
|
data: { roles: rolesData, user: expectedUser },
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
|
||||||
|
|
||||||
it('should fetch the user roles and then set token in the sessionStorage', async () => {
|
it('should fetch the user roles and then set token in the sessionStorage', async () => {
|
||||||
const expectedRoles = ['salesPerson', 'admin'];
|
const expectedRoles = ['salesPerson', 'admin'];
|
||||||
const expectedToken = 'mySessionToken';
|
const expectedToken = 'mySessionToken';
|
||||||
const expectedTokenMultimedia = 'mySessionTokenMultimedia';
|
const expectedTokenMultimedia = 'mySessionTokenMultimedia';
|
||||||
const keepLogin = false;
|
const keepLogin = false;
|
||||||
|
|
||||||
await session.login({
|
await session.login({
|
||||||
token: expectedToken,
|
token: expectedToken,
|
||||||
tokenMultimedia: expectedTokenMultimedia,
|
tokenMultimedia: expectedTokenMultimedia,
|
||||||
keepLogin,
|
keepLogin,
|
||||||
|
});
|
||||||
|
|
||||||
|
const roles = state.getRoles();
|
||||||
|
const localToken = localStorage.getItem('token');
|
||||||
|
const sessionToken = sessionStorage.getItem('token');
|
||||||
|
|
||||||
|
expect(roles.value).toEqual(expectedRoles);
|
||||||
|
expect(localToken).toBeNull();
|
||||||
|
expect(sessionToken).toEqual(expectedToken);
|
||||||
|
|
||||||
|
await session.destroy(); // this clears token and user for any other test
|
||||||
});
|
});
|
||||||
|
|
||||||
const roles = state.getRoles();
|
it('should fetch the user roles and then set token in the localStorage', async () => {
|
||||||
const localToken = localStorage.getItem('token');
|
const expectedRoles = ['salesPerson', 'admin'];
|
||||||
const sessionToken = sessionStorage.getItem('token');
|
const expectedToken = 'myLocalToken';
|
||||||
|
const expectedTokenMultimedia = 'myLocalTokenMultimedia';
|
||||||
|
const keepLogin = true;
|
||||||
|
|
||||||
expect(roles.value).toEqual(expectedRoles);
|
await session.login({
|
||||||
expect(localToken).toBeNull();
|
token: expectedToken,
|
||||||
expect(sessionToken).toEqual(expectedToken);
|
tokenMultimedia: expectedTokenMultimedia,
|
||||||
|
keepLogin,
|
||||||
|
});
|
||||||
|
|
||||||
await session.destroy(); // this clears token and user for any other test
|
const roles = state.getRoles();
|
||||||
});
|
const localToken = localStorage.getItem('token');
|
||||||
|
const sessionToken = sessionStorage.getItem('token');
|
||||||
|
|
||||||
it('should fetch the user roles and then set token in the localStorage', async () => {
|
expect(roles.value).toEqual(expectedRoles);
|
||||||
const expectedRoles = ['salesPerson', 'admin'];
|
expect(localToken).toEqual(expectedToken);
|
||||||
const expectedToken = 'myLocalToken';
|
expect(sessionToken).toBeNull();
|
||||||
const expectedTokenMultimedia = 'myLocalTokenMultimedia';
|
|
||||||
const keepLogin = true;
|
|
||||||
|
|
||||||
await session.login({
|
await session.destroy(); // this clears token and user for any other test
|
||||||
token: expectedToken,
|
|
||||||
tokenMultimedia: expectedTokenMultimedia,
|
|
||||||
keepLogin,
|
|
||||||
});
|
});
|
||||||
|
},
|
||||||
const roles = state.getRoles();
|
{},
|
||||||
const localToken = localStorage.getItem('token');
|
);
|
||||||
const sessionToken = sessionStorage.getItem('token');
|
|
||||||
|
|
||||||
expect(roles.value).toEqual(expectedRoles);
|
|
||||||
expect(localToken).toEqual(expectedToken);
|
|
||||||
expect(sessionToken).toBeNull();
|
|
||||||
|
|
||||||
await session.destroy(); // this clears token and user for any other test
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('RenewToken', () => {
|
describe('RenewToken', () => {
|
||||||
const expectedToken = 'myToken';
|
const expectedToken = 'myToken';
|
||||||
|
|
|
@ -6,7 +6,6 @@ globals:
|
||||||
quantity: Quantity
|
quantity: Quantity
|
||||||
entity: Entity
|
entity: Entity
|
||||||
preview: Preview
|
preview: Preview
|
||||||
scrollToTop: Go up
|
|
||||||
user: User
|
user: User
|
||||||
details: Details
|
details: Details
|
||||||
collapseMenu: Collapse lateral menu
|
collapseMenu: Collapse lateral menu
|
||||||
|
@ -20,7 +19,6 @@ globals:
|
||||||
logOut: Log out
|
logOut: Log out
|
||||||
date: Date
|
date: Date
|
||||||
dataSaved: Data saved
|
dataSaved: Data saved
|
||||||
openDetail: Open detail
|
|
||||||
dataDeleted: Data deleted
|
dataDeleted: Data deleted
|
||||||
delete: Delete
|
delete: Delete
|
||||||
search: Search
|
search: Search
|
||||||
|
@ -163,8 +161,8 @@ globals:
|
||||||
noData: No data available
|
noData: No data available
|
||||||
vehicle: Vehicle
|
vehicle: Vehicle
|
||||||
selectDocumentId: Select document id
|
selectDocumentId: Select document id
|
||||||
document: Document
|
|
||||||
import: Import from existing
|
import: Import from existing
|
||||||
|
document: Document
|
||||||
pageTitles:
|
pageTitles:
|
||||||
logIn: Login
|
logIn: Login
|
||||||
addressEdit: Update address
|
addressEdit: Update address
|
||||||
|
@ -346,6 +344,7 @@ globals:
|
||||||
parking: Parking
|
parking: Parking
|
||||||
vehicleList: Vehicles
|
vehicleList: Vehicles
|
||||||
vehicle: Vehicle
|
vehicle: Vehicle
|
||||||
|
properties: Properties
|
||||||
unsavedPopup:
|
unsavedPopup:
|
||||||
title: Unsaved changes will be lost
|
title: Unsaved changes will be lost
|
||||||
subtitle: Are you sure exit without saving?
|
subtitle: Are you sure exit without saving?
|
||||||
|
@ -394,6 +393,7 @@ errors:
|
||||||
updateUserConfig: Error updating user config
|
updateUserConfig: Error updating user config
|
||||||
tokenConfig: Error fetching token config
|
tokenConfig: Error fetching token config
|
||||||
writeRequest: The requested operation could not be completed
|
writeRequest: The requested operation could not be completed
|
||||||
|
documentIdEmpty: The document identifier can't be empty
|
||||||
login:
|
login:
|
||||||
title: Login
|
title: Login
|
||||||
username: Username
|
username: Username
|
||||||
|
|
|
@ -6,7 +6,6 @@ globals:
|
||||||
quantity: Cantidad
|
quantity: Cantidad
|
||||||
entity: Entidad
|
entity: Entidad
|
||||||
preview: Vista previa
|
preview: Vista previa
|
||||||
scrollToTop: Ir arriba
|
|
||||||
user: Usuario
|
user: Usuario
|
||||||
details: Detalles
|
details: Detalles
|
||||||
collapseMenu: Contraer menú lateral
|
collapseMenu: Contraer menú lateral
|
||||||
|
@ -21,11 +20,10 @@ globals:
|
||||||
date: Fecha
|
date: Fecha
|
||||||
dataSaved: Datos guardados
|
dataSaved: Datos guardados
|
||||||
dataDeleted: Datos eliminados
|
dataDeleted: Datos eliminados
|
||||||
dataCreated: Datos creados
|
|
||||||
openDetail: Ver detalle
|
|
||||||
delete: Eliminar
|
delete: Eliminar
|
||||||
search: Buscar
|
search: Buscar
|
||||||
changes: Cambios
|
changes: Cambios
|
||||||
|
dataCreated: Datos creados
|
||||||
add: Añadir
|
add: Añadir
|
||||||
create: Crear
|
create: Crear
|
||||||
edit: Modificar
|
edit: Modificar
|
||||||
|
@ -166,9 +164,9 @@ globals:
|
||||||
noData: Datos no disponibles
|
noData: Datos no disponibles
|
||||||
department: Departamento
|
department: Departamento
|
||||||
vehicle: Vehículo
|
vehicle: Vehículo
|
||||||
selectDocumentId: Seleccione el id de gestión documental
|
selectDocumentId: Introduzca id de gestión documental
|
||||||
document: Documento
|
|
||||||
import: Importar desde existente
|
import: Importar desde existente
|
||||||
|
document: Documento
|
||||||
pageTitles:
|
pageTitles:
|
||||||
logIn: Inicio de sesión
|
logIn: Inicio de sesión
|
||||||
addressEdit: Modificar consignatario
|
addressEdit: Modificar consignatario
|
||||||
|
@ -349,6 +347,7 @@ globals:
|
||||||
parking: Parking
|
parking: Parking
|
||||||
vehicleList: Vehículos
|
vehicleList: Vehículos
|
||||||
vehicle: Vehículo
|
vehicle: Vehículo
|
||||||
|
properties: Propiedades
|
||||||
unsavedPopup:
|
unsavedPopup:
|
||||||
title: Los cambios que no haya guardado se perderán
|
title: Los cambios que no haya guardado se perderán
|
||||||
subtitle: ¿Seguro que quiere salir sin guardar?
|
subtitle: ¿Seguro que quiere salir sin guardar?
|
||||||
|
@ -390,6 +389,7 @@ errors:
|
||||||
updateUserConfig: Error al actualizar la configuración de usuario
|
updateUserConfig: Error al actualizar la configuración de usuario
|
||||||
tokenConfig: Error al obtener configuración de token
|
tokenConfig: Error al obtener configuración de token
|
||||||
writeRequest: No se pudo completar la operación solicitada
|
writeRequest: No se pudo completar la operación solicitada
|
||||||
|
documentIdEmpty: El número de documento no puede estar vacío
|
||||||
login:
|
login:
|
||||||
title: Inicio de sesión
|
title: Inicio de sesión
|
||||||
username: Nombre de usuario
|
username: Nombre de usuario
|
||||||
|
|
|
@ -23,7 +23,7 @@ const claimDms = ref([
|
||||||
]);
|
]);
|
||||||
const client = ref({});
|
const client = ref({});
|
||||||
const inputFile = ref();
|
const inputFile = ref();
|
||||||
const files = ref([]);
|
const files = ref({});
|
||||||
const spinnerRef = ref();
|
const spinnerRef = ref();
|
||||||
const claimDmsRef = ref();
|
const claimDmsRef = ref();
|
||||||
const dmsType = ref({});
|
const dmsType = ref({});
|
||||||
|
@ -255,8 +255,9 @@ function onDrag() {
|
||||||
icon="add"
|
icon="add"
|
||||||
color="primary"
|
color="primary"
|
||||||
>
|
>
|
||||||
<QFile
|
<QInput
|
||||||
ref="inputFile"
|
ref="inputFile"
|
||||||
|
type="file"
|
||||||
style="display: none"
|
style="display: none"
|
||||||
multiple
|
multiple
|
||||||
v-model="files"
|
v-model="files"
|
||||||
|
|
|
@ -52,7 +52,7 @@ describe('ClaimLines', () => {
|
||||||
expectedData,
|
expectedData,
|
||||||
{
|
{
|
||||||
signal: canceller.signal,
|
signal: canceller.signal,
|
||||||
},
|
}
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -69,7 +69,7 @@ describe('ClaimLines', () => {
|
||||||
expect.objectContaining({
|
expect.objectContaining({
|
||||||
message: 'Discount updated',
|
message: 'Discount updated',
|
||||||
type: 'positive',
|
type: 'positive',
|
||||||
}),
|
})
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -14,9 +14,6 @@ describe('ClaimLinesImport', () => {
|
||||||
fetch: vi.fn(),
|
fetch: vi.fn(),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
propsData: {
|
|
||||||
ticketId: 1,
|
|
||||||
},
|
|
||||||
}).vm;
|
}).vm;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -43,7 +40,7 @@ describe('ClaimLinesImport', () => {
|
||||||
expect.objectContaining({
|
expect.objectContaining({
|
||||||
message: 'Lines added to claim',
|
message: 'Lines added to claim',
|
||||||
type: 'positive',
|
type: 'positive',
|
||||||
}),
|
})
|
||||||
);
|
);
|
||||||
expect(vm.canceller).toEqual(null);
|
expect(vm.canceller).toEqual(null);
|
||||||
});
|
});
|
||||||
|
|
|
@ -41,10 +41,10 @@ describe('ClaimPhoto', () => {
|
||||||
await vm.deleteDms({ index: 0 });
|
await vm.deleteDms({ index: 0 });
|
||||||
|
|
||||||
expect(axios.post).toHaveBeenCalledWith(
|
expect(axios.post).toHaveBeenCalledWith(
|
||||||
`ClaimDms/${claimMock.claimDms[0].dmsFk}/removeFile`,
|
`ClaimDms/${claimMock.claimDms[0].dmsFk}/removeFile`
|
||||||
);
|
);
|
||||||
expect(vm.quasar.notify).toHaveBeenCalledWith(
|
expect(vm.quasar.notify).toHaveBeenCalledWith(
|
||||||
expect.objectContaining({ type: 'positive' }),
|
expect.objectContaining({ type: 'positive' })
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -63,7 +63,7 @@ describe('ClaimPhoto', () => {
|
||||||
data: { index: 1 },
|
data: { index: 1 },
|
||||||
promise: vm.deleteDms,
|
promise: vm.deleteDms,
|
||||||
},
|
},
|
||||||
}),
|
})
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -102,10 +102,10 @@ describe('ClaimPhoto', () => {
|
||||||
new FormData(),
|
new FormData(),
|
||||||
expect.objectContaining({
|
expect.objectContaining({
|
||||||
params: expect.objectContaining({ hasFile: false }),
|
params: expect.objectContaining({ hasFile: false }),
|
||||||
}),
|
})
|
||||||
);
|
);
|
||||||
expect(vm.quasar.notify).toHaveBeenCalledWith(
|
expect(vm.quasar.notify).toHaveBeenCalledWith(
|
||||||
expect.objectContaining({ type: 'positive' }),
|
expect.objectContaining({ type: 'positive' })
|
||||||
);
|
);
|
||||||
|
|
||||||
expect(vm.claimDmsRef.fetch).toHaveBeenCalledOnce();
|
expect(vm.claimDmsRef.fetch).toHaveBeenCalledOnce();
|
||||||
|
|
|
@ -39,7 +39,7 @@ const route = useRoute();
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
|
|
||||||
const entityId = computed(() => {
|
const entityId = computed(() => {
|
||||||
return Number($props.id || route.params.id);
|
return $props.id || route.params.id;
|
||||||
});
|
});
|
||||||
|
|
||||||
const data = ref(useCardDescription());
|
const data = ref(useCardDescription());
|
||||||
|
|
|
@ -11,7 +11,7 @@ const $props = defineProps({
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<QPopupProxy data-cy="CustomerDescriptor">
|
<QPopupProxy>
|
||||||
<CustomerDescriptor v-if="$props.id" :id="$props.id" :summary="CustomerSummary" />
|
<CustomerDescriptor v-if="$props.id" :id="$props.id" :summary="CustomerSummary" />
|
||||||
</QPopupProxy>
|
</QPopupProxy>
|
||||||
</template>
|
</template>
|
||||||
|
|
|
@ -86,7 +86,7 @@ async function acceptPropagate({ isEqualizated }) {
|
||||||
:required="true"
|
:required="true"
|
||||||
:rules="validate('client.socialName')"
|
:rules="validate('client.socialName')"
|
||||||
clearable
|
clearable
|
||||||
:uppercase="true"
|
uppercase="true"
|
||||||
v-model="data.socialName"
|
v-model="data.socialName"
|
||||||
>
|
>
|
||||||
<template #append>
|
<template #append>
|
||||||
|
|
|
@ -4,6 +4,7 @@ import { useI18n } from 'vue-i18n';
|
||||||
import { useRoute } from 'vue-router';
|
import { useRoute } from 'vue-router';
|
||||||
|
|
||||||
import { QBtn, useQuasar } from 'quasar';
|
import { QBtn, useQuasar } from 'quasar';
|
||||||
|
|
||||||
import WorkerDescriptorProxy from 'src/pages/Worker/Card/WorkerDescriptorProxy.vue';
|
import WorkerDescriptorProxy from 'src/pages/Worker/Card/WorkerDescriptorProxy.vue';
|
||||||
import { toDateTimeFormat } from 'src/filters/date';
|
import { toDateTimeFormat } from 'src/filters/date';
|
||||||
import VnTable from 'src/components/VnTable/VnTable.vue';
|
import VnTable from 'src/components/VnTable/VnTable.vue';
|
||||||
|
@ -73,11 +74,12 @@ const tableRef = ref();
|
||||||
<template>
|
<template>
|
||||||
<VnTable
|
<VnTable
|
||||||
ref="tableRef"
|
ref="tableRef"
|
||||||
data-key="CustomerSamples"
|
data-key="ClientSamples"
|
||||||
auto-load
|
auto-load
|
||||||
:user-filter="filter"
|
:filter="filter"
|
||||||
url="ClientSamples"
|
url="ClientSamples"
|
||||||
:columns="columns"
|
:columns="columns"
|
||||||
|
:pagination="{ rowsPerPage: 12 }"
|
||||||
:disable-option="{ card: true }"
|
:disable-option="{ card: true }"
|
||||||
:right-search="false"
|
:right-search="false"
|
||||||
:rows="rows"
|
:rows="rows"
|
||||||
|
|
|
@ -25,7 +25,7 @@ const $props = defineProps({
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const entityId = computed(() => Number($props.id || route.params.id));
|
const entityId = computed(() => $props.id || route.params.id);
|
||||||
const customer = computed(() => summary.value.entity);
|
const customer = computed(() => summary.value.entity);
|
||||||
const summary = ref();
|
const summary = ref();
|
||||||
const defaulterAmount = computed(() => customer.value.defaulters[0]?.amount);
|
const defaulterAmount = computed(() => customer.value.defaulters[0]?.amount);
|
||||||
|
|
|
@ -72,7 +72,7 @@ const exprBuilder = (param, value) => {
|
||||||
option-value="id"
|
option-value="id"
|
||||||
option-label="name"
|
option-label="name"
|
||||||
url="Departments"
|
url="Departments"
|
||||||
:no-one="true"
|
no-one="true"
|
||||||
/>
|
/>
|
||||||
</QItemSection>
|
</QItemSection>
|
||||||
</QItem>
|
</QItem>
|
||||||
|
|
|
@ -32,7 +32,7 @@ describe('CustomerPayments', () => {
|
||||||
expect.objectContaining({
|
expect.objectContaining({
|
||||||
message: 'Payment confirmed',
|
message: 'Payment confirmed',
|
||||||
type: 'positive',
|
type: 'positive',
|
||||||
}),
|
})
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -41,6 +41,7 @@ const sampleType = ref({ hasPreview: false });
|
||||||
const initialData = reactive({});
|
const initialData = reactive({});
|
||||||
const entityId = computed(() => route.params.id);
|
const entityId = computed(() => route.params.id);
|
||||||
const customer = computed(() => useArrayData('Customer').store?.data);
|
const customer = computed(() => useArrayData('Customer').store?.data);
|
||||||
|
const filterEmailUsers = { where: { userFk: user.value.id } };
|
||||||
const filterClientsAddresses = {
|
const filterClientsAddresses = {
|
||||||
include: [
|
include: [
|
||||||
{ relation: 'province', scope: { fields: ['name'] } },
|
{ relation: 'province', scope: { fields: ['name'] } },
|
||||||
|
@ -72,7 +73,7 @@ onBeforeMount(async () => {
|
||||||
|
|
||||||
const setEmailUser = (data) => {
|
const setEmailUser = (data) => {
|
||||||
optionsEmailUsers.value = data;
|
optionsEmailUsers.value = data;
|
||||||
initialData.replyTo = data[0]?.notificationEmail;
|
initialData.replyTo = data[0]?.email;
|
||||||
};
|
};
|
||||||
|
|
||||||
const setClientsAddresses = (data) => {
|
const setClientsAddresses = (data) => {
|
||||||
|
@ -181,12 +182,10 @@ const toCustomerSamples = () => {
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<FetchData
|
<FetchData
|
||||||
:filter="{
|
:filter="filterEmailUsers"
|
||||||
where: { id: customer.departmentFk },
|
|
||||||
}"
|
|
||||||
@on-fetch="setEmailUser"
|
@on-fetch="setEmailUser"
|
||||||
auto-load
|
auto-load
|
||||||
url="Departments"
|
url="EmailUsers"
|
||||||
/>
|
/>
|
||||||
<FetchData
|
<FetchData
|
||||||
:filter="filterClientsAddresses"
|
:filter="filterClientsAddresses"
|
||||||
|
|
|
@ -5,7 +5,7 @@ import InvoiceInSummary from './InvoiceInSummary.vue';
|
||||||
const $props = defineProps({
|
const $props = defineProps({
|
||||||
id: {
|
id: {
|
||||||
type: Number,
|
type: Number,
|
||||||
default: null,
|
required: true,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
|
@ -164,7 +164,6 @@ onMounted(async () => {
|
||||||
unelevated
|
unelevated
|
||||||
filled
|
filled
|
||||||
dense
|
dense
|
||||||
data-cy="formSubmitBtn"
|
|
||||||
/>
|
/>
|
||||||
<QBtn
|
<QBtn
|
||||||
v-else
|
v-else
|
||||||
|
@ -175,7 +174,6 @@ onMounted(async () => {
|
||||||
filled
|
filled
|
||||||
dense
|
dense
|
||||||
@click="getStatus = 'stopping'"
|
@click="getStatus = 'stopping'"
|
||||||
data-cy="formStopBtn"
|
|
||||||
/>
|
/>
|
||||||
</QForm>
|
</QForm>
|
||||||
</template>
|
</template>
|
||||||
|
|
|
@ -6,12 +6,10 @@ import { toCurrency } from 'filters/index';
|
||||||
import VnStockValueDisplay from 'src/components/ui/VnStockValueDisplay.vue';
|
import VnStockValueDisplay from 'src/components/ui/VnStockValueDisplay.vue';
|
||||||
import VnTable from 'src/components/VnTable/VnTable.vue';
|
import VnTable from 'src/components/VnTable/VnTable.vue';
|
||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
import { displayResults } from 'src/pages/Ticket/Negative/composables/notifyResults';
|
import notifyResults from 'src/utils/notifyResults';
|
||||||
import FetchData from 'components/FetchData.vue';
|
import FetchData from 'components/FetchData.vue';
|
||||||
import { useState } from 'src/composables/useState';
|
|
||||||
|
|
||||||
const MATCH = 'match';
|
const MATCH = 'match';
|
||||||
const { notifyResults } = displayResults();
|
|
||||||
|
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
const $props = defineProps({
|
const $props = defineProps({
|
||||||
|
@ -20,20 +18,14 @@ const $props = defineProps({
|
||||||
required: true,
|
required: true,
|
||||||
default: () => {},
|
default: () => {},
|
||||||
},
|
},
|
||||||
filter: {
|
|
||||||
type: Object,
|
|
||||||
required: true,
|
|
||||||
default: () => {},
|
|
||||||
},
|
|
||||||
replaceAction: {
|
replaceAction: {
|
||||||
type: Boolean,
|
type: Boolean,
|
||||||
required: true,
|
required: false,
|
||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
|
|
||||||
sales: {
|
sales: {
|
||||||
type: Array,
|
type: Array,
|
||||||
required: true,
|
required: false,
|
||||||
default: () => [],
|
default: () => [],
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
@ -44,8 +36,6 @@ const proposalTableRef = ref(null);
|
||||||
const sale = computed(() => $props.sales[0]);
|
const sale = computed(() => $props.sales[0]);
|
||||||
const saleFk = computed(() => sale.value.saleFk);
|
const saleFk = computed(() => sale.value.saleFk);
|
||||||
const filter = computed(() => ({
|
const filter = computed(() => ({
|
||||||
where: $props.filter,
|
|
||||||
|
|
||||||
itemFk: $props.itemLack.itemFk,
|
itemFk: $props.itemLack.itemFk,
|
||||||
sales: saleFk.value,
|
sales: saleFk.value,
|
||||||
}));
|
}));
|
||||||
|
@ -238,15 +228,11 @@ async function handleTicketConfig(data) {
|
||||||
url="TicketConfigs"
|
url="TicketConfigs"
|
||||||
:filter="{ fields: ['lackAlertPrice'] }"
|
:filter="{ fields: ['lackAlertPrice'] }"
|
||||||
@on-fetch="handleTicketConfig"
|
@on-fetch="handleTicketConfig"
|
||||||
></FetchData>
|
auto-load
|
||||||
<QInnerLoading
|
|
||||||
:showing="isLoading"
|
|
||||||
:label="t && t('globals.pleaseWait')"
|
|
||||||
color="primary"
|
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<VnTable
|
<VnTable
|
||||||
v-if="!isLoading"
|
v-if="ticketConfig"
|
||||||
auto-load
|
auto-load
|
||||||
data-cy="proposalTable"
|
data-cy="proposalTable"
|
||||||
ref="proposalTableRef"
|
ref="proposalTableRef"
|
||||||
|
|
|
@ -1,17 +1,13 @@
|
||||||
<script setup>
|
<script setup>
|
||||||
import ItemProposal from './ItemProposal.vue';
|
import ItemProposal from './ItemProposal.vue';
|
||||||
import { useDialogPluginComponent } from 'quasar';
|
import { useDialogPluginComponent } from 'quasar';
|
||||||
|
|
||||||
const $props = defineProps({
|
const $props = defineProps({
|
||||||
itemLack: {
|
itemLack: {
|
||||||
type: Object,
|
type: Object,
|
||||||
required: true,
|
required: true,
|
||||||
default: () => {},
|
default: () => {},
|
||||||
},
|
},
|
||||||
filter: {
|
|
||||||
type: Object,
|
|
||||||
required: true,
|
|
||||||
default: () => {},
|
|
||||||
},
|
|
||||||
replaceAction: {
|
replaceAction: {
|
||||||
type: Boolean,
|
type: Boolean,
|
||||||
required: false,
|
required: false,
|
||||||
|
@ -35,7 +31,7 @@ defineExpose({ show: () => dialogRef.value.show(), hide: () => dialogRef.value.h
|
||||||
<QDialog ref="dialogRef" transition-show="scale" transition-hide="scale">
|
<QDialog ref="dialogRef" transition-show="scale" transition-hide="scale">
|
||||||
<QCard class="dialog-width">
|
<QCard class="dialog-width">
|
||||||
<QCardSection class="row items-center q-pb-none">
|
<QCardSection class="row items-center q-pb-none">
|
||||||
<span class="text-h6 text-grey">{{ $t('itemProposal') }}</span>
|
<span class="text-h6 text-grey">{{ $t('Item proposal') }}</span>
|
||||||
<QSpace />
|
<QSpace />
|
||||||
<QBtn icon="close" flat round dense v-close-popup />
|
<QBtn icon="close" flat round dense v-close-popup />
|
||||||
</QCardSection>
|
</QCardSection>
|
||||||
|
|
|
@ -8,14 +8,14 @@ import VnRow from 'src/components/ui/VnRow.vue';
|
||||||
class="q-pa-md"
|
class="q-pa-md"
|
||||||
:style="{ 'flex-direction': $q.screen.lt.lg ? 'column' : 'row', gap: '0px' }"
|
:style="{ 'flex-direction': $q.screen.lt.lg ? 'column' : 'row', gap: '0px' }"
|
||||||
>
|
>
|
||||||
<div style="flex: 0.3" data-cy="clientsOnWebsite">
|
<div style="flex: 0.3">
|
||||||
<span
|
<span
|
||||||
class="q-ml-md text-body1"
|
class="q-ml-md text-body1"
|
||||||
v-text="$t('salesMonitor.clientsOnWebsite')"
|
v-text="$t('salesMonitor.clientsOnWebsite')"
|
||||||
/>
|
/>
|
||||||
<SalesClientTable />
|
<SalesClientTable />
|
||||||
</div>
|
</div>
|
||||||
<div style="flex: 0.7" data-cy="recentOrderActions">
|
<div style="flex: 0.7">
|
||||||
<span
|
<span
|
||||||
class="q-ml-md text-body1"
|
class="q-ml-md text-body1"
|
||||||
v-text="$t('salesMonitor.recentOrderActions')"
|
v-text="$t('salesMonitor.recentOrderActions')"
|
||||||
|
|
|
@ -9,7 +9,6 @@ import { toDateFormat, toDateTimeFormat } from 'src/filters/date.js';
|
||||||
import { toCurrency } from 'src/filters';
|
import { toCurrency } from 'src/filters';
|
||||||
import { useVnConfirm } from 'composables/useVnConfirm';
|
import { useVnConfirm } from 'composables/useVnConfirm';
|
||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
import VnDateBadge from 'src/components/common/VnDateBadge.vue';
|
|
||||||
import useOpenURL from 'src/composables/useOpenURL';
|
import useOpenURL from 'src/composables/useOpenURL';
|
||||||
|
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
|
@ -166,7 +165,16 @@ const openTab = (id) => useOpenURL(`#/order/${id}/summary`);
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<template #column-dateSend="{ row }">
|
<template #column-dateSend="{ row }">
|
||||||
<VnDateBadge :date="row.date_send" />
|
<QTd>
|
||||||
|
<QBadge
|
||||||
|
:color="getBadgeColor(row.date_send)"
|
||||||
|
text-color="black"
|
||||||
|
class="q-pa-sm"
|
||||||
|
style="font-size: 14px"
|
||||||
|
>
|
||||||
|
{{ toDateFormat(row.date_send) }}
|
||||||
|
</QBadge>
|
||||||
|
</QTd>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<template #column-clientFk="{ row }">
|
<template #column-clientFk="{ row }">
|
||||||
|
|
|
@ -9,7 +9,6 @@ import VnInput from 'src/components/common/VnInput.vue';
|
||||||
import VnInputNumber from 'src/components/common/VnInputNumber.vue';
|
import VnInputNumber from 'src/components/common/VnInputNumber.vue';
|
||||||
import FetchData from 'src/components/FetchData.vue';
|
import FetchData from 'src/components/FetchData.vue';
|
||||||
import { dateRange } from 'src/filters';
|
import { dateRange } from 'src/filters';
|
||||||
import VnCheckbox from 'src/components/common/VnCheckbox.vue';
|
|
||||||
|
|
||||||
defineProps({ dataKey: { type: String, required: true } });
|
defineProps({ dataKey: { type: String, required: true } });
|
||||||
const { t, te } = useI18n();
|
const { t, te } = useI18n();
|
||||||
|
@ -210,7 +209,7 @@ const getLocale = (label) => {
|
||||||
</QItem>
|
</QItem>
|
||||||
<QItem>
|
<QItem>
|
||||||
<QItemSection>
|
<QItemSection>
|
||||||
<VnCheckbox
|
<QCheckbox
|
||||||
:label="t('params.myTeam')"
|
:label="t('params.myTeam')"
|
||||||
v-model="params.myTeam"
|
v-model="params.myTeam"
|
||||||
toggle-indeterminate
|
toggle-indeterminate
|
||||||
|
@ -219,7 +218,7 @@ const getLocale = (label) => {
|
||||||
</QItem>
|
</QItem>
|
||||||
<QItem>
|
<QItem>
|
||||||
<QItemSection>
|
<QItemSection>
|
||||||
<VnCheckbox
|
<QCheckbox
|
||||||
:label="t('params.problems')"
|
:label="t('params.problems')"
|
||||||
v-model="params.problems"
|
v-model="params.problems"
|
||||||
toggle-indeterminate
|
toggle-indeterminate
|
||||||
|
@ -228,7 +227,7 @@ const getLocale = (label) => {
|
||||||
</QItem>
|
</QItem>
|
||||||
<QItem>
|
<QItem>
|
||||||
<QItemSection>
|
<QItemSection>
|
||||||
<VnCheckbox
|
<QCheckbox
|
||||||
:label="t('params.pending')"
|
:label="t('params.pending')"
|
||||||
v-model="params.pending"
|
v-model="params.pending"
|
||||||
toggle-indeterminate
|
toggle-indeterminate
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
<script setup>
|
<script setup>
|
||||||
import { ref, computed, onMounted } from 'vue';
|
import { ref, computed, onMounted } from 'vue';
|
||||||
import { useI18n } from 'vue-i18n';
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import FetchData from 'components/FetchData.vue';
|
||||||
import DepartmentDescriptorProxy from 'src/pages/Worker/Department/Card/DepartmentDescriptorProxy.vue';
|
import DepartmentDescriptorProxy from 'src/pages/Worker/Department/Card/DepartmentDescriptorProxy.vue';
|
||||||
import CustomerDescriptorProxy from 'src/pages/Customer/Card/CustomerDescriptorProxy.vue';
|
import CustomerDescriptorProxy from 'src/pages/Customer/Card/CustomerDescriptorProxy.vue';
|
||||||
import TicketDescriptorProxy from 'src/pages/Ticket/Card/TicketDescriptorProxy.vue';
|
import TicketDescriptorProxy from 'src/pages/Ticket/Card/TicketDescriptorProxy.vue';
|
||||||
|
@ -167,11 +168,9 @@ const columns = computed(() => [
|
||||||
component: 'select',
|
component: 'select',
|
||||||
name: 'provinceFk',
|
name: 'provinceFk',
|
||||||
attrs: {
|
attrs: {
|
||||||
url: 'Provinces',
|
options: provinceOpts.value,
|
||||||
fields: ['id', 'name'],
|
'option-value': 'id',
|
||||||
sortBy: ['name ASC'],
|
'option-label': 'name',
|
||||||
optionValue: 'id',
|
|
||||||
optionLabel: 'name',
|
|
||||||
dense: true,
|
dense: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -184,11 +183,9 @@ const columns = computed(() => [
|
||||||
component: 'select',
|
component: 'select',
|
||||||
name: 'stateFk',
|
name: 'stateFk',
|
||||||
attrs: {
|
attrs: {
|
||||||
sortBy: ['name ASC'],
|
options: stateOpts.value,
|
||||||
url: 'States',
|
'option-value': 'id',
|
||||||
fields: ['id', 'name'],
|
'option-label': 'name',
|
||||||
optionValue: 'id',
|
|
||||||
optionLabel: 'name',
|
|
||||||
dense: true,
|
dense: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -215,12 +212,9 @@ const columns = computed(() => [
|
||||||
component: 'select',
|
component: 'select',
|
||||||
name: 'zoneFk',
|
name: 'zoneFk',
|
||||||
attrs: {
|
attrs: {
|
||||||
url: 'Zones',
|
options: zoneOpts.value,
|
||||||
fields: ['id', 'name'],
|
'option-value': 'id',
|
||||||
sortBy: ['name ASC'],
|
'option-label': 'name',
|
||||||
|
|
||||||
optionValue: 'id',
|
|
||||||
optionLabel: 'name',
|
|
||||||
dense: true,
|
dense: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -231,12 +225,11 @@ const columns = computed(() => [
|
||||||
align: 'left',
|
align: 'left',
|
||||||
columnFilter: {
|
columnFilter: {
|
||||||
component: 'select',
|
component: 'select',
|
||||||
|
url: 'PayMethods',
|
||||||
attrs: {
|
attrs: {
|
||||||
url: 'PayMethods',
|
options: PayMethodOpts.value,
|
||||||
fields: ['id', 'name'],
|
|
||||||
sortBy: ['id ASC'],
|
|
||||||
optionLabel: 'name',
|
|
||||||
optionValue: 'id',
|
optionValue: 'id',
|
||||||
|
optionLabel: 'name',
|
||||||
dense: true,
|
dense: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -261,9 +254,7 @@ const columns = computed(() => [
|
||||||
columnFilter: {
|
columnFilter: {
|
||||||
component: 'select',
|
component: 'select',
|
||||||
attrs: {
|
attrs: {
|
||||||
url: 'Departments',
|
options: DepartmentOpts.value,
|
||||||
fields: ['id', 'name'],
|
|
||||||
sortBy: ['id ASC'],
|
|
||||||
dense: true,
|
dense: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -274,12 +265,11 @@ const columns = computed(() => [
|
||||||
align: 'left',
|
align: 'left',
|
||||||
columnFilter: {
|
columnFilter: {
|
||||||
component: 'select',
|
component: 'select',
|
||||||
|
url: 'ItemPackingTypes',
|
||||||
attrs: {
|
attrs: {
|
||||||
url: 'ItemPackingTypes',
|
options: ItemPackingTypeOpts.value,
|
||||||
fields: ['code'],
|
'option-value': 'code',
|
||||||
sortBy: ['code ASC'],
|
'option-label': 'code',
|
||||||
optionValue: 'code',
|
|
||||||
optionCode: 'code',
|
|
||||||
dense: true,
|
dense: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -334,6 +324,60 @@ const totalPriceColor = (ticket) => {
|
||||||
const openTab = (id) => useOpenURL(`#/ticket/${id}/sale`);
|
const openTab = (id) => useOpenURL(`#/ticket/${id}/sale`);
|
||||||
</script>
|
</script>
|
||||||
<template>
|
<template>
|
||||||
|
<FetchData
|
||||||
|
url="Provinces"
|
||||||
|
:filter="{
|
||||||
|
fields: ['id', 'name'],
|
||||||
|
order: 'name ASC',
|
||||||
|
}"
|
||||||
|
auto-load
|
||||||
|
@on-fetch="(data) => (provinceOpts = data)"
|
||||||
|
/>
|
||||||
|
<FetchData
|
||||||
|
url="States"
|
||||||
|
:filter="{
|
||||||
|
fields: ['id', 'name'],
|
||||||
|
order: 'name ASC',
|
||||||
|
}"
|
||||||
|
auto-load
|
||||||
|
@on-fetch="(data) => (stateOpts = data)"
|
||||||
|
/>
|
||||||
|
<FetchData
|
||||||
|
url="Zones"
|
||||||
|
:filter="{
|
||||||
|
fields: ['id', 'name'],
|
||||||
|
order: 'name ASC',
|
||||||
|
}"
|
||||||
|
auto-load
|
||||||
|
@on-fetch="(data) => (zoneOpts = data)"
|
||||||
|
/>
|
||||||
|
<FetchData
|
||||||
|
url="ItemPackingTypes"
|
||||||
|
:filter="{
|
||||||
|
fields: ['code'],
|
||||||
|
order: 'code ASC',
|
||||||
|
}"
|
||||||
|
auto-load
|
||||||
|
@on-fetch="(data) => (ItemPackingTypeOpts = data)"
|
||||||
|
/>
|
||||||
|
<FetchData
|
||||||
|
url="Departments"
|
||||||
|
:filter="{
|
||||||
|
fields: ['id', 'name'],
|
||||||
|
order: 'id ASC',
|
||||||
|
}"
|
||||||
|
auto-load
|
||||||
|
@on-fetch="(data) => (DepartmentOpts = data)"
|
||||||
|
/>
|
||||||
|
<FetchData
|
||||||
|
url="PayMethods"
|
||||||
|
:filter="{
|
||||||
|
fields: ['id', 'name'],
|
||||||
|
order: 'id ASC',
|
||||||
|
}"
|
||||||
|
auto-load
|
||||||
|
@on-fetch="(data) => (PayMethodOpts = data)"
|
||||||
|
/>
|
||||||
<MonitorTicketSearchbar />
|
<MonitorTicketSearchbar />
|
||||||
<RightMenu>
|
<RightMenu>
|
||||||
<template #right-panel>
|
<template #right-panel>
|
||||||
|
|
|
@ -120,6 +120,7 @@ watch(
|
||||||
:data-key="dataKey"
|
:data-key="dataKey"
|
||||||
:tag-value="tagValue"
|
:tag-value="tagValue"
|
||||||
:tags="tags"
|
:tags="tags"
|
||||||
|
:initial-catalog-params="catalogParams"
|
||||||
:arrayData
|
:arrayData
|
||||||
/>
|
/>
|
||||||
</template>
|
</template>
|
||||||
|
|
|
@ -27,7 +27,7 @@ const getTotalRef = ref();
|
||||||
const total = ref(0);
|
const total = ref(0);
|
||||||
|
|
||||||
const entityId = computed(() => {
|
const entityId = computed(() => {
|
||||||
return Number($props.id || route.params.id);
|
return $props.id || route.params.id;
|
||||||
});
|
});
|
||||||
|
|
||||||
const orderTotal = computed(() => state.get('orderTotal') ?? 0);
|
const orderTotal = computed(() => state.get('orderTotal') ?? 0);
|
||||||
|
|
|
@ -0,0 +1,191 @@
|
||||||
|
<script setup>
|
||||||
|
import { useRoute } from 'vue-router';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import FormModel from 'components/FormModel.vue';
|
||||||
|
import VnRow from 'components/ui/VnRow.vue';
|
||||||
|
import VnSelect from 'components/common/VnSelect.vue';
|
||||||
|
import VnInput from 'components/common/VnInput.vue';
|
||||||
|
import VnInputDate from 'components/common/VnInputDate.vue';
|
||||||
|
import VnSubToolbar from 'src/components/ui/VnSubToolbar.vue';
|
||||||
|
|
||||||
|
const { t } = useI18n();
|
||||||
|
const route = useRoute();
|
||||||
|
|
||||||
|
const isNew = Boolean(!route.params.id);
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<VnSubToolbar v-if="isNew" />
|
||||||
|
<div class="q-pa-md">
|
||||||
|
<FormModel
|
||||||
|
:url-update="`Properties/${route.params.id}`"
|
||||||
|
model="Property"
|
||||||
|
auto-load
|
||||||
|
>
|
||||||
|
<template #form="{ data }">
|
||||||
|
<VnRow>
|
||||||
|
<VnInput
|
||||||
|
:label="t('property.name')"
|
||||||
|
v-model="data.name"
|
||||||
|
fill-input
|
||||||
|
/>
|
||||||
|
<VnInput
|
||||||
|
:label="t('property.cadaster')"
|
||||||
|
v-model="data.cadaster"
|
||||||
|
fill-input
|
||||||
|
/>
|
||||||
|
</VnRow>
|
||||||
|
<VnRow>
|
||||||
|
<VnSelect
|
||||||
|
url="propertyGroups"
|
||||||
|
:label="t('property.group')"
|
||||||
|
v-model="data.propertyGroupFk"
|
||||||
|
option-value="id"
|
||||||
|
option-label="description"
|
||||||
|
hide-selected
|
||||||
|
>
|
||||||
|
<template #option="scope">
|
||||||
|
<QItem v-bind="scope.itemProps">
|
||||||
|
<QItemSection>
|
||||||
|
<QItemLabel>
|
||||||
|
{{ `${scope.opt.id}: ${scope.opt.description}` }}
|
||||||
|
</QItemLabel>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
</template>
|
||||||
|
</VnSelect>
|
||||||
|
<VnSelect
|
||||||
|
url="Companies"
|
||||||
|
:label="t('property.owner')"
|
||||||
|
v-model="data.companyFk"
|
||||||
|
option-value="id"
|
||||||
|
option-label="code"
|
||||||
|
hide-selected
|
||||||
|
>
|
||||||
|
<template #option="scope">
|
||||||
|
<QItem v-bind="scope.itemProps">
|
||||||
|
<QItemSection>
|
||||||
|
<QItemLabel>
|
||||||
|
{{ `${scope.opt.id}: ${scope.opt.code}` }}
|
||||||
|
</QItemLabel>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
</template>
|
||||||
|
</VnSelect>
|
||||||
|
</VnRow>
|
||||||
|
<VnRow>
|
||||||
|
<VnInput
|
||||||
|
:label="t('property.map')"
|
||||||
|
v-model="data.url"
|
||||||
|
fill-input
|
||||||
|
/>
|
||||||
|
</VnRow>
|
||||||
|
<VnRow>
|
||||||
|
<VnInputDate
|
||||||
|
placeholder="dd-mm-aaa"
|
||||||
|
:label="t('property.purchased')"
|
||||||
|
v-model="data.purchased"
|
||||||
|
fill-input
|
||||||
|
/>
|
||||||
|
<VnInput
|
||||||
|
:label="t('property.value')"
|
||||||
|
v-model="data.value"
|
||||||
|
fill-input
|
||||||
|
/>
|
||||||
|
<VnInput
|
||||||
|
:label="t('property.protocol')"
|
||||||
|
v-model="data.protocol"
|
||||||
|
fill-input
|
||||||
|
/>
|
||||||
|
</VnRow>
|
||||||
|
<VnRow>
|
||||||
|
<VnInput
|
||||||
|
:label="t('property.smallHolding')"
|
||||||
|
v-model="data.smallHolding"
|
||||||
|
fill-input
|
||||||
|
/>
|
||||||
|
<VnInput
|
||||||
|
:label="t('property.area')"
|
||||||
|
v-model="data.area"
|
||||||
|
fill-input
|
||||||
|
/>
|
||||||
|
<VnInput
|
||||||
|
:label="t('property.allocation')"
|
||||||
|
v-model="data.allocation"
|
||||||
|
fill-input
|
||||||
|
/>
|
||||||
|
</VnRow>
|
||||||
|
<VnRow>
|
||||||
|
<VnSelect
|
||||||
|
url="Towns"
|
||||||
|
:label="t('property.town')"
|
||||||
|
v-model="data.townFk"
|
||||||
|
option-value="id"
|
||||||
|
option-label="name"
|
||||||
|
hide-selected
|
||||||
|
>
|
||||||
|
<template #option="scope">
|
||||||
|
<QItem v-bind="scope.itemProps">
|
||||||
|
<QItemSection>
|
||||||
|
<QItemLabel>
|
||||||
|
{{ `${scope.opt.id}: ${scope.opt.name}` }}
|
||||||
|
</QItemLabel>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
</template>
|
||||||
|
</VnSelect>
|
||||||
|
<VnInput
|
||||||
|
:label="t('property.m2')"
|
||||||
|
v-model="data.m2"
|
||||||
|
fill-input
|
||||||
|
/>
|
||||||
|
<VnInput
|
||||||
|
:label="t('property.registry')"
|
||||||
|
v-model="data.registry"
|
||||||
|
fill-input
|
||||||
|
/>
|
||||||
|
</VnRow>
|
||||||
|
<VnRow>
|
||||||
|
<VnInput
|
||||||
|
:label="t('property.tome')"
|
||||||
|
v-model="data.tome"
|
||||||
|
fill-input
|
||||||
|
/>
|
||||||
|
<VnInput
|
||||||
|
:label="t('property.book')"
|
||||||
|
v-model="data.book"
|
||||||
|
fill-input
|
||||||
|
/>
|
||||||
|
<VnInput
|
||||||
|
:label="t('property.page')"
|
||||||
|
v-model="data.page"
|
||||||
|
fill-input
|
||||||
|
/>
|
||||||
|
<VnInput
|
||||||
|
:label="t('property.farm')"
|
||||||
|
v-model="data.farm"
|
||||||
|
fill-input
|
||||||
|
/>
|
||||||
|
</VnRow>
|
||||||
|
<VnRow>
|
||||||
|
<VnInput
|
||||||
|
:label="t('property.registration')"
|
||||||
|
v-model="data.registration"
|
||||||
|
fill-input
|
||||||
|
/>
|
||||||
|
<VnInputDate
|
||||||
|
placeholder="dd-mm-aaa"
|
||||||
|
:label="t('property.booked')"
|
||||||
|
v-model="data.booked"
|
||||||
|
fill-input
|
||||||
|
/>
|
||||||
|
<VnInput
|
||||||
|
:label="t('property.volume')"
|
||||||
|
v-model="data.volume"
|
||||||
|
fill-input
|
||||||
|
/>
|
||||||
|
</VnRow>
|
||||||
|
</template>
|
||||||
|
</FormModel>
|
||||||
|
</div>
|
||||||
|
</template>
|
|
@ -0,0 +1,12 @@
|
||||||
|
<script setup>
|
||||||
|
import VnCard from 'components/common/VnCard.vue';
|
||||||
|
import PropertyDescriptor from 'pages/Property/Card/PropertyDescriptor.vue';
|
||||||
|
</script>
|
||||||
|
<template>
|
||||||
|
<VnCard
|
||||||
|
data-key="Property"
|
||||||
|
url="Properties"
|
||||||
|
:descriptor="PropertyDescriptor"
|
||||||
|
:filter="{ where: { id: $route.params.id } }"
|
||||||
|
/>
|
||||||
|
</template>
|
|
@ -0,0 +1,60 @@
|
||||||
|
<script setup>
|
||||||
|
import { computed } from 'vue';
|
||||||
|
import { useRoute } from 'vue-router';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import CardDescriptor from 'src/components/ui/CardDescriptor.vue';
|
||||||
|
import VnLv from 'src/components/ui/VnLv.vue';
|
||||||
|
import axios from 'axios';
|
||||||
|
import useNotify from 'src/composables/useNotify.js';
|
||||||
|
import PropertyCard from './PropertyCard.vue';
|
||||||
|
import SupplierDescriptorProxy from 'src/pages/Supplier/Card/SupplierDescriptorProxy.vue';
|
||||||
|
|
||||||
|
const { notify } = useNotify();
|
||||||
|
const { t } = useI18n();
|
||||||
|
const route = useRoute();
|
||||||
|
|
||||||
|
const entityId = computed(() => {
|
||||||
|
return Number(props.id || route.params.id);
|
||||||
|
});
|
||||||
|
|
||||||
|
const props = defineProps({
|
||||||
|
id: {
|
||||||
|
type: Number,
|
||||||
|
required: false,
|
||||||
|
default: null,
|
||||||
|
},
|
||||||
|
summary: {
|
||||||
|
type: Object,
|
||||||
|
default: null,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
<template>
|
||||||
|
<CardDescriptor
|
||||||
|
v-bind="$attrs"
|
||||||
|
:id="entityId"
|
||||||
|
:card="PropertyCard"
|
||||||
|
title="name"
|
||||||
|
module="Property"
|
||||||
|
>
|
||||||
|
<template #body="{ entity }">
|
||||||
|
<VnLv :label="$t('property.owner')" :value="entity.company.code" />
|
||||||
|
<VnLv
|
||||||
|
:label="$t('property.group')"
|
||||||
|
:value="entity.propertyGroup.description"
|
||||||
|
/>
|
||||||
|
<VnLv :label="$t('property.notary')">
|
||||||
|
<template #value>
|
||||||
|
<span class="link">
|
||||||
|
{{ entity?.supplier?.name || '-' }}
|
||||||
|
<SupplierDescriptorProxy :id="entity?.supplierFk" />
|
||||||
|
</span>
|
||||||
|
</template>
|
||||||
|
</VnLv>
|
||||||
|
<VnLv :label="$t('property.protocol')" :value="entity.protocol" copy />
|
||||||
|
<VnLv :label="$t('property.cadaster')" :value="entity.cadaster" copy />
|
||||||
|
<VnLv :label="$t('property.farm')" :value="entity.farm" />
|
||||||
|
<VnLv :label="$t('property.area')" :value="entity.area" />
|
||||||
|
</template>
|
||||||
|
</CardDescriptor>
|
||||||
|
</template>
|
|
@ -3,7 +3,7 @@ import { ref } from 'vue';
|
||||||
import { useI18n } from 'vue-i18n';
|
import { useI18n } from 'vue-i18n';
|
||||||
|
|
||||||
import VnDmsList from 'src/components/common/VnDmsList.vue';
|
import VnDmsList from 'src/components/common/VnDmsList.vue';
|
||||||
import VehicleDmsImportForm from 'src/pages/Route/Vehicle/Card/VehicleDmsImportForm.vue';
|
import PropertyDmsImportForm from 'src/pages/Property/Card/PropertyDmsImportForm.vue';
|
||||||
|
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
|
|
||||||
|
@ -15,15 +15,15 @@ const onDataSaved = () => dmsListRef.value.dmsRef.fetch();
|
||||||
<template>
|
<template>
|
||||||
<VnDmsList
|
<VnDmsList
|
||||||
ref="dmsListRef"
|
ref="dmsListRef"
|
||||||
model="VehicleDms"
|
model="PropertyDms"
|
||||||
update-model="vehicles"
|
update-model="PropertyDms"
|
||||||
delete-model="VehicleDms"
|
delete-model="PropertyDms"
|
||||||
download-model="dms"
|
download-model="dms"
|
||||||
default-dms-code="vehicles"
|
default-dms-code="property"
|
||||||
filter="vehicleFk"
|
filter="propertyFk"
|
||||||
/>
|
/>
|
||||||
<QDialog v-model="showImportDialog">
|
<QDialog v-model="showImportDialog">
|
||||||
<VehicleDmsImportForm @on-data-saved="onDataSaved()" />
|
<PropertyDmsImportForm @on-data-saved="onDataSaved()" />
|
||||||
</QDialog>
|
</QDialog>
|
||||||
<QPageSticky position="bottom-right" :offset="[25, 90]">
|
<QPageSticky position="bottom-right" :offset="[25, 90]">
|
||||||
<QBtn
|
<QBtn
|
|
@ -6,8 +6,8 @@ import { useRoute } from 'vue-router';
|
||||||
import VnSelect from 'src/components/common/VnSelect.vue';
|
import VnSelect from 'src/components/common/VnSelect.vue';
|
||||||
import FormModelPopup from 'components/FormModelPopup.vue';
|
import FormModelPopup from 'components/FormModelPopup.vue';
|
||||||
import FetchData from 'components/FetchData.vue';
|
import FetchData from 'components/FetchData.vue';
|
||||||
import useNotify from 'src/composables/useNotify.js';
|
|
||||||
|
|
||||||
|
import useNotify from 'src/composables/useNotify.js';
|
||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
|
|
||||||
const emit = defineEmits(['onDataSaved']);
|
const emit = defineEmits(['onDataSaved']);
|
||||||
|
@ -21,14 +21,14 @@ const dmsId = ref(null);
|
||||||
|
|
||||||
const importDms = async () => {
|
const importDms = async () => {
|
||||||
try {
|
try {
|
||||||
if (!dmsId.value) throw new Error(t(`vehicle.errors.documentIdEmpty`));
|
if (!dmsId.value) throw new Error(t(`globals.errors.documentIdEmpty`));
|
||||||
|
|
||||||
const data = {
|
const data = {
|
||||||
vehicleFk: route.params.id,
|
propertyFk: route.params.id,
|
||||||
dmsFk: dmsId.value,
|
dmsFk: dmsId.value,
|
||||||
};
|
};
|
||||||
|
|
||||||
await axios.post('vehicleDms', data);
|
await axios.post('propertyDms', data);
|
||||||
notify(t('globals.dataSaved'), 'positive');
|
notify(t('globals.dataSaved'), 'positive');
|
||||||
dmsId.value = null;
|
dmsId.value = null;
|
||||||
emit('onDataSaved');
|
emit('onDataSaved');
|
|
@ -0,0 +1,6 @@
|
||||||
|
<script setup>
|
||||||
|
import VnLog from 'src/components/common/VnLog.vue';
|
||||||
|
</script>
|
||||||
|
<template>
|
||||||
|
<VnLog model="Property" url="/PropertyLogs" />
|
||||||
|
</template>
|
|
@ -0,0 +1,33 @@
|
||||||
|
<script setup>
|
||||||
|
import { computed } from 'vue';
|
||||||
|
import { useRoute } from 'vue-router';
|
||||||
|
import { useState } from 'src/composables/useState';
|
||||||
|
import VnNotes from 'src/components/ui/VnNotes.vue';
|
||||||
|
|
||||||
|
const route = useRoute();
|
||||||
|
const state = useState();
|
||||||
|
const user = state.getUser();
|
||||||
|
const propertyId = computed(() => route.params.id);
|
||||||
|
|
||||||
|
const noteFilter = computed(() => {
|
||||||
|
return {
|
||||||
|
order: 'created DESC',
|
||||||
|
where: { propertyFk: propertyId.value },
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
const body = {
|
||||||
|
propertyFk: propertyId.value,
|
||||||
|
workerFk: user.value.id,
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<VnNotes
|
||||||
|
url="propertyObservations"
|
||||||
|
:add-note="true"
|
||||||
|
:filter="noteFilter"
|
||||||
|
:body="body"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</template>
|
|
@ -0,0 +1,349 @@
|
||||||
|
<script setup>
|
||||||
|
import { onMounted, ref, computed } from 'vue';
|
||||||
|
import { useRoute } from 'vue-router';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import { dashIfEmpty, toDate, toCurrency } from 'src/filters';
|
||||||
|
import CardSummary from 'components/ui/CardSummary.vue';
|
||||||
|
import VnLv from 'src/components/ui/VnLv.vue';
|
||||||
|
import { getUrl } from 'src/composables/getUrl';
|
||||||
|
import useNotify from 'src/composables/useNotify.js';
|
||||||
|
import { useArrayData } from 'composables/useArrayData';
|
||||||
|
import VnTitle from 'src/components/common/VnTitle.vue';
|
||||||
|
import VnToSummary from 'src/components/ui/VnToSummary.vue';
|
||||||
|
import FetchData from 'src/components/FetchData.vue';
|
||||||
|
import VnAvatar from 'src/components/ui/VnAvatar.vue';
|
||||||
|
import VnUserLink from 'src/components/ui/VnUserLink.vue';
|
||||||
|
import VnCheckbox from 'src/components/common/VnCheckbox.vue';
|
||||||
|
import WorkerDescriptorProxy from 'src/pages/Worker/Card/WorkerDescriptorProxy.vue';
|
||||||
|
import SupplierDescriptorProxy from 'src/pages/Supplier/Card/SupplierDescriptorProxy.vue';
|
||||||
|
|
||||||
|
const route = useRoute();
|
||||||
|
const { notify } = useNotify();
|
||||||
|
const { t } = useI18n();
|
||||||
|
|
||||||
|
const $props = defineProps({
|
||||||
|
id: {
|
||||||
|
type: Number,
|
||||||
|
default: null,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const dmsColumns = ref([
|
||||||
|
{
|
||||||
|
align: 'left',
|
||||||
|
label: t('globals.id'),
|
||||||
|
name: 'id',
|
||||||
|
field: ({ id }) => id,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
align: 'left',
|
||||||
|
label: t('globals.type'),
|
||||||
|
name: 'type',
|
||||||
|
field: ({ type }) => type?.name,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
align: 'left',
|
||||||
|
label: t('globals.order'),
|
||||||
|
name: 'order',
|
||||||
|
field: ({ hardCopyNumber }) => dashIfEmpty(hardCopyNumber),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
align: 'left',
|
||||||
|
label: t('globals.reference'),
|
||||||
|
name: 'reference',
|
||||||
|
field: ({ reference }) => dashIfEmpty(reference),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
align: 'left',
|
||||||
|
label: t('globals.description'),
|
||||||
|
name: 'description',
|
||||||
|
field: ({ description }) => dashIfEmpty(description),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
align: 'left',
|
||||||
|
label: t('globals.original'),
|
||||||
|
name: 'hasFile',
|
||||||
|
toolTip: t('The documentation is available in paper form'),
|
||||||
|
component: 'checkbox',
|
||||||
|
field: ({ hasFile }) => hasFile,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
align: 'left',
|
||||||
|
label: t('globals.worker'),
|
||||||
|
name: 'worker',
|
||||||
|
field: ({ worker }) => worker?.name,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
align: 'left',
|
||||||
|
label: t('globals.created'),
|
||||||
|
name: 'created',
|
||||||
|
field: ({ created }) => toDate(created),
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
const entityId = computed(() => $props.id || route.params.id);
|
||||||
|
|
||||||
|
const summary = ref();
|
||||||
|
const property = computed(() => summary.value?.entity);
|
||||||
|
const propertyUrl = ref();
|
||||||
|
const propertyDms = ref(null);
|
||||||
|
|
||||||
|
const descriptorData = useArrayData('Property');
|
||||||
|
|
||||||
|
onMounted(async () => {
|
||||||
|
propertyUrl.value = (await getUrl('property/')) + entityId.value + '/';
|
||||||
|
});
|
||||||
|
|
||||||
|
function toPropertyUrl(section) {
|
||||||
|
return '#/property/' + entityId.value + '/' + section;
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<FetchData
|
||||||
|
ref="propertyDms"
|
||||||
|
:url="`Properties/${entityId}/dms`"
|
||||||
|
@on-fetch="
|
||||||
|
(data) => {
|
||||||
|
propertyDms.value = data;
|
||||||
|
}
|
||||||
|
"
|
||||||
|
/>
|
||||||
|
<CardSummary
|
||||||
|
ref="summary"
|
||||||
|
:url="`Properties/${entityId}/summary`"
|
||||||
|
data-key="PropertySummary"
|
||||||
|
v-bind="$attrs.width"
|
||||||
|
>
|
||||||
|
<template #header-left>
|
||||||
|
<VnToSummary
|
||||||
|
v-if="route?.name !== 'PropertySummary'"
|
||||||
|
:route-name="'PropertySummary'"
|
||||||
|
:entity-id="entityId"
|
||||||
|
:url="PropertyUrl"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
<template #header="{ entity }">
|
||||||
|
<div>Property #{{ entity.id }} - {{ entity.name }}</div>
|
||||||
|
</template>
|
||||||
|
<template #menu="{ entity }">
|
||||||
|
<!-- <PropertyDescriptorMenu :ticket="entity" /> -->
|
||||||
|
</template>
|
||||||
|
<template #body="{ entity }">
|
||||||
|
<QCard class="vn-one">
|
||||||
|
<VnTitle
|
||||||
|
:url="toPropertyUrl('basic-data')"
|
||||||
|
:text="t('globals.summary.basicData')"
|
||||||
|
data-cy="titleBasicDataBlock1"
|
||||||
|
/>
|
||||||
|
<div class="vn-card-group">
|
||||||
|
<div class="vn-card-content">
|
||||||
|
<VnLv
|
||||||
|
:label="t('property.owner')"
|
||||||
|
:value="dashIfEmpty(entity.company.code)"
|
||||||
|
/>
|
||||||
|
<VnLv
|
||||||
|
:label="t('property.map')"
|
||||||
|
:value="dashIfEmpty(entity.url)"
|
||||||
|
copy
|
||||||
|
>
|
||||||
|
<template #value>
|
||||||
|
<a
|
||||||
|
v-if="entity?.url"
|
||||||
|
:href="`${entity?.url}`"
|
||||||
|
target="_blank"
|
||||||
|
class="grafana"
|
||||||
|
data-cy="propertyMapLink"
|
||||||
|
>
|
||||||
|
{{ t('property.goToMap') }}
|
||||||
|
</a>
|
||||||
|
</template>
|
||||||
|
</VnLv>
|
||||||
|
<VnLv
|
||||||
|
:label="t('property.value')"
|
||||||
|
:value="toCurrency(entity.value)"
|
||||||
|
/>
|
||||||
|
<VnLv :label="$t('property.notary')">
|
||||||
|
<template #value>
|
||||||
|
<span class="link">
|
||||||
|
{{ entity?.supplier?.name || '-' }}
|
||||||
|
<SupplierDescriptorProxy :id="entity?.supplierFk" />
|
||||||
|
</span>
|
||||||
|
</template>
|
||||||
|
</VnLv>
|
||||||
|
<VnLv
|
||||||
|
:label="t('property.protocol')"
|
||||||
|
:value="dashIfEmpty(entity.protocol)"
|
||||||
|
copy
|
||||||
|
/>
|
||||||
|
<VnLv
|
||||||
|
:label="t('property.purchased')"
|
||||||
|
:value="toDate(entity.purchased)"
|
||||||
|
/>
|
||||||
|
<VnLv
|
||||||
|
:label="t('property.booked')"
|
||||||
|
:value="toDate(entity.booked)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</QCard>
|
||||||
|
<QCard class="vn-one">
|
||||||
|
<VnTitle
|
||||||
|
:url="toPropertyUrl('basic-data')"
|
||||||
|
:text="t('globals.summary.basicData')"
|
||||||
|
data-cy="titleBasicDataBlock2"
|
||||||
|
/>
|
||||||
|
<div class="vn-card-content">
|
||||||
|
<VnLv
|
||||||
|
:label="t('property.cadaster')"
|
||||||
|
:value="dashIfEmpty(entity.cadaster)"
|
||||||
|
copy
|
||||||
|
/>
|
||||||
|
<VnLv
|
||||||
|
:label="t('property.smallHolding')"
|
||||||
|
:value="dashIfEmpty(entity.smallHolding)"
|
||||||
|
/>
|
||||||
|
<VnLv :label="t('property.area')" :value="dashIfEmpty(entity.area)" />
|
||||||
|
<VnLv
|
||||||
|
:label="t('property.allocation')"
|
||||||
|
:value="dashIfEmpty(entity.allocation)"
|
||||||
|
/>
|
||||||
|
<VnLv
|
||||||
|
:label="t('property.town')"
|
||||||
|
:value="dashIfEmpty(entity.town.name)"
|
||||||
|
/>
|
||||||
|
<VnLv :label="t('property.m2')" :value="dashIfEmpty(entity.m2)" />
|
||||||
|
</div>
|
||||||
|
</QCard>
|
||||||
|
<QCard class="vn-one">
|
||||||
|
<VnTitle
|
||||||
|
:url="toPropertyUrl('basic-data')"
|
||||||
|
:text="t('globals.summary.basicData')"
|
||||||
|
data-cy="titleBasicDataBlock3"
|
||||||
|
/>
|
||||||
|
<div class="vn-card-content">
|
||||||
|
<VnLv
|
||||||
|
:label="t('property.registry')"
|
||||||
|
:value="dashIfEmpty(entity.registry)"
|
||||||
|
/>
|
||||||
|
<VnLv :label="t('property.tome')" :value="dashIfEmpty(entity.tome)" />
|
||||||
|
<VnLv :label="t('property.book')" :value="dashIfEmpty(entity.book)" />
|
||||||
|
<VnLv :label="t('property.page')" :value="dashIfEmpty(entity.page)" />
|
||||||
|
<VnLv :label="t('property.farm')" :value="dashIfEmpty(entity.farm)" />
|
||||||
|
<VnLv
|
||||||
|
:label="t('property.registration')"
|
||||||
|
:value="dashIfEmpty(entity.registration)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</QCard>
|
||||||
|
<QCard v-if="entity?.propertyDms.length > 0" class="vn-max">
|
||||||
|
<VnTitle
|
||||||
|
:url="toPropertyUrl('dms')"
|
||||||
|
:text="t('globals.pageTitles.dms')"
|
||||||
|
data-cy="titleDmsBlock"
|
||||||
|
/>
|
||||||
|
<QTable :columns="dmsColumns" :rows="entity?.propertyDms" flat>
|
||||||
|
<template #header="props">
|
||||||
|
<QTr :props="props">
|
||||||
|
<QTh auto-width class="text-left">{{ t('globals.id') }}</QTh>
|
||||||
|
<QTh auto-width class="text-left">{{
|
||||||
|
t('globals.type')
|
||||||
|
}}</QTh>
|
||||||
|
<QTh auto-width class="text-left">{{
|
||||||
|
t('globals.order')
|
||||||
|
}}</QTh>
|
||||||
|
<QTh auto-width class="text-left">{{
|
||||||
|
t('globals.reference')
|
||||||
|
}}</QTh>
|
||||||
|
<QTh auto-width class="text-left">{{
|
||||||
|
t('globals.description')
|
||||||
|
}}</QTh>
|
||||||
|
<QTh auto-width class="text-center">{{
|
||||||
|
t('globals.original')
|
||||||
|
}}</QTh>
|
||||||
|
<QTh auto-width class="text-left">{{
|
||||||
|
t('globals.worker')
|
||||||
|
}}</QTh>
|
||||||
|
<QTh auto-width class="text-center">{{
|
||||||
|
t('globals.created')
|
||||||
|
}}</QTh>
|
||||||
|
</QTr>
|
||||||
|
</template>
|
||||||
|
<template #body="props">
|
||||||
|
<QTr :props="props">
|
||||||
|
<QTd class="text-left">{{ props.row.dms.id }}</QTd>
|
||||||
|
<QTd class="text-left">{{ props.row.dms.dmsType.name }}</QTd>
|
||||||
|
<QTd class="text-left">{{
|
||||||
|
props.row.dms.hardCopyNumber
|
||||||
|
}}</QTd>
|
||||||
|
<QTd class="text-left">{{ props.row.dms.reference }}</QTd>
|
||||||
|
<QTd class="text-left">{{ props.row.dms.description }}</QTd>
|
||||||
|
<QTd class="text-center"
|
||||||
|
><VnCheckbox
|
||||||
|
:disable="true"
|
||||||
|
v-model="props.row.dms.hasFile"
|
||||||
|
/></QTd>
|
||||||
|
<QTd class="text-left"
|
||||||
|
><span class="link" @click.stop
|
||||||
|
>{{ props.row.dms.worker.firstName
|
||||||
|
}}<WorkerDescriptorProxy
|
||||||
|
:id="props.row.dms.worker.id" /></span
|
||||||
|
></QTd>
|
||||||
|
<QTd class="text-center">{{
|
||||||
|
toDate(props.row.dms.created)
|
||||||
|
}}</QTd>
|
||||||
|
</QTr>
|
||||||
|
</template>
|
||||||
|
</QTable>
|
||||||
|
</QCard>
|
||||||
|
<QCard v-if="entity?.notes.length > 0" class="vn-max">
|
||||||
|
<VnTitle
|
||||||
|
:url="toPropertyUrl('notes')"
|
||||||
|
:text="t('globals.notes')"
|
||||||
|
data-cy="titleNotesBlock"
|
||||||
|
/>
|
||||||
|
<QCardSection
|
||||||
|
v-if="entity?.notes"
|
||||||
|
v-for="note in entity?.notes"
|
||||||
|
horizontal
|
||||||
|
class="q-pb-sm"
|
||||||
|
>
|
||||||
|
<VnAvatar
|
||||||
|
:descriptor="false"
|
||||||
|
:worker-id="note.workerFk"
|
||||||
|
size="md"
|
||||||
|
:title="note.worker?.user.nickname"
|
||||||
|
class="q-pr-xs"
|
||||||
|
/>
|
||||||
|
<VnUserLink
|
||||||
|
:name="`${note.worker.user.name}`"
|
||||||
|
:worker-id="note.worker.id"
|
||||||
|
/>
|
||||||
|
<span class="q-pr-xs">{{ ':' }}</span>
|
||||||
|
<span class="no-margin">
|
||||||
|
{{ note.text }}
|
||||||
|
</span>
|
||||||
|
</QCardSection>
|
||||||
|
</QCard>
|
||||||
|
</template>
|
||||||
|
</CardSummary>
|
||||||
|
</template>
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.q-card.q-card--dark.q-dark.vn-one {
|
||||||
|
& > .bodyCard {
|
||||||
|
padding: 1%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.q-table {
|
||||||
|
tr,
|
||||||
|
th,
|
||||||
|
.q-td {
|
||||||
|
border-bottom: 1px solid black;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.grafana {
|
||||||
|
color: $primary-light;
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -0,0 +1,182 @@
|
||||||
|
<script setup>
|
||||||
|
import { ref } from 'vue';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
|
||||||
|
import FetchData from 'components/FetchData.vue';
|
||||||
|
import VnFilterPanel from 'src/components/ui/VnFilterPanel.vue';
|
||||||
|
import VnInput from 'src/components/common/VnInput.vue';
|
||||||
|
import VnInputDate from 'components/common/VnInputDate.vue';
|
||||||
|
import VnSelect from 'src/components/common/VnSelect.vue';
|
||||||
|
import VnCheckbox from 'src/components/common/VnCheckbox.vue';
|
||||||
|
|
||||||
|
const { t } = useI18n();
|
||||||
|
const props = defineProps({
|
||||||
|
dataKey: {
|
||||||
|
type: String,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<VnFilterPanel :data-key="props.dataKey" :search-button="true">
|
||||||
|
<template #tags="{ tag, formatFn }">
|
||||||
|
<div class="q-gutter-x-xs">
|
||||||
|
<strong>{{ t(`property.params.${tag.label}`) }}: </strong>
|
||||||
|
<span>{{ formatFn(tag.value) }}</span>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
<template #body="{ params, searchFn }">
|
||||||
|
<QItem>
|
||||||
|
<QItemSection>
|
||||||
|
<VnInput
|
||||||
|
v-model="params.id"
|
||||||
|
:label="t('property.id')"
|
||||||
|
dense
|
||||||
|
filled
|
||||||
|
data-cy="idInput"
|
||||||
|
/>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
<QItem>
|
||||||
|
<QItemSection>
|
||||||
|
<VnInput
|
||||||
|
v-model="params.name"
|
||||||
|
:label="t('property.name')"
|
||||||
|
dense
|
||||||
|
filled
|
||||||
|
data-cy="nameInput"
|
||||||
|
/>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
<QItem>
|
||||||
|
<QItemSection>
|
||||||
|
<VnSelect
|
||||||
|
url="PropertyGroups"
|
||||||
|
:label="t('property.group')"
|
||||||
|
v-model="params.propertyGroupFk"
|
||||||
|
option-value="id"
|
||||||
|
option-label="description"
|
||||||
|
dense
|
||||||
|
filled
|
||||||
|
data-cy="propertyGroupSelect"
|
||||||
|
/>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
<QItem>
|
||||||
|
<QItemSection>
|
||||||
|
<VnSelect
|
||||||
|
url="Suppliers"
|
||||||
|
:label="t('property.notary')"
|
||||||
|
v-model="params.supplierFk"
|
||||||
|
option-value="id"
|
||||||
|
option-label="name"
|
||||||
|
dense
|
||||||
|
filled
|
||||||
|
data-cy="notarySelect"
|
||||||
|
/>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
<QItem>
|
||||||
|
<QItemSection>
|
||||||
|
<VnInput
|
||||||
|
v-model="params.protocol"
|
||||||
|
:label="t('property.protocol')"
|
||||||
|
dense
|
||||||
|
filled
|
||||||
|
data-cy="protocolInput"
|
||||||
|
/>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
<QItem>
|
||||||
|
<QItemSection>
|
||||||
|
<VnInput
|
||||||
|
v-model="params.cadaster"
|
||||||
|
:label="t('property.cadaster')"
|
||||||
|
filled
|
||||||
|
dense
|
||||||
|
data-cy="cadasterInput"
|
||||||
|
/>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
<QItem>
|
||||||
|
<QItemSection>
|
||||||
|
<VnSelect
|
||||||
|
url="Companies"
|
||||||
|
:label="t('property.owner')"
|
||||||
|
v-model="params.companyFk"
|
||||||
|
option-value="id"
|
||||||
|
option-label="code"
|
||||||
|
dense
|
||||||
|
filled
|
||||||
|
data-cy="ownerSelect"
|
||||||
|
/>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
<QItem>
|
||||||
|
<QItemSection>
|
||||||
|
<VnInputDate
|
||||||
|
v-model="params.purchased"
|
||||||
|
:label="t('property.purchased')"
|
||||||
|
dense
|
||||||
|
filled
|
||||||
|
data-cy="purchasedDateInput"
|
||||||
|
/>
|
||||||
|
</QItemSection>
|
||||||
|
<QItemSection>
|
||||||
|
<VnInputDate
|
||||||
|
v-model="params.booked"
|
||||||
|
:label="t('property.booked')"
|
||||||
|
dense
|
||||||
|
filled
|
||||||
|
data-cy="bookedDateInput"
|
||||||
|
/>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
<QItem>
|
||||||
|
<QItemSection>
|
||||||
|
<VnCheckbox
|
||||||
|
v-model="params.hasFormalization"
|
||||||
|
:label="t('property.hasFormalization')"
|
||||||
|
dense
|
||||||
|
filled
|
||||||
|
data-cy="hasFormalizationCheckbox"
|
||||||
|
/>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
<QItem>
|
||||||
|
<QItemSection>
|
||||||
|
<VnCheckbox
|
||||||
|
v-model="params.hasSimpleCopy"
|
||||||
|
:label="t('property.hasSimpleCopy')"
|
||||||
|
dense
|
||||||
|
filled
|
||||||
|
data-cy="hasSimpleCopyCheckbox"
|
||||||
|
/>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
<QItem>
|
||||||
|
<QItemSection>
|
||||||
|
<VnCheckbox
|
||||||
|
v-model="params.hasOriginalPropertyDeed"
|
||||||
|
:label="t('property.hasOriginalPropertyDeed')"
|
||||||
|
dense
|
||||||
|
filled
|
||||||
|
data-cy="hasOriginalPropertyDeedCheckbox"
|
||||||
|
/>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
<QItem>
|
||||||
|
<QItemSection>
|
||||||
|
<VnCheckbox
|
||||||
|
v-model="params.hasFundProvision"
|
||||||
|
:label="t('property.hasFundProvision')"
|
||||||
|
dense
|
||||||
|
filled
|
||||||
|
data-cy="hasFundProvisionCheckbox"
|
||||||
|
/>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
</template>
|
||||||
|
</VnFilterPanel>
|
||||||
|
</template>
|
|
@ -0,0 +1,307 @@
|
||||||
|
<script setup>
|
||||||
|
import { computed, ref } from 'vue';
|
||||||
|
import { toDate } from 'src/filters/index';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import VnTable from 'src/components/VnTable/VnTable.vue';
|
||||||
|
import VnSection from 'src/components/common/VnSection.vue';
|
||||||
|
import PropertyFilter from './PropertyFilter.vue';
|
||||||
|
import VnInput from 'src/components/common/VnInput.vue';
|
||||||
|
import VnSelect from 'src/components/common/VnSelect.vue';
|
||||||
|
import FetchData from 'src/components/FetchData.vue';
|
||||||
|
import VnInputDate from 'src/components/common/VnInputDate.vue';
|
||||||
|
import VnRow from 'src/components/ui/VnRow.vue';
|
||||||
|
import SupplierDescriptorProxy from '../Supplier/Card/SupplierDescriptorProxy.vue';
|
||||||
|
|
||||||
|
const { t } = useI18n();
|
||||||
|
const tableRef = ref();
|
||||||
|
|
||||||
|
const dataKey = 'PropertyList';
|
||||||
|
|
||||||
|
const columns = computed(() => [
|
||||||
|
{
|
||||||
|
align: 'right',
|
||||||
|
name: 'id',
|
||||||
|
label: t('property.id'),
|
||||||
|
width: '35px',
|
||||||
|
chip: {
|
||||||
|
condition: () => true,
|
||||||
|
},
|
||||||
|
isId: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
align: 'left',
|
||||||
|
name: 'name',
|
||||||
|
label: t('property.name'),
|
||||||
|
isTitle: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
align: 'left',
|
||||||
|
name: 'propertyGroupFk',
|
||||||
|
label: t('property.group'),
|
||||||
|
component: 'select',
|
||||||
|
attrs: {
|
||||||
|
url: 'propertyGroups',
|
||||||
|
fields: ['id', 'description'],
|
||||||
|
optionValue: 'id',
|
||||||
|
optionLabel: 'description',
|
||||||
|
hideSelected: true,
|
||||||
|
},
|
||||||
|
format: ({ propertyGroup }) => propertyGroup,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: t('property.group'),
|
||||||
|
name: 'propertyGroup',
|
||||||
|
cardVisible: true,
|
||||||
|
visible: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
align: 'left',
|
||||||
|
name: 'notary',
|
||||||
|
label: t('property.notary'),
|
||||||
|
cardVisible: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
align: 'left',
|
||||||
|
name: 'protocol',
|
||||||
|
label: t('property.protocol'),
|
||||||
|
cardVisible: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
align: 'left',
|
||||||
|
name: 'cadaster',
|
||||||
|
label: t('property.cadaster'),
|
||||||
|
cardVisible: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
align: 'left',
|
||||||
|
name: 'companyFk',
|
||||||
|
label: t('property.owner'),
|
||||||
|
component: 'select',
|
||||||
|
attrs: {
|
||||||
|
url: 'Companies',
|
||||||
|
fields: ['id', 'code'],
|
||||||
|
optionValue: 'id',
|
||||||
|
optionLabel: 'code',
|
||||||
|
hideSelected: true,
|
||||||
|
},
|
||||||
|
format: ({ companyCode }) => companyCode,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: t('property.owner'),
|
||||||
|
name: 'companyCode',
|
||||||
|
cardVisible: true,
|
||||||
|
visible: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
align: 'left',
|
||||||
|
name: 'purchased',
|
||||||
|
label: t('property.purchased'),
|
||||||
|
component: 'date',
|
||||||
|
format: ({ purchased }) => toDate(purchased),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: t('property.purchased'),
|
||||||
|
name: 'purchased',
|
||||||
|
format: ({ purchased }) => toDate(purchased),
|
||||||
|
cardVisible: true,
|
||||||
|
visible: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
align: 'left',
|
||||||
|
name: 'booked',
|
||||||
|
label: t('property.booked'),
|
||||||
|
component: 'date',
|
||||||
|
format: ({ booked }) => toDate(booked),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: t('property.booked'),
|
||||||
|
name: 'booked',
|
||||||
|
format: ({ booked }) => toDate(booked),
|
||||||
|
cardVisible: true,
|
||||||
|
visible: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
labelAbbreviation: 'Fo',
|
||||||
|
label: t('property.hasFormalization'),
|
||||||
|
toolTip: t('property.hasFormalization'),
|
||||||
|
name: 'hasFormalization',
|
||||||
|
component: 'checkbox',
|
||||||
|
width: '35px',
|
||||||
|
cardVisible: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
labelAbbreviation: 'SC',
|
||||||
|
label: t('property.hasSimpleCopy'),
|
||||||
|
toolTip: t('property.hasSimpleCopy'),
|
||||||
|
name: 'hasSimpleCopy',
|
||||||
|
component: 'checkbox',
|
||||||
|
width: '35px',
|
||||||
|
cardVisible: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
labelAbbreviation: 'OP',
|
||||||
|
label: t('property.hasOriginalPropertyDeed'),
|
||||||
|
toolTip: t('property.hasOriginalPropertyDeed'),
|
||||||
|
name: 'hasOriginalPropertyDeed',
|
||||||
|
component: 'checkbox',
|
||||||
|
width: '35px',
|
||||||
|
cardVisible: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
labelAbbreviation: 'FP',
|
||||||
|
label: t('property.hasFundProvision'),
|
||||||
|
toolTip: t('property.hasFundProvision'),
|
||||||
|
name: 'hasFundProvision',
|
||||||
|
component: 'checkbox',
|
||||||
|
width: '35px',
|
||||||
|
cardVisible: true,
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<VnSection
|
||||||
|
:data-key="dataKey"
|
||||||
|
:columns="columns"
|
||||||
|
prefix="property"
|
||||||
|
:array-data-props="{
|
||||||
|
url: 'Properties/filter',
|
||||||
|
}"
|
||||||
|
>
|
||||||
|
<template #advanced-menu>
|
||||||
|
<PropertyFilter :data-key="dataKey" />
|
||||||
|
</template>
|
||||||
|
<template #body>
|
||||||
|
<VnTable
|
||||||
|
ref="tableRef"
|
||||||
|
:data-key="dataKey"
|
||||||
|
:columns="columns"
|
||||||
|
redirect="property"
|
||||||
|
search-url="PropertyList"
|
||||||
|
:create="{
|
||||||
|
urlCreate: 'Properties',
|
||||||
|
title: t('property.createProperty'),
|
||||||
|
onDataSaved: ({ id }) => tableRef.redirect(id),
|
||||||
|
formInitialData: {},
|
||||||
|
}"
|
||||||
|
:with-filters="false"
|
||||||
|
:right-search="false"
|
||||||
|
>
|
||||||
|
<template #column-name="{ row }">
|
||||||
|
<span>
|
||||||
|
{{ row.name }}
|
||||||
|
<QTooltip>
|
||||||
|
{{ row.name }}
|
||||||
|
</QTooltip>
|
||||||
|
</span>
|
||||||
|
</template>
|
||||||
|
<template #column-notary="{ row }">
|
||||||
|
<span class="link" @click.stop>
|
||||||
|
{{ row.notary }}
|
||||||
|
<SupplierDescriptorProxy :id="row.supplierFk" />
|
||||||
|
</span>
|
||||||
|
</template>
|
||||||
|
<template #more-create-dialog="{ data }">
|
||||||
|
<div class="col-span-2">
|
||||||
|
<VnRow>
|
||||||
|
<VnInput v-model="data.name" :label="$t('property.name')" />
|
||||||
|
</VnRow>
|
||||||
|
<VnRow>
|
||||||
|
<VnSelect
|
||||||
|
url="Suppliers"
|
||||||
|
v-model="data.supplierFk"
|
||||||
|
:label="$t('property.notary')"
|
||||||
|
:fields="['id', 'name']"
|
||||||
|
hide-selected
|
||||||
|
/>
|
||||||
|
<VnInput
|
||||||
|
v-model="data.protocol"
|
||||||
|
:label="$t('property.protocol')"
|
||||||
|
/>
|
||||||
|
</VnRow>
|
||||||
|
<VnRow>
|
||||||
|
<VnSelect
|
||||||
|
url="Companies"
|
||||||
|
:fields="['id', 'code']"
|
||||||
|
v-model="data.companyFk"
|
||||||
|
:label="$t('property.owner')"
|
||||||
|
option-value="id"
|
||||||
|
option-label="code"
|
||||||
|
hide-selected
|
||||||
|
/>
|
||||||
|
<VnSelect
|
||||||
|
url="PropertyGroups"
|
||||||
|
:fields="['id', 'description']"
|
||||||
|
v-model="data.propertyGroupFk"
|
||||||
|
:label="$t('property.group')"
|
||||||
|
option-value="id"
|
||||||
|
option-label="description"
|
||||||
|
hide-selected
|
||||||
|
/>
|
||||||
|
</VnRow>
|
||||||
|
<VnRow>
|
||||||
|
<VnInput v-model="data.url" :label="$t('property.map')" />
|
||||||
|
</VnRow>
|
||||||
|
<VnRow>
|
||||||
|
<VnInput v-model="data.value" :label="$t('property.value')" />
|
||||||
|
<VnInputDate
|
||||||
|
placeholder="dd-mm-aaa"
|
||||||
|
:label="$t('property.purchased')"
|
||||||
|
v-model="data.purchased"
|
||||||
|
/>
|
||||||
|
<VnInputDate
|
||||||
|
placeholder="dd-mm-aaa"
|
||||||
|
:label="$t('property.booked')"
|
||||||
|
v-model="data.booked"
|
||||||
|
/>
|
||||||
|
</VnRow>
|
||||||
|
<VnRow>
|
||||||
|
<VnInput
|
||||||
|
v-model="data.cadaster"
|
||||||
|
:label="$t('property.cadaster')"
|
||||||
|
/>
|
||||||
|
</VnRow>
|
||||||
|
<VnRow>
|
||||||
|
<VnInput
|
||||||
|
v-model="data.smallHolding"
|
||||||
|
:label="$t('property.smallHolding')"
|
||||||
|
/>
|
||||||
|
<VnInput v-model="data.area" :label="$t('property.area')" />
|
||||||
|
<VnInput
|
||||||
|
v-model="data.allocation"
|
||||||
|
:label="$t('property.allocation')"
|
||||||
|
/>
|
||||||
|
</VnRow>
|
||||||
|
<VnRow>
|
||||||
|
<VnSelect
|
||||||
|
url="Towns"
|
||||||
|
v-model="data.townFk"
|
||||||
|
:label="$t('property.town')"
|
||||||
|
:fields="['id', 'name']"
|
||||||
|
:options="towns"
|
||||||
|
option-value="id"
|
||||||
|
option-label="name"
|
||||||
|
hide-selected
|
||||||
|
/>
|
||||||
|
<VnInput v-model="data.m2" :label="$t('property.m2')" />
|
||||||
|
<VnInput
|
||||||
|
v-model="data.registry"
|
||||||
|
:label="$t('property.registry')"
|
||||||
|
/>
|
||||||
|
</VnRow>
|
||||||
|
<VnRow>
|
||||||
|
<VnInput v-model="data.tome" :label="$t('property.tome')" />
|
||||||
|
<VnInput v-model="data.book" :label="$t('property.book')" />
|
||||||
|
<VnInput v-model="data.page" :label="$t('property.page')" />
|
||||||
|
<VnInput v-model="data.farm" :label="$t('property.farm')" />
|
||||||
|
</VnRow>
|
||||||
|
<VnInput
|
||||||
|
v-model="data.registration"
|
||||||
|
:label="$t('property.registration')"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</VnTable>
|
||||||
|
</template>
|
||||||
|
</VnSection>
|
||||||
|
</template>
|
|
@ -0,0 +1,47 @@
|
||||||
|
property:
|
||||||
|
search: Search property
|
||||||
|
searchInfo: Search property by id or name
|
||||||
|
id: Id
|
||||||
|
name: Name
|
||||||
|
cadaster: Cadaster
|
||||||
|
smallHolding: Small holding
|
||||||
|
area: Area
|
||||||
|
town: Town
|
||||||
|
registry: Registry
|
||||||
|
volume: Volume
|
||||||
|
book: Book
|
||||||
|
page: Page
|
||||||
|
farm: Farm
|
||||||
|
registration: Registration
|
||||||
|
value: Value
|
||||||
|
group: Group
|
||||||
|
m2: m2
|
||||||
|
owner: Owner
|
||||||
|
allocation: Allocation
|
||||||
|
tome: Tome
|
||||||
|
map: Map
|
||||||
|
notary: Notary
|
||||||
|
protocol: Notarial protocol
|
||||||
|
purchased: Purchased date
|
||||||
|
booked: Booked date
|
||||||
|
createProperty: Create property
|
||||||
|
goToMap: Link to map
|
||||||
|
hasFormalization: Formalization
|
||||||
|
hasSimpleCopy: Simple copy
|
||||||
|
hasOriginalPropertyDeed: Original property deed
|
||||||
|
hasFundProvision: Fund provision
|
||||||
|
params:
|
||||||
|
search: General search
|
||||||
|
id: Id
|
||||||
|
name: Name
|
||||||
|
supplierFk: Notary
|
||||||
|
propertyGroupFk: Group
|
||||||
|
protocol: Notarial protocol
|
||||||
|
companyFk: Owner
|
||||||
|
purchased: Purchased date
|
||||||
|
booked: Booked date
|
||||||
|
cadaster: Cadaster
|
||||||
|
hasFormalization: Formalization
|
||||||
|
hasSimpleCopy: Simple copy
|
||||||
|
hasOriginalPropertyDeed: Original property deed
|
||||||
|
hasFundProvision: Fund provision
|
|
@ -0,0 +1,47 @@
|
||||||
|
property:
|
||||||
|
search: Buscar propiedad
|
||||||
|
searchInfo: Buscar propiedad por id o nombre
|
||||||
|
id: Id
|
||||||
|
name: Nombre
|
||||||
|
cadaster: Ref. catastral
|
||||||
|
smallHolding: Parcela
|
||||||
|
area: Polígono
|
||||||
|
town: Municipio
|
||||||
|
registry: Registro urbanístico
|
||||||
|
volume: Volumen
|
||||||
|
book: Libro
|
||||||
|
page: Folio
|
||||||
|
farm: Finca
|
||||||
|
registration: Inscripción
|
||||||
|
value: Valor
|
||||||
|
group: Grupo
|
||||||
|
m2: m2
|
||||||
|
owner: Propietario
|
||||||
|
allocation: Partida
|
||||||
|
tome: Volumen
|
||||||
|
map: Mapa
|
||||||
|
notary: Notario
|
||||||
|
protocol: Prot. notarial
|
||||||
|
purchased: F. compra
|
||||||
|
booked: F. finalización
|
||||||
|
createProperty: Nueva propiedad
|
||||||
|
goToMap: Ir al mapa
|
||||||
|
hasFormalization: Formalización
|
||||||
|
hasSimpleCopy: Copia simple
|
||||||
|
hasOriginalPropertyDeed: Escritura original
|
||||||
|
hasFundProvision: Provisión de fondos
|
||||||
|
params:
|
||||||
|
search: Búsqueda general
|
||||||
|
id: Id
|
||||||
|
name: Nombre
|
||||||
|
supplierFk: Notario
|
||||||
|
propertyGroupFk: Grupo
|
||||||
|
protocol: Prot. notarial
|
||||||
|
companyFk: Propietario
|
||||||
|
purchased: F. compra
|
||||||
|
booked: F. finalización
|
||||||
|
cadaster: Ref. catastral
|
||||||
|
hasFormalization: Formalización
|
||||||
|
hasSimpleCopy: Copia simple
|
||||||
|
hasOriginalPropertyDeed: Escritura original
|
||||||
|
hasFundProvision: Provisión de fondos
|
|
@ -233,10 +233,10 @@ const ticketColumns = ref([
|
||||||
</span>
|
</span>
|
||||||
</QTd>
|
</QTd>
|
||||||
</template>
|
</template>
|
||||||
<template #body-cell-client="{ row }">
|
<template #body-cell-client="{ value, row }">
|
||||||
<QTd>
|
<QTd auto-width>
|
||||||
<span class="link">
|
<span class="link">
|
||||||
{{ row.clientFk }}
|
{{ value }}
|
||||||
<CustomerDescriptorProxy :id="row?.clientFk" />
|
<CustomerDescriptorProxy :id="row?.clientFk" />
|
||||||
</span>
|
</span>
|
||||||
</QTd>
|
</QTd>
|
||||||
|
|
|
@ -28,7 +28,6 @@ const body = {
|
||||||
:add-note="true"
|
:add-note="true"
|
||||||
:filter="noteFilter"
|
:filter="noteFilter"
|
||||||
:body="body"
|
:body="body"
|
||||||
style="overflow-y: auto"
|
|
||||||
required
|
required
|
||||||
deletable
|
deletable
|
||||||
/>
|
/>
|
||||||
|
|
|
@ -18,5 +18,3 @@ vehicle:
|
||||||
params:
|
params:
|
||||||
vehicleTypeFk: Type
|
vehicleTypeFk: Type
|
||||||
vehicleStateFk: State
|
vehicleStateFk: State
|
||||||
errors:
|
|
||||||
documentIdEmpty: The document identifier can't be empty
|
|
||||||
|
|
|
@ -18,5 +18,3 @@ vehicle:
|
||||||
params:
|
params:
|
||||||
vehicleTypeFk: Tipo
|
vehicleTypeFk: Tipo
|
||||||
vehicleStateFk: Estado
|
vehicleStateFk: Estado
|
||||||
errors:
|
|
||||||
documentIdEmpty: El número de documento no puede estar vacío
|
|
||||||
|
|
|
@ -80,7 +80,7 @@ const columns = computed(() => [
|
||||||
<VnTable
|
<VnTable
|
||||||
:data-key="dataKey"
|
:data-key="dataKey"
|
||||||
:columns="columns"
|
:columns="columns"
|
||||||
:is-editable="false"
|
is-editable="false"
|
||||||
:right-search="false"
|
:right-search="false"
|
||||||
:use-model="true"
|
:use-model="true"
|
||||||
:disable-option="{ table: true }"
|
:disable-option="{ table: true }"
|
||||||
|
|
|
@ -90,7 +90,7 @@ const onDataSaved = ({ id }) => {
|
||||||
<VnTable
|
<VnTable
|
||||||
:data-key="dataKey"
|
:data-key="dataKey"
|
||||||
:columns="columns"
|
:columns="columns"
|
||||||
:is-editable="false"
|
is-editable="false"
|
||||||
:right-search="false"
|
:right-search="false"
|
||||||
:use-model="true"
|
:use-model="true"
|
||||||
:disable-option="{ table: true }"
|
:disable-option="{ table: true }"
|
||||||
|
|
|
@ -65,13 +65,15 @@ function findBankFk(value, row) {
|
||||||
if (bankEntityFk) row.bankEntityFk = bankEntityFk.id;
|
if (bankEntityFk) row.bankEntityFk = bankEntityFk.id;
|
||||||
}
|
}
|
||||||
|
|
||||||
function bankEntityFilter(val) {
|
function bankEntityFilter(val, update) {
|
||||||
const needle = val.toLowerCase();
|
update(() => {
|
||||||
filteredBankEntitiesOptions.value = bankEntitiesOptions.value.filter(
|
const needle = val.toLowerCase();
|
||||||
(bank) =>
|
filteredBankEntitiesOptions.value = bankEntitiesOptions.value.filter(
|
||||||
bank.bic.toLowerCase().startsWith(needle) ||
|
(bank) =>
|
||||||
bank.name.toLowerCase().includes(needle),
|
bank.bic.toLowerCase().startsWith(needle) ||
|
||||||
);
|
bank.name.toLowerCase().includes(needle),
|
||||||
|
);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
<template>
|
<template>
|
||||||
|
@ -80,8 +82,7 @@ function bankEntityFilter(val) {
|
||||||
url="BankEntities"
|
url="BankEntities"
|
||||||
@on-fetch="
|
@on-fetch="
|
||||||
(data) => {
|
(data) => {
|
||||||
bankEntitiesOptions = data;
|
(bankEntitiesOptions = data), (filteredBankEntitiesOptions = data);
|
||||||
filteredBankEntitiesOptions = data;
|
|
||||||
}
|
}
|
||||||
"
|
"
|
||||||
auto-load
|
auto-load
|
||||||
|
@ -134,8 +135,10 @@ function bankEntityFilter(val) {
|
||||||
:label="t('worker.create.bankEntity')"
|
:label="t('worker.create.bankEntity')"
|
||||||
v-model="row.bankEntityFk"
|
v-model="row.bankEntityFk"
|
||||||
:options="filteredBankEntitiesOptions"
|
:options="filteredBankEntitiesOptions"
|
||||||
:filter-fn="bankEntityFilter"
|
:default-filter="false"
|
||||||
|
@filter="(val, update) => bankEntityFilter(val, update)"
|
||||||
option-label="bic"
|
option-label="bic"
|
||||||
|
option-value="id"
|
||||||
hide-selected
|
hide-selected
|
||||||
:required="true"
|
:required="true"
|
||||||
:roles-allowed-to-create="['financial']"
|
:roles-allowed-to-create="['financial']"
|
||||||
|
|
|
@ -5,7 +5,7 @@ import SupplierSummary from './SupplierSummary.vue';
|
||||||
const $props = defineProps({
|
const $props = defineProps({
|
||||||
id: {
|
id: {
|
||||||
type: Number,
|
type: Number,
|
||||||
default: null,
|
required: true,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
|
@ -101,7 +101,7 @@ const onNextStep = async () => {
|
||||||
t('basicData.negativesConfirmMessage'),
|
t('basicData.negativesConfirmMessage'),
|
||||||
submitWithNegatives,
|
submitWithNegatives,
|
||||||
);
|
);
|
||||||
else await submit();
|
else submit();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
|
@ -28,7 +28,6 @@ const props = defineProps({
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
restoreTicket();
|
restoreTicket();
|
||||||
hasDocuware();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
watch(
|
watch(
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
<script setup>
|
<script setup>
|
||||||
import TicketDescriptor from './TicketDescriptor.vue';
|
import TicketDescriptor from './TicketDescriptor.vue';
|
||||||
import TicketSummary from './TicketSummary.vue';
|
import TicketSummary from './TicketSummary.vue';
|
||||||
|
|
||||||
const $props = defineProps({
|
const $props = defineProps({
|
||||||
id: {
|
id: {
|
||||||
type: Number,
|
type: Number,
|
||||||
|
@ -9,7 +10,7 @@ const $props = defineProps({
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
<template>
|
<template>
|
||||||
<QPopupProxy data-cy="TicketDescriptor">
|
<QPopupProxy>
|
||||||
<TicketDescriptor v-if="$props.id" :id="$props.id" :summary="TicketSummary" />
|
<TicketDescriptor v-if="$props.id" :id="$props.id" :summary="TicketSummary" />
|
||||||
</QPopupProxy>
|
</QPopupProxy>
|
||||||
</template>
|
</template>
|
||||||
|
|
|
@ -35,13 +35,8 @@ const onDataSaved = () => dmsListRef.value.dmsRef.fetch();
|
||||||
class="fill-icon"
|
class="fill-icon"
|
||||||
>
|
>
|
||||||
<QTooltip>
|
<QTooltip>
|
||||||
{{ t('Import from existing') }}
|
{{ t('globals.import') }}
|
||||||
</QTooltip>
|
</QTooltip>
|
||||||
</QBtn>
|
</QBtn>
|
||||||
</QPageSticky>
|
</QPageSticky>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<i18n>
|
|
||||||
es:
|
|
||||||
Import from existing: Importar desde existente
|
|
||||||
</i18n>
|
|
||||||
|
|
|
@ -22,7 +22,7 @@ const dmsId = ref(null);
|
||||||
|
|
||||||
const importDms = async () => {
|
const importDms = async () => {
|
||||||
try {
|
try {
|
||||||
if (!dmsId.value) throw new Error(t(`The document identifier can't be empty`));
|
if (!dmsId.value) throw new Error(t(`globals.errors.documentIdEmpty`));
|
||||||
|
|
||||||
const data = {
|
const data = {
|
||||||
ticketFk: route.params.id,
|
ticketFk: route.params.id,
|
||||||
|
@ -56,7 +56,7 @@ const importDms = async () => {
|
||||||
<template #form-inputs>
|
<template #form-inputs>
|
||||||
<VnRow>
|
<VnRow>
|
||||||
<VnSelect
|
<VnSelect
|
||||||
:label="t('Document')"
|
:label="t('globals.document')"
|
||||||
:options="dmsOptions"
|
:options="dmsOptions"
|
||||||
hide-selected
|
hide-selected
|
||||||
option-label="id"
|
option-label="id"
|
||||||
|
@ -67,9 +67,3 @@ const importDms = async () => {
|
||||||
</template>
|
</template>
|
||||||
</FormModelPopup>
|
</FormModelPopup>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<i18n>
|
|
||||||
es:
|
|
||||||
Document: Documento
|
|
||||||
The document indentifier can't be empty: El número de documento no puede estar vacío
|
|
||||||
</i18n>
|
|
||||||
|
|
|
@ -773,7 +773,6 @@ watch(
|
||||||
v-model="row.itemFk"
|
v-model="row.itemFk"
|
||||||
:use-like="false"
|
:use-like="false"
|
||||||
@update:model-value="changeItem(row)"
|
@update:model-value="changeItem(row)"
|
||||||
autofocus
|
|
||||||
>
|
>
|
||||||
<template #option="scope">
|
<template #option="scope">
|
||||||
<QItem v-bind="scope.itemProps">
|
<QItem v-bind="scope.itemProps">
|
||||||
|
|
|
@ -3,9 +3,7 @@ import { ref } from 'vue';
|
||||||
|
|
||||||
import VnInputDate from 'src/components/common/VnInputDate.vue';
|
import VnInputDate from 'src/components/common/VnInputDate.vue';
|
||||||
import split from './components/split';
|
import split from './components/split';
|
||||||
import { displayResults } from 'src/pages/Ticket/Negative/composables/notifyResults';
|
const emit = defineEmits(['ticketTransfered']);
|
||||||
const { notifyResults } = displayResults();
|
|
||||||
const emit = defineEmits(['ticketTransferred']);
|
|
||||||
|
|
||||||
const $props = defineProps({
|
const $props = defineProps({
|
||||||
ticket: {
|
ticket: {
|
||||||
|
@ -18,20 +16,13 @@ const splitDate = ref(Date.vnNew());
|
||||||
|
|
||||||
const splitSelectedRows = async () => {
|
const splitSelectedRows = async () => {
|
||||||
const tickets = Array.isArray($props.ticket) ? $props.ticket : [$props.ticket];
|
const tickets = Array.isArray($props.ticket) ? $props.ticket : [$props.ticket];
|
||||||
const results = await split(tickets, splitDate.value);
|
await split(tickets, splitDate.value);
|
||||||
notifyResults(results, 'ticketFk');
|
emit('ticketTransfered', tickets);
|
||||||
emit('ticketTransferred', tickets);
|
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<VnInputDate
|
<VnInputDate class="q-mr-sm" :label="$t('New date')" v-model="splitDate" clearable />
|
||||||
class="q-mr-sm"
|
|
||||||
:label="$t('New date')"
|
|
||||||
v-model="splitDate"
|
|
||||||
clearable
|
|
||||||
autofocus
|
|
||||||
/>
|
|
||||||
<QBtn class="q-mr-sm" color="primary" label="Split" @click="splitSelectedRows"></QBtn>
|
<QBtn class="q-mr-sm" color="primary" label="Split" @click="splitSelectedRows"></QBtn>
|
||||||
</template>
|
</template>
|
||||||
<style lang="scss">
|
<style lang="scss">
|
||||||
|
|
|
@ -5,7 +5,7 @@ import VnInput from 'src/components/common/VnInput.vue';
|
||||||
import TicketTransferForm from './TicketTransferForm.vue';
|
import TicketTransferForm from './TicketTransferForm.vue';
|
||||||
|
|
||||||
import { toDateFormat } from 'src/filters/date.js';
|
import { toDateFormat } from 'src/filters/date.js';
|
||||||
const emit = defineEmits(['ticketTransferred']);
|
const emit = defineEmits(['ticketTransfered']);
|
||||||
|
|
||||||
const $props = defineProps({
|
const $props = defineProps({
|
||||||
mana: {
|
mana: {
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
<script setup>
|
<script setup>
|
||||||
import { ref } from 'vue';
|
import { ref } from 'vue';
|
||||||
import TicketTransfer from './TicketTransfer.vue';
|
import TicketTransfer from './TicketTransfer.vue';
|
||||||
import TicketSplit from './TicketSplit.vue';
|
import Split from './TicketSplit.vue';
|
||||||
const emit = defineEmits(['ticketTransferred']);
|
const emit = defineEmits(['ticketTransfered']);
|
||||||
|
|
||||||
const $props = defineProps({
|
const $props = defineProps({
|
||||||
mana: {
|
mana: {
|
||||||
|
@ -35,7 +35,7 @@ const transferRef = ref(null);
|
||||||
<template>
|
<template>
|
||||||
<QPopupProxy ref="popupProxyRef" data-cy="ticketTransferPopup">
|
<QPopupProxy ref="popupProxyRef" data-cy="ticketTransferPopup">
|
||||||
<div class="flex row items-center q-ma-lg" v-if="$props.split">
|
<div class="flex row items-center q-ma-lg" v-if="$props.split">
|
||||||
<TicketSplit
|
<Split
|
||||||
ref="splitRef"
|
ref="splitRef"
|
||||||
@splitSelectedRows="splitSelectedRows"
|
@splitSelectedRows="splitSelectedRows"
|
||||||
:ticket="$props.ticket"
|
:ticket="$props.ticket"
|
||||||
|
|
|
@ -1,11 +1,13 @@
|
||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
|
import notifyResults from 'src/utils/notifyResults';
|
||||||
|
|
||||||
export default async function (data, date) {
|
export default async function (data, date) {
|
||||||
const reducedData = data.reduce((acc, item) => {
|
const reducedData = data.reduce((acc, item) => {
|
||||||
const existing = acc.find(({ ticketFk }) => ticketFk === item.id);
|
const existing = acc.find(({ ticketFk }) => ticketFk === item.id);
|
||||||
if (existing) {
|
if (existing) {
|
||||||
existing.sales.push(item.saleFk);
|
existing.sales.push(item.saleFk);
|
||||||
} else {
|
} else {
|
||||||
acc.push({ ticketFk: item.ticketFk, sales: [item.saleFk], date });
|
acc.push({ ticketFk: item.id, sales: [item.saleFk], date });
|
||||||
}
|
}
|
||||||
return acc;
|
return acc;
|
||||||
}, []);
|
}, []);
|
||||||
|
@ -14,5 +16,7 @@ export default async function (data, date) {
|
||||||
|
|
||||||
const results = await Promise.allSettled(promises);
|
const results = await Promise.allSettled(promises);
|
||||||
|
|
||||||
|
notifyResults(results, 'ticketFk');
|
||||||
|
|
||||||
return results;
|
return results;
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,6 +23,7 @@ const tableRef = ref();
|
||||||
const changeItemDialogRef = ref(null);
|
const changeItemDialogRef = ref(null);
|
||||||
const changeStateDialogRef = ref(null);
|
const changeStateDialogRef = ref(null);
|
||||||
const changeQuantityDialogRef = ref(null);
|
const changeQuantityDialogRef = ref(null);
|
||||||
|
const showProposalDialog = ref(false);
|
||||||
const showChangeQuantityDialog = ref(false);
|
const showChangeQuantityDialog = ref(false);
|
||||||
const selectedRows = ref([]);
|
const selectedRows = ref([]);
|
||||||
const route = useRoute();
|
const route = useRoute();
|
||||||
|
@ -62,7 +63,6 @@ const showItemProposal = () => {
|
||||||
.dialog({
|
.dialog({
|
||||||
component: ItemProposalProxy,
|
component: ItemProposalProxy,
|
||||||
componentProps: {
|
componentProps: {
|
||||||
filter: filter.value,
|
|
||||||
itemLack: tableRef.value.itemLack,
|
itemLack: tableRef.value.itemLack,
|
||||||
replaceAction: true,
|
replaceAction: true,
|
||||||
sales: selectedRows.value,
|
sales: selectedRows.value,
|
||||||
|
@ -117,17 +117,21 @@ const showItemProposal = () => {
|
||||||
sales: selectedRows,
|
sales: selectedRows,
|
||||||
lastActiveTickets: selectedRows.map((row) => row.id),
|
lastActiveTickets: selectedRows.map((row) => row.id),
|
||||||
}"
|
}"
|
||||||
@ticket-transferred="reload"
|
@ticket-transfered="reload"
|
||||||
></TicketTransferProxy>
|
></TicketTransferProxy>
|
||||||
</template>
|
</template>
|
||||||
</QBtn>
|
</QBtn>
|
||||||
<QBtn
|
<QBtn
|
||||||
color="primary"
|
color="primary"
|
||||||
@click="showItemProposal"
|
@click="showProposalDialog = true"
|
||||||
:disable="!(selectedRows.length === 1)"
|
:disable="selectedRows.length < 1"
|
||||||
data-cy="itemProposal"
|
data-cy="itemProposal"
|
||||||
>
|
>
|
||||||
<QIcon name="import_export" class="rotate-90" />
|
<QIcon
|
||||||
|
name="import_export"
|
||||||
|
class="rotate-90"
|
||||||
|
@click="showItemProposal"
|
||||||
|
></QIcon>
|
||||||
<QTooltip bottom anchor="bottom right">
|
<QTooltip bottom anchor="bottom right">
|
||||||
{{ t('itemProposal') }}
|
{{ t('itemProposal') }}
|
||||||
</QTooltip>
|
</QTooltip>
|
||||||
|
@ -135,7 +139,7 @@ const showItemProposal = () => {
|
||||||
<VnPopupProxy
|
<VnPopupProxy
|
||||||
data-cy="changeItem"
|
data-cy="changeItem"
|
||||||
icon="sync"
|
icon="sync"
|
||||||
:disable="!(selectedRows.length === 1)"
|
:disable="selectedRows.length < 1"
|
||||||
:tooltip="t('negative.detail.modal.changeItem.title')"
|
:tooltip="t('negative.detail.modal.changeItem.title')"
|
||||||
>
|
>
|
||||||
<template #extraIcon> <QIcon name="vn:item" /> </template>
|
<template #extraIcon> <QIcon name="vn:item" /> </template>
|
||||||
|
@ -149,7 +153,7 @@ const showItemProposal = () => {
|
||||||
<VnPopupProxy
|
<VnPopupProxy
|
||||||
data-cy="changeState"
|
data-cy="changeState"
|
||||||
icon="sync"
|
icon="sync"
|
||||||
:disable="!(selectedRows.length === 1)"
|
:disable="selectedRows.length < 1"
|
||||||
:tooltip="t('negative.detail.modal.changeState.title')"
|
:tooltip="t('negative.detail.modal.changeState.title')"
|
||||||
>
|
>
|
||||||
<template #extraIcon> <QIcon name="vn:eye" /> </template>
|
<template #extraIcon> <QIcon name="vn:eye" /> </template>
|
||||||
|
@ -163,7 +167,7 @@ const showItemProposal = () => {
|
||||||
<VnPopupProxy
|
<VnPopupProxy
|
||||||
data-cy="changeQuantity"
|
data-cy="changeQuantity"
|
||||||
icon="sync"
|
icon="sync"
|
||||||
:disable="!(selectedRows.length === 1)"
|
:disable="selectedRows.length < 1"
|
||||||
:tooltip="t('negative.detail.modal.changeQuantity.title')"
|
:tooltip="t('negative.detail.modal.changeQuantity.title')"
|
||||||
@click="showChangeQuantityDialog = true"
|
@click="showChangeQuantityDialog = true"
|
||||||
>
|
>
|
||||||
|
|
|
@ -6,7 +6,6 @@ import FetchData from 'components/FetchData.vue';
|
||||||
import VnFilterPanel from 'src/components/ui/VnFilterPanel.vue';
|
import VnFilterPanel from 'src/components/ui/VnFilterPanel.vue';
|
||||||
import VnInput from 'src/components/common/VnInput.vue';
|
import VnInput from 'src/components/common/VnInput.vue';
|
||||||
import VnSelect from 'src/components/common/VnSelect.vue';
|
import VnSelect from 'src/components/common/VnSelect.vue';
|
||||||
import VnInputDateTime from 'src/components/common/VnInputDateTime.vue';
|
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
dataKey: {
|
dataKey: {
|
||||||
|
@ -67,7 +66,6 @@ const setUserParams = (params) => {
|
||||||
:data-key="props.dataKey"
|
:data-key="props.dataKey"
|
||||||
:search-button="true"
|
:search-button="true"
|
||||||
@set-user-params="setUserParams"
|
@set-user-params="setUserParams"
|
||||||
:unremovable-params="['warehouseFk']"
|
|
||||||
>
|
>
|
||||||
<template #tags="{ tag, formatFn }">
|
<template #tags="{ tag, formatFn }">
|
||||||
<div class="q-gutter-x-xs">
|
<div class="q-gutter-x-xs">
|
||||||
|
@ -79,11 +77,12 @@ const setUserParams = (params) => {
|
||||||
<QList dense class="q-gutter-y-sm q-mt-sm">
|
<QList dense class="q-gutter-y-sm q-mt-sm">
|
||||||
<QItem>
|
<QItem>
|
||||||
<QItemSection>
|
<QItemSection>
|
||||||
<VnInputDateTime
|
<VnInput
|
||||||
v-model="params.availabled"
|
v-model="params.days"
|
||||||
:label="t('negative.availabled')"
|
:label="t('negative.days')"
|
||||||
dense
|
dense
|
||||||
filled
|
filled
|
||||||
|
type="number"
|
||||||
@update:model-value="
|
@update:model-value="
|
||||||
(value) => {
|
(value) => {
|
||||||
setUserParams(params);
|
setUserParams(params);
|
||||||
|
|
|
@ -21,13 +21,14 @@ const selectedRows = ref([]);
|
||||||
const tableRef = ref();
|
const tableRef = ref();
|
||||||
const filterParams = ref({});
|
const filterParams = ref({});
|
||||||
const negativeParams = reactive({
|
const negativeParams = reactive({
|
||||||
|
days: useRole().likeAny('buyer') ? 2 : 0,
|
||||||
warehouseFk: useState().getUser().value.warehouseFk,
|
warehouseFk: useState().getUser().value.warehouseFk,
|
||||||
availabled: Date.getCurrentDateTimeFormatted(),
|
|
||||||
});
|
});
|
||||||
const redirectToCreateView = ({ itemFk }) => {
|
const redirectToCreateView = ({ itemFk }) => {
|
||||||
router.push({
|
router.push({
|
||||||
name: 'NegativeDetail',
|
name: 'NegativeDetail',
|
||||||
params: { id: itemFk },
|
params: { id: itemFk },
|
||||||
|
query: { days: filterParams.value.days ?? negativeParams.days },
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
const columns = computed(() => [
|
const columns = computed(() => [
|
||||||
|
@ -64,19 +65,15 @@ const columns = computed(() => [
|
||||||
columnFilter: {
|
columnFilter: {
|
||||||
component: 'input',
|
component: 'input',
|
||||||
type: 'number',
|
type: 'number',
|
||||||
inWhere: false,
|
columnClass: 'shrink',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'longName',
|
name: 'longName',
|
||||||
align: 'left',
|
align: 'center',
|
||||||
label: t('negative.longName'),
|
label: t('negative.longName'),
|
||||||
field: ({ longName }) => longName,
|
field: ({ longName }) => longName,
|
||||||
columnFilter: {
|
|
||||||
component: 'input',
|
|
||||||
inWhere: false,
|
|
||||||
useLike: true,
|
|
||||||
},
|
|
||||||
sortable: true,
|
sortable: true,
|
||||||
headerStyle: 'width: 350px',
|
headerStyle: 'width: 350px',
|
||||||
cardVisible: true,
|
cardVisible: true,
|
||||||
|
@ -97,11 +94,6 @@ const columns = computed(() => [
|
||||||
field: ({ inkFk }) => inkFk,
|
field: ({ inkFk }) => inkFk,
|
||||||
sortable: true,
|
sortable: true,
|
||||||
cardVisible: true,
|
cardVisible: true,
|
||||||
columnFilter: {
|
|
||||||
component: 'input',
|
|
||||||
columnClass: 'shrink',
|
|
||||||
inWhere: false,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'size',
|
name: 'size',
|
||||||
|
@ -163,6 +155,7 @@ const setUserParams = (params) => {
|
||||||
<TicketLackFilter data-key="NegativeList" @set-user-params="setUserParams" />
|
<TicketLackFilter data-key="NegativeList" @set-user-params="setUserParams" />
|
||||||
</template>
|
</template>
|
||||||
</RightMenu>
|
</RightMenu>
|
||||||
|
{{ filterRef }}
|
||||||
<VnTable
|
<VnTable
|
||||||
ref="tableRef"
|
ref="tableRef"
|
||||||
data-key="NegativeList"
|
data-key="NegativeList"
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue