Merge branch 'dev' of https://gitea.verdnatura.es/verdnatura/salix-front into 6994-vnLog_descriptors

This commit is contained in:
Alex Moreno 2025-03-12 14:18:16 +01:00
commit 6ddc1c77f8
12 changed files with 12411 additions and 8856 deletions

View File

@ -25,6 +25,8 @@ RUN apt-get update \
libnss3 \ libnss3 \
libxss1 \ libxss1 \
libxtst6 \ libxtst6 \
mesa-vulkan-drivers \
vulkan-tools \
xauth \ xauth \
xvfb \ xvfb \
&& apt-get clean \ && apt-get clean \

File diff suppressed because it is too large Load Diff

View File

@ -7,6 +7,7 @@ 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 VnInputDate from 'components/common/VnInputDate.vue'; import VnInputDate from 'components/common/VnInputDate.vue';
import VnInputNumber from 'src/components/common/VnInputNumber.vue'; import VnInputNumber from 'src/components/common/VnInputNumber.vue';
import VnSelect from 'src/components/common/VnSelect.vue';
const { t } = useI18n(); const { t } = useI18n();
const props = defineProps({ const props = defineProps({
@ -30,7 +31,7 @@ const states = ref();
<QItem> <QItem>
<QItemSection> <QItemSection>
<VnInput <VnInput
:label="t('Customer ID')" :label="t('globals.params.clientFk')"
v-model="params.clientFk" v-model="params.clientFk"
is-outlined is-outlined
/> />
@ -38,13 +39,17 @@ const states = ref();
</QItem> </QItem>
<QItem> <QItem>
<QItemSection> <QItemSection>
<VnInput v-model="params.fi" :label="t('FI')" is-outlined /> <VnInput
v-model="params.fi"
:label="t('globals.params.fi')"
is-outlined
/>
</QItemSection> </QItemSection>
</QItem> </QItem>
<QItem> <QItem>
<QItemSection> <QItemSection>
<VnInputNumber <VnInputNumber
:label="t('Amount')" :label="t('globals.amount')"
v-model="params.amount" v-model="params.amount"
is-outlined is-outlined
data-cy="InvoiceOutFilterAmountBtn" data-cy="InvoiceOutFilterAmountBtn"
@ -54,7 +59,7 @@ const states = ref();
<QItem> <QItem>
<QItemSection> <QItemSection>
<QInput <QInput
:label="t('Min')" :label="t('invoiceOut.params.min')"
dense dense
lazy-rules lazy-rules
outlined outlined
@ -65,7 +70,7 @@ const states = ref();
</QItemSection> </QItemSection>
<QItemSection> <QItemSection>
<QInput <QInput
:label="t('Max')" :label="t('invoiceOut.params.max')"
dense dense
lazy-rules lazy-rules
outlined outlined
@ -78,7 +83,7 @@ const states = ref();
<QItem> <QItem>
<QItemSection> <QItemSection>
<QCheckbox <QCheckbox
:label="t('Has PDF')" :label="t('invoiceOut.params.hasPdf')"
toggle-indeterminate toggle-indeterminate
v-model="params.hasPdf" v-model="params.hasPdf"
/> />
@ -88,14 +93,31 @@ const states = ref();
<QItemSection> <QItemSection>
<VnInputDate <VnInputDate
v-model="params.created" v-model="params.created"
:label="t('Created')" :label="t('invoiceOut.params.created')"
is-outlined is-outlined
/> />
</QItemSection> </QItemSection>
</QItem> </QItem>
<QItem> <QItem>
<QItemSection> <QItemSection>
<VnInputDate v-model="params.dued" :label="t('Dued')" is-outlined /> <VnInputDate
v-model="params.dued"
:label="t('invoiceOut.params.dued')"
is-outlined
/>
</QItemSection>
</QItem>
<QItem>
<QItemSection>
<VnSelect
outlined
rounded
:label="t('globals.params.departmentFk')"
v-model="params.departmentFk"
option-value="id"
option-label="name"
url="Departments"
/>
</QItemSection> </QItemSection>
</QItem> </QItem>
</template> </template>

View File

@ -16,6 +16,7 @@ import VnRow from 'src/components/ui/VnRow.vue';
import VnRadio from 'src/components/common/VnRadio.vue'; import VnRadio from 'src/components/common/VnRadio.vue';
import VnInput from 'src/components/common/VnInput.vue'; import VnInput from 'src/components/common/VnInput.vue';
import CustomerDescriptorProxy from '../Customer/Card/CustomerDescriptorProxy.vue'; import CustomerDescriptorProxy from '../Customer/Card/CustomerDescriptorProxy.vue';
import DepartmentDescriptorProxy from 'src/pages/Worker/Department/Card/DepartmentDescriptorProxy.vue';
import VnSection from 'src/components/common/VnSection.vue'; import VnSection from 'src/components/common/VnSection.vue';
const { t } = useI18n(); const { t } = useI18n();
@ -86,6 +87,20 @@ const columns = computed(() => [
component: null, component: null,
}, },
}, },
{
align: 'left',
name: 'departmentFk',
label: t('globals.params.departmentFk'),
component: 'select',
attrs: {
url: 'Departments',
},
create: true,
columnField: {
component: null,
},
format: (row, dashIfEmpty) => dashIfEmpty(row.departmentName),
},
{ {
align: 'left', align: 'left',
name: 'companyFk', name: 'companyFk',
@ -229,6 +244,12 @@ watchEffect(selectedRows);
<CustomerDescriptorProxy :id="row.clientFk" /> <CustomerDescriptorProxy :id="row.clientFk" />
</span> </span>
</template> </template>
<template #column-departmentFk="{ row }">
<span class="link" @click.stop>
{{ row.departmentName || '-' }}
<DepartmentDescriptorProxy :id="row?.departmentFk" />
</span>
</template>
<template #more-create-dialog="{ data }"> <template #more-create-dialog="{ data }">
<div class="row q-col-gutter-xs col-span-2"> <div class="row q-col-gutter-xs col-span-2">
<div class="col-12"> <div class="col-12">

View File

@ -221,7 +221,7 @@ en:
attenderFk: Atender attenderFk: Atender
clientFk: Client id clientFk: Client id
warehouseFk: Warehouse warehouseFk: Warehouse
requesterFk: Salesperson requesterFk: Requester
from: From from: From
to: To to: To
mine: For me mine: For me
@ -239,7 +239,7 @@ es:
attenderFk: Comprador attenderFk: Comprador
clientFk: Id cliente clientFk: Id cliente
warehouseFk: Almacén warehouseFk: Almacén
requesterFk: Comercial requesterFk: Solicitante
from: Desde from: Desde
to: Hasta to: Hasta
mine: Para mi mine: Para mi

View File

@ -84,7 +84,7 @@ item:
attenderFk: Atender attenderFk: Atender
clientFk: Client id clientFk: Client id
warehouseFk: Warehouse warehouseFk: Warehouse
requesterFk: Salesperson requesterFk: Requester
from: From from: From
to: To to: To
mine: For me mine: For me

View File

@ -93,7 +93,7 @@ item:
attenderFk: Comprador attenderFk: Comprador
clientFk: Id cliente clientFk: Id cliente
warehouseFk: Almacén warehouseFk: Almacén
requesterFk: Comercial requesterFk: Solicitante
from: Desde from: Desde
to: Hasta to: Hasta
mine: Para mi mine: Para mi

View File

@ -11,6 +11,7 @@ import AgencyDescriptorProxy from 'src/pages/Route/Agency/Card/AgencyDescriptorP
import VehicleDescriptorProxy from 'src/pages/Route/Vehicle/Card/VehicleDescriptorProxy.vue'; import VehicleDescriptorProxy from 'src/pages/Route/Vehicle/Card/VehicleDescriptorProxy.vue';
import VnSection from 'src/components/common/VnSection.vue'; import VnSection from 'src/components/common/VnSection.vue';
import VnSelectWorker from 'src/components/common/VnSelectWorker.vue'; import VnSelectWorker from 'src/components/common/VnSelectWorker.vue';
import RouteTickets from './RouteTickets.vue';
const { t } = useI18n(); const { t } = useI18n();
const { viewSummary } = useSummaryDialog(); const { viewSummary } = useSummaryDialog();
@ -26,6 +27,12 @@ const routeFilter = {
}, },
], ],
}; };
function redirectToTickets(id) {
const url = `#/route/${id}/tickets`;
window.open(url, '_blank');
}
const columns = computed(() => [ const columns = computed(() => [
{ {
align: 'right', align: 'right',
@ -141,6 +148,12 @@ const columns = computed(() => [
align: 'right', align: 'right',
name: 'tableActions', name: 'tableActions',
actions: [ actions: [
{
title: t('globals.pageTitles.tickets'),
icon: 'vn:ticket',
action: (row) => redirectToTickets(row?.id),
isPrimary: true,
},
{ {
title: t('components.smartCard.viewSummary'), title: t('components.smartCard.viewSummary'),
icon: 'preview', icon: 'preview',

View File

@ -37,9 +37,9 @@ const columns = computed(() => [
align: 'left', align: 'left',
}, },
{ {
name: 'city', name: 'client',
label: t('City'), label: t('Client'),
field: (row) => row?.city, field: (row) => row?.nickname,
sortable: false, sortable: false,
align: 'left', align: 'left',
}, },
@ -51,9 +51,9 @@ const columns = computed(() => [
align: 'center', align: 'center',
}, },
{ {
name: 'client', name: 'city',
label: t('Client'), label: t('City'),
field: (row) => row?.nickname, field: (row) => row?.city,
sortable: false, sortable: false,
align: 'left', align: 'left',
}, },

View File

@ -73,6 +73,7 @@ const onNodeExpanded = async (nodeKeysArray) => {
const response = await axios.get(`Zones/${route.params.id}/getLeaves`, { const response = await axios.get(`Zones/${route.params.id}/getLeaves`, {
params, params,
}); });
response.data = JSON.parse(response.data);
if (response.data) { if (response.data) {
node.childs = response.data.map((n) => { node.childs = response.data.map((n) => {
if (n.sons > 0) n.childs = [{}]; if (n.sons > 0) n.childs = [{}];
@ -126,14 +127,17 @@ watch(
async (val) => { async (val) => {
if (!val) return; if (!val) return;
// // Se triggerea cuando se actualiza el store.data, el cual es el resultado del fetch de la searchbar // // Se triggerea cuando se actualiza el store.data, el cual es el resultado del fetch de la searchbar
val = JSON.parse(val);
if (!nodes.value[0]) nodes.value = [defaultNode]; if (!nodes.value[0]) nodes.value = [defaultNode];
nodes.value[0].childs = [...val]; nodes.value[0].childs = [...val];
const fetchedNodeKeys = val.flatMap(getNodeIds); const fetchedNodeKeys = val.flatMap(getNodeIds);
state.set('Tree', [...fetchedNodeKeys]); state.set('Tree', [...fetchedNodeKeys]);
expanded.value = [null, ...fetchedNodeKeys]; expanded.value = [null, ...fetchedNodeKeys];
const fetchs = [];
for (let n of state.get('Tree')) { for (let n of state.get('Tree')) {
await fetchNodeLeaves(n); fetchs.push(fetchNodeLeaves(n));
} }
await Promise.all(fetchs);
previousExpandedNodes.value = new Set(expanded.value); previousExpandedNodes.value = new Set(expanded.value);
}, },
{ immediate: true }, { immediate: true },

View File

@ -1,14 +1,13 @@
/// <reference types="cypress" /> /// <reference types="cypress" />
describe('Client balance', () => { describe('Client balance', () => {
beforeEach(() => { beforeEach(() => {
cy.viewport(1280, 720);
cy.login('developer'); cy.login('developer');
cy.visit('#/customer/1101/balance'); cy.visit('#/customer/1101/balance');
}); });
it('Should create a mandate', () => { it('Should create a mandate', () => {
cy.get('.q-page-sticky > div > .q-btn').click(); cy.get('.q-page-sticky > div > .q-btn').click();
cy.selectOption('[data-cy="paymentBank"]', 2); cy.selectOption('[data-cy="paymentBank"]', 2);
cy.dataCy('paymentAmount_input').type('100'); cy.dataCy('paymentAmount_input').clear().type('100');
cy.saveCard(); cy.saveCard();
}); });
}); });

View File

@ -62,12 +62,7 @@ Cypress.Commands.overwrite('visit', (originalFn, url, options, waitRequest = tru
originalFn(url, options); originalFn(url, options);
cy.waitUntil(() => cy.document().then((doc) => doc.readyState === 'complete')); cy.waitUntil(() => cy.document().then((doc) => doc.readyState === 'complete'));
cy.waitUntil(() => cy.get('main').should('exist')); cy.waitUntil(() => cy.get('main').should('exist'));
if (waitRequest) if (waitRequest) cy.waitSpinner();
cy.get('body').then(($body) => {
if ($body.find('[data-cy="loading-spinner"]').length) {
cy.get('[data-cy="loading-spinner"]').should('not.be.visible');
}
});
}); });
Cypress.Commands.add('waitForElement', (element) => { Cypress.Commands.add('waitForElement', (element) => {
@ -99,6 +94,14 @@ Cypress.Commands.add('getValue', (selector) => {
}); });
}); });
Cypress.Commands.add('waitSpinner', () => {
cy.get('body').then(($body) => {
if ($body.find('[data-cy="loading-spinner"]').length) {
cy.get('[data-cy="loading-spinner"]').should('not.be.visible');
}
});
});
// Fill Inputs // Fill Inputs
Cypress.Commands.add('selectOption', (selector, option, timeout = 2500) => { Cypress.Commands.add('selectOption', (selector, option, timeout = 2500) => {
cy.waitForElement(selector, timeout); cy.waitForElement(selector, timeout);
@ -116,6 +119,7 @@ Cypress.Commands.add('selectOption', (selector, option, timeout = 2500) => {
function selectItem(selector, option, ariaControl, hasWrite = true) { function selectItem(selector, option, ariaControl, hasWrite = true) {
if (!hasWrite) cy.wait(100); if (!hasWrite) cy.wait(100);
cy.waitSpinner();
getItems(ariaControl).then((items) => { getItems(ariaControl).then((items) => {
const matchingItem = items const matchingItem = items
@ -135,6 +139,7 @@ function getItems(ariaControl, startTime = Cypress._.now(), timeout = 2500) {
.should('exist') .should('exist')
.find('.q-item') .find('.q-item')
.should('exist') .should('exist')
.should('be.visible')
.then(($items) => { .then(($items) => {
if (!$items?.length || $items.first().text().trim() === '') { if (!$items?.length || $items.first().text().trim() === '') {
if (Cypress._.now() - startTime > timeout) { if (Cypress._.now() - startTime > timeout) {