Compare commits

...

31 Commits

Author SHA1 Message Date
Javier Segarra f615c5c196 Merge branch 'master' into hotfix_negative_available
gitea/salix-front/pipeline/pr-master This commit looks good Details
2025-04-15 13:35:38 +00:00
Jon Elias 77db8392cc Merge pull request 'Hotfix[routeSummary]: Show in table client id instead of client's name' (!1715) from Hotfix-RouteClient into master
gitea/salix-front/pipeline/head This commit looks good Details
Reviewed-on: #1715
Reviewed-by: Alex Moreno <alexm@verdnatura.es>
2025-04-15 13:22:36 +00:00
Jon Elias d8234ba141 Merge branch 'master' into Hotfix-RouteClient
gitea/salix-front/pipeline/pr-master This commit looks good Details
2025-04-15 13:00:44 +00:00
Pau Rovira 2eb70aadcb Merge pull request 'hotfix: make accountNumber to work on blur' (!1714) from hotfix_vnAccountNumber into master
gitea/salix-front/pipeline/head This commit looks good Details
Reviewed-on: #1714
Reviewed-by: Jorge Penadés <jorgep@verdnatura.es>
2025-04-15 12:38:24 +00:00
Jon Elias 1d573b5dee fix: show in table client id instead of client's name
gitea/salix-front/pipeline/pr-master This commit looks good Details
2025-04-15 14:37:09 +02:00
Pau Rovira d122752672 hotfix: make accountNumber to work on blur
gitea/salix-front/pipeline/pr-master This commit looks good Details
2025-04-15 11:54:26 +00:00
Javier Segarra 82b6fb6c2f fix: correct data retrieval after fetching in getData function
gitea/salix-front/pipeline/pr-master This commit looks good Details
2025-04-15 13:49:21 +02:00
Javier Segarra 31acc3d0a1 Merge branch 'master' into hotfix_negative_available 2025-04-15 12:23:31 +02:00
Javier Segarra a66849f361 fix: ensure submit is awaited
gitea/salix-front/pipeline/head This commit looks good Details
2025-04-15 11:38:04 +02:00
Javier Segarra 64e30b7143 refactor: remove redundant form filling step in order creation test
gitea/salix-front/pipeline/pr-master This commit is unstable Details
2025-04-15 10:38:00 +02:00
Javier Segarra de359043af Merge branch 'master' into hotfix_negative_available
gitea/salix-front/pipeline/pr-master This commit is unstable Details
2025-04-15 07:43:11 +00:00
Javier Segarra f49f694970 Merge branch 'master' into hotfix_negative_available
gitea/salix-front/pipeline/pr-master This commit is unstable Details
2025-04-14 11:09:13 +00:00
Javier Segarra 510eab8af0 feat: fix minor bugs
gitea/salix-front/pipeline/pr-master This commit is unstable Details
2025-04-11 15:04:24 +02:00
Javier Segarra b32a889dfe feat: updates 2025-04-11 12:30:37 +02:00
Javier Segarra 37bc464a98 Merge branch 'master' of https://gitea.verdnatura.es/verdnatura/salix-front into hotfix_negative_available
gitea/salix-front/pipeline/pr-master This commit is unstable Details
2025-04-11 10:06:21 +02:00
Javier Segarra 5fca08aaa2 test: add VnInputDateTime component tests 2025-04-11 10:05:47 +02:00
Javier Segarra d0f4eb9db9 feat: handle action notification 2025-04-09 10:59:43 +02:00
Javier Segarra 8f5b4af7cc Merge branch 'master' of https://gitea.verdnatura.es/verdnatura/salix-front into hotfix_negative_available 2025-04-08 12:30:29 +02:00
Javier Segarra bc75bd9dc0 feat: update order creation test to use form data and improve readability
gitea/salix-front/pipeline/pr-master There was a failure building this commit Details
2025-03-24 09:20:39 +01:00
Javier Segarra c672d33b0c feat: change i18n hasObservation
gitea/salix-front/pipeline/pr-master This commit is unstable Details
2025-03-21 11:41:55 +01:00
Javier Segarra e3c2ae935c style: improve css
gitea/salix-front/pipeline/pr-master There was a failure building this commit Details
2025-03-21 11:15:26 +01:00
Javier Segarra 69548456c3 feat: hide url 2025-03-21 11:14:13 +01:00
Javier Segarra 64fe998053 feat: add availabled item 2025-03-21 11:14:01 +01:00
Javier Segarra ae856ec0bc perf: remove comments 2025-03-21 11:13:42 +01:00
Javier Segarra 239805515d feat: show substitution note
gitea/salix-front/pipeline/pr-master This commit is unstable Details
2025-03-21 10:49:33 +01:00
Javier Segarra 56da11345b revert: remove substitutionAllowed 2025-03-21 10:49:17 +01:00
Javier Segarra 78d0f31b11 Merge branch 'master' into hotfix_negative_available
gitea/salix-front/pipeline/pr-master This commit is unstable Details
2025-03-21 09:35:30 +00:00
Javier Segarra bb949cd00c Merge branch 'master' into hotfix_negative_available
gitea/salix-front/pipeline/pr-master This commit is unstable Details
2025-03-13 08:01:56 +00:00
Javier Segarra f4c22938c7 perf: minor changes
gitea/salix-front/pipeline/pr-master This commit is unstable Details
2025-03-12 23:47:31 +01:00
Javier Segarra 23670debd1 feat: add VnInputDateTime component and enhance date handling functions
gitea/salix-front/pipeline/pr-master This commit is unstable Details
2025-03-12 15:34:30 +01:00
Javier Segarra 311c361f11 fix: minor bugs 2025-03-12 15:34:20 +01:00
30 changed files with 367 additions and 118 deletions

View File

@ -1,4 +1,6 @@
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 = () => {
@ -25,4 +27,34 @@ 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();
};
}); });

View File

@ -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,11 +219,7 @@ 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>
@ -233,11 +229,7 @@ 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">
@ -265,7 +257,6 @@ 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>{{

View File

@ -8,7 +8,8 @@ const model = defineModel({ prop: 'modelValue' });
<VnInput <VnInput
v-model="model" v-model="model"
ref="inputRef" ref="inputRef"
@keydown.tab="model = useAccountShortToStandard($event.target.value) ?? model" @keydown.tab="$refs.inputRef.vnInputRef.blur()"
@blur="model = useAccountShortToStandard(model) ?? model"
@input="model = $event.target.value.replace(/[^\d.]/g, '')" @input="model = $event.target.value.replace(/[^\d.]/g, '')"
/> />
</template> </template>

View File

@ -0,0 +1,79 @@
<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] });
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 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>

View File

@ -0,0 +1,81 @@
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).toBeInstanceOf(Date);
});
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);
});
});
});

View File

@ -44,7 +44,8 @@ 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) await getData(); if (!isSameDataKey.value || !route.params.id || $props.id !== route.params.id)
await getData();
watch( watch(
() => [$props.url, $props.filter], () => [$props.url, $props.filter],
async () => { async () => {
@ -58,7 +59,8 @@ async function getData() {
store.filter = $props.filter ?? {}; store.filter = $props.filter ?? {};
isLoading.value = true; isLoading.value = true;
try { try {
const { data } = await arrayData.fetch({ append: false, updateRouter: false }); 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 {

View File

@ -19,6 +19,7 @@ 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

View File

@ -20,10 +20,11 @@ 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

View File

@ -6,10 +6,12 @@ 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 notifyResults from 'src/utils/notifyResults'; import { displayResults } from 'src/pages/Ticket/Negative/composables/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({
@ -18,14 +20,20 @@ const $props = defineProps({
required: true, required: true,
default: () => {}, default: () => {},
}, },
filter: {
type: Object,
required: true,
default: () => {},
},
replaceAction: { replaceAction: {
type: Boolean, type: Boolean,
required: false, required: true,
default: false, default: false,
}, },
sales: { sales: {
type: Array, type: Array,
required: false, required: true,
default: () => [], default: () => [],
}, },
}); });
@ -36,6 +44,8 @@ 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,
})); }));

View File

@ -1,13 +1,17 @@
<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,
@ -31,7 +35,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('Item proposal') }}</span> <span class="text-h6 text-grey">{{ $t('itemProposal') }}</span>
<QSpace /> <QSpace />
<QBtn icon="close" flat round dense v-close-popup /> <QBtn icon="close" flat round dense v-close-popup />
</QCardSection> </QCardSection>

View File

@ -233,10 +233,10 @@ const ticketColumns = ref([
</span> </span>
</QTd> </QTd>
</template> </template>
<template #body-cell-client="{ value, row }"> <template #body-cell-client="{ row }">
<QTd auto-width> <QTd>
<span class="link"> <span class="link">
{{ value }} {{ row.clientFk }}
<CustomerDescriptorProxy :id="row?.clientFk" /> <CustomerDescriptorProxy :id="row?.clientFk" />
</span> </span>
</QTd> </QTd>

View File

@ -101,7 +101,7 @@ const onNextStep = async () => {
t('basicData.negativesConfirmMessage'), t('basicData.negativesConfirmMessage'),
submitWithNegatives, submitWithNegatives,
); );
else submit(); else await submit();
} }
}; };
</script> </script>

View File

@ -28,6 +28,7 @@ const props = defineProps({
onMounted(() => { onMounted(() => {
restoreTicket(); restoreTicket();
hasDocuware();
}); });
watch( watch(

View File

@ -1,7 +1,6 @@
<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,

View File

@ -768,6 +768,7 @@ 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">

View File

@ -3,7 +3,9 @@ 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';
const emit = defineEmits(['ticketTransfered']); import { displayResults } from 'src/pages/Ticket/Negative/composables/notifyResults';
const { notifyResults } = displayResults();
const emit = defineEmits(['ticketTransferred']);
const $props = defineProps({ const $props = defineProps({
ticket: { ticket: {
@ -16,13 +18,20 @@ 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];
await split(tickets, splitDate.value); const results = await split(tickets, splitDate.value);
emit('ticketTransfered', tickets); notifyResults(results, 'ticketFk');
emit('ticketTransferred', tickets);
}; };
</script> </script>
<template> <template>
<VnInputDate class="q-mr-sm" :label="$t('New date')" v-model="splitDate" clearable /> <VnInputDate
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">

View File

@ -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(['ticketTransfered']); const emit = defineEmits(['ticketTransferred']);
const $props = defineProps({ const $props = defineProps({
mana: { mana: {

View File

@ -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 Split from './TicketSplit.vue'; import TicketSplit from './TicketSplit.vue';
const emit = defineEmits(['ticketTransfered']); const emit = defineEmits(['ticketTransferred']);
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">
<Split <TicketSplit
ref="splitRef" ref="splitRef"
@splitSelectedRows="splitSelectedRows" @splitSelectedRows="splitSelectedRows"
:ticket="$props.ticket" :ticket="$props.ticket"

View File

@ -1,13 +1,11 @@
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.id, sales: [item.saleFk], date }); acc.push({ ticketFk: item.ticketFk, sales: [item.saleFk], date });
} }
return acc; return acc;
}, []); }, []);
@ -16,7 +14,5 @@ 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;
} }

View File

@ -23,7 +23,6 @@ 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();
@ -63,6 +62,7 @@ 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,21 +117,17 @@ const showItemProposal = () => {
sales: selectedRows, sales: selectedRows,
lastActiveTickets: selectedRows.map((row) => row.id), lastActiveTickets: selectedRows.map((row) => row.id),
}" }"
@ticket-transfered="reload" @ticket-transferred="reload"
></TicketTransferProxy> ></TicketTransferProxy>
</template> </template>
</QBtn> </QBtn>
<QBtn <QBtn
color="primary" color="primary"
@click="showProposalDialog = true" @click="showItemProposal"
:disable="selectedRows.length < 1" :disable="!(selectedRows.length === 1)"
data-cy="itemProposal" data-cy="itemProposal"
> >
<QIcon <QIcon name="import_export" class="rotate-90" />
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>
@ -139,7 +135,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>
@ -153,7 +149,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>
@ -167,7 +163,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"
> >

View File

@ -6,6 +6,7 @@ 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: {
@ -66,6 +67,7 @@ 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">
@ -77,12 +79,11 @@ 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>
<VnInput <VnInputDateTime
v-model="params.days" v-model="params.availabled"
:label="t('negative.days')" :label="t('negative.availabled')"
dense dense
filled filled
type="number"
@update:model-value=" @update:model-value="
(value) => { (value) => {
setUserParams(params); setUserParams(params);

View File

@ -21,14 +21,13 @@ 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(() => [
@ -65,15 +64,19 @@ const columns = computed(() => [
columnFilter: { columnFilter: {
component: 'input', component: 'input',
type: 'number', type: 'number',
columnClass: 'shrink', inWhere: false,
}, },
}, },
{ {
name: 'longName', name: 'longName',
align: 'center', align: 'left',
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,
@ -94,6 +97,11 @@ 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',
@ -155,7 +163,6 @@ 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"

View File

@ -5,7 +5,7 @@ import { computed, ref, watch } from 'vue';
import { useI18n } from 'vue-i18n'; import { useI18n } from 'vue-i18n';
import axios from 'axios'; import axios from 'axios';
import FetchData from 'src/components/FetchData.vue'; import FetchData from 'src/components/FetchData.vue';
import { toDate, toHour } from 'src/filters'; import { toDate } from 'src/filters';
import useNotify from 'src/composables/useNotify.js'; import useNotify from 'src/composables/useNotify.js';
import ZoneDescriptorProxy from 'pages/Zone/Card/ZoneDescriptorProxy.vue'; import ZoneDescriptorProxy from 'pages/Zone/Card/ZoneDescriptorProxy.vue';
import { useRoute } from 'vue-router'; import { useRoute } from 'vue-router';
@ -22,14 +22,6 @@ const $props = defineProps({
}, },
}); });
watch(
() => $props.filter,
(v) => {
filterLack.value.where = v;
tableRef.value.reload(filterLack);
},
);
const filterLack = ref({ const filterLack = ref({
include: [ include: [
{ {
@ -90,6 +82,7 @@ const columns = computed(() => [
}, },
{ {
name: 'alertLevelCode', name: 'alertLevelCode',
field: 'alertLevelCode',
label: t('negative.detail.state'), label: t('negative.detail.state'),
columnFilter: { columnFilter: {
name: 'alertLevelCode', name: 'alertLevelCode',
@ -158,7 +151,6 @@ const saveChange = async (field, { row }) => {
fetchItemLack.value.fetch(); fetchItemLack.value.fetch();
} catch (err) { } catch (err) {
console.error('Error saving changes', err); console.error('Error saving changes', err);
f;
} }
}; };
@ -171,7 +163,11 @@ function onBuysFetched(data) {
<FetchData <FetchData
ref="fetchItemLack" ref="fetchItemLack"
:url="`Tickets/itemLack`" :url="`Tickets/itemLack`"
:params="{ id: entityId }" :params="{
itemFk: entityId,
warehouseFk: $props.filter.warehouseFk,
availabled: Date.getCurrentDateTimeFormatted(),
}"
@on-fetch="(data) => (itemLack = data[0])" @on-fetch="(data) => (itemLack = data[0])"
auto-load auto-load
/> />
@ -207,6 +203,7 @@ function onBuysFetched(data) {
dense dense
:is-editable="true" :is-editable="true"
:row-click="false" :row-click="false"
:search-url="false"
:right-search="false" :right-search="false"
:right-search-icon="false" :right-search-icon="false"
v-model:selected="selectedRows" v-model:selected="selectedRows"
@ -214,7 +211,6 @@ function onBuysFetched(data) {
> >
<template #top-left> <template #top-left>
<div style="display: flex; align-items: center" v-if="itemLack"> <div style="display: flex; align-items: center" v-if="itemLack">
<!-- <VnImg :id="itemLack.itemFk" class="rounded image-wrapper"></VnImg> -->
<div class="flex column" style="align-items: center"> <div class="flex column" style="align-items: center">
<QBadge <QBadge
ref="badgeLackRef" ref="badgeLackRef"
@ -239,7 +235,7 @@ function onBuysFetched(data) {
<template #column-status="{ row }"> <template #column-status="{ row }">
<QTd style="min-width: 150px"> <QTd style="min-width: 150px">
<div class="icon-container"> <div class="flex items-start q-ma-sm">
<QIcon <QIcon
v-if="row.isBasket" v-if="row.isBasket"
name="vn:basket" name="vn:basket"
@ -266,9 +262,10 @@ function onBuysFetched(data) {
size="xs" size="xs"
> >
<QTooltip>{{ <QTooltip>{{
t('negative.detail.hasObservation') row?.note || t('negative.detail.hasObservation')
}}</QTooltip> </QIcon }}</QTooltip>
><QIcon </QIcon>
<QIcon
v-if="row.isRookie" v-if="row.isRookie"
name="vn:Person" name="vn:Person"
size="xs" size="xs"
@ -305,9 +302,9 @@ function onBuysFetched(data) {
</span> </span>
</template> </template>
<template #column-ticketFk="{ row }"> <template #column-ticketFk="{ row }">
<span class="q-pa-sm link"> <span class="link" @click.stop>
{{ row.id }} {{ row.ticketFk }}
<TicketDescriptorProxy :id="row.id" /> <TicketDescriptorProxy :id="row.ticketFk" />
</span> </span>
</template> </template>
<template #column-alertLevelCode="props"> <template #column-alertLevelCode="props">
@ -335,15 +332,6 @@ function onBuysFetched(data) {
</VnTable> </VnTable>
</template> </template>
<style lang="scss" scoped> <style lang="scss" scoped>
.icon-container {
display: grid;
grid-template-columns: repeat(3, 0.2fr);
row-gap: 5px; /* Ajusta el espacio entre los iconos según sea necesario */
}
.icon-container > * {
width: 100%;
height: auto;
}
.list-enter-active, .list-enter-active,
.list-leave-active { .list-leave-active {
transition: all 1s ease; transition: all 1s ease;

View File

@ -2,7 +2,9 @@
import { ref } from 'vue'; import { ref } from 'vue';
import axios from 'axios'; import axios from 'axios';
import VnSelect from 'src/components/common/VnSelect.vue'; import VnSelect from 'src/components/common/VnSelect.vue';
import notifyResults from 'src/utils/notifyResults'; import { displayResults } from 'src/pages/Ticket/Negative/composables/notifyResults';
const { notifyResults } = displayResults();
const emit = defineEmits(['update-item']); const emit = defineEmits(['update-item']);
const showChangeItemDialog = ref(false); const showChangeItemDialog = ref(false);
@ -37,7 +39,6 @@ const updateItem = async () => {
<template> <template>
<QCard class="q-pa-sm"> <QCard class="q-pa-sm">
<QCardSection class="row items-center justify-center column items-stretch"> <QCardSection class="row items-center justify-center column items-stretch">
{{ showChangeItemDialog }}
<span>{{ $t('negative.detail.modal.changeItem.title') }}</span> <span>{{ $t('negative.detail.modal.changeItem.title') }}</span>
<VnSelect <VnSelect
url="Items/WithName" url="Items/WithName"
@ -47,6 +48,7 @@ const updateItem = async () => {
option-label="name" option-label="name"
option-value="id" option-value="id"
v-model="newItem" v-model="newItem"
autofocus
> >
</VnSelect> </VnSelect>
</QCardSection> </QCardSection>

View File

@ -2,8 +2,9 @@
import { ref } from 'vue'; import { ref } from 'vue';
import axios from 'axios'; import axios from 'axios';
import VnInput from 'src/components/common/VnInput.vue'; import VnInput from 'src/components/common/VnInput.vue';
import notifyResults from 'src/utils/notifyResults'; import { displayResults } from 'src/pages/Ticket/Negative/composables/notifyResults';
const { notifyResults } = displayResults();
const showChangeQuantityDialog = ref(false); const showChangeQuantityDialog = ref(false);
const newQuantity = ref(null); const newQuantity = ref(null);
const $props = defineProps({ const $props = defineProps({
@ -16,15 +17,16 @@ const emit = defineEmits(['update-quantity']);
const updateQuantity = async () => { const updateQuantity = async () => {
try { try {
showChangeQuantityDialog.value = true; showChangeQuantityDialog.value = true;
const rowsToUpdate = $props.selectedRows.map(({ saleFk }) => const rowsToUpdate = $props.selectedRows.map(({ saleFk, ticketFk }) =>
axios.post(`Sales/${saleFk}/updateQuantity`, { axios.post(`Sales/${saleFk}/updateQuantity`, {
saleFk, saleFk,
ticketFk,
quantity: +newQuantity.value, quantity: +newQuantity.value,
}), }),
); );
const result = await Promise.allSettled(rowsToUpdate); const result = await Promise.allSettled(rowsToUpdate);
notifyResults(result, 'saleFk'); notifyResults(result, 'ticketFk');
emit('update-quantity', newQuantity.value); emit('update-quantity', newQuantity.value);
} catch (err) { } catch (err) {
@ -42,6 +44,7 @@ const updateQuantity = async () => {
:min="0" :min="0"
:label="$t('negative.detail.modal.changeQuantity.placeholder')" :label="$t('negative.detail.modal.changeQuantity.placeholder')"
v-model="newQuantity" v-model="newQuantity"
autofocus
/> />
</QCardSection> </QCardSection>
<QCardActions align="right"> <QCardActions align="right">

View File

@ -3,8 +3,9 @@ import { ref } from 'vue';
import axios from 'axios'; import axios from 'axios';
import VnSelect from 'src/components/common/VnSelect.vue'; import VnSelect from 'src/components/common/VnSelect.vue';
import FetchData from 'components/FetchData.vue'; import FetchData from 'components/FetchData.vue';
import notifyResults from 'src/utils/notifyResults'; import { displayResults } from 'src/pages/Ticket/Negative/composables/notifyResults';
const { notifyResults } = displayResults();
const emit = defineEmits(['update-state']); const emit = defineEmits(['update-state']);
const editableStates = ref([]); const editableStates = ref([]);
const showChangeStateDialog = ref(false); const showChangeStateDialog = ref(false);
@ -49,6 +50,7 @@ const updateState = async () => {
:options="editableStates" :options="editableStates"
option-label="name" option-label="name"
option-value="code" option-value="code"
autofocus
/> />
</QCardSection> </QCardSection>
<QCardActions align="right"> <QCardActions align="right">

View File

@ -0,0 +1,56 @@
import { Notify } from 'quasar';
import useOpenURL from 'src/composables/useOpenURL';
import { useI18n } from 'vue-i18n';
export function displayResults() {
const { t } = useI18n();
const createSuccessNotification = (id, path) => {
Notify.create({
type: 'positive',
message: t('globals.dataSaved'),
actions: [
{
label: t('globals.openDetail'),
color: 'white',
handler: () => useOpenURL(`#/ticket/${id}/${path}`),
},
],
});
};
const handleResult = (result, key, path) => {
if (result.status !== 'fulfilled') {
const data = JSON.parse(result.reason.config.data);
const error =
result.reason.response?.data?.error?.message ?? result.reason.message;
Notify.create({
type: 'negative',
message: `Ticket ${data[key]}: ${error}`,
});
return;
}
const data = JSON.parse(result.value.config.data);
let id = data[key];
if (result.value.data.status === 'noSplit') {
Notify.create({
type: 'warning',
message: `Ticket ${id}: ${t('negative.split.noSplit')}`,
});
return;
}
if (result.value.data.status === 'split') {
id = result.value.data.newTicket;
}
createSuccessNotification(id, path);
};
const notifyResults = (results, key, path = 'sale') =>
results.forEach((result) => handleResult(result, key, path));
return { notifyResults };
}

View File

@ -236,6 +236,7 @@ negative:
minTimed: minTimed minTimed: minTimed
negativeAction: Negative negativeAction: Negative
totalNegative: Total negatives totalNegative: Total negatives
availabled: Availabled
days: Days days: Days
buttonsUpdate: buttonsUpdate:
item: Item item: Item
@ -265,7 +266,7 @@ negative:
isRookie: Is rookie isRookie: Is rookie
turno: Turn line turno: Turn line
isBasket: Basket isBasket: Basket
hasObservation: Has substitution hasObservation: Has substitution note
hasToIgnore: VIP hasToIgnore: VIP
modal: modal:
changeItem: changeItem:
@ -288,3 +289,4 @@ negative:
newTicket: New ticket newTicket: New ticket
status: Result status: Result
message: Message message: Message
noSplit: No split

View File

@ -216,7 +216,8 @@ ticketList:
negative: negative:
hour: Hora hour: Hora
id: Id Articulo id: Id Articulo
longName: Articulo longName: Artículo
itemFk: Artículo
supplier: Productor supplier: Productor
colour: Color colour: Color
size: Medida size: Medida
@ -232,6 +233,7 @@ negative:
inkFk: Color inkFk: Color
timed: Hora timed: Hora
date: Fecha date: Fecha
availabled: Disponible
minTimed: Hora minTimed: Hora
type: Tipo type: Tipo
negativeAction: Negativo negativeAction: Negativo
@ -265,7 +267,7 @@ negative:
isRookie: Cliente nuevo isRookie: Cliente nuevo
turno: Linea turno turno: Linea turno
isBasket: Cesta isBasket: Cesta
hasObservation: Tiene sustitución hasObservation: Tiene nota de sustitución
hasToIgnore: VIP hasToIgnore: VIP
modal: modal:
changeItem: changeItem:
@ -284,10 +286,11 @@ negative:
title: Gestionar tickets spliteados title: Gestionar tickets spliteados
subTitle: Confir fecha y agencia subTitle: Confir fecha y agencia
split: split:
ticket: Ticket viejo ticket: Ticket origen
newTicket: Ticket nuevo newTicket: Ticket nuevo
status: Estado status: Estado
message: Mensaje message: Mensaje
noSplit: No se puede splitar
rounding: Redondeo rounding: Redondeo
noVerifiedData: Sin datos comprobados noVerifiedData: Sin datos comprobados
purchaseRequest: Petición de compra purchaseRequest: Petición de compra

View File

@ -1,19 +0,0 @@
import { Notify } from 'quasar';
export default function (results, key) {
results.forEach((result, index) => {
if (result.status === 'fulfilled') {
const data = JSON.parse(result.value.config.data);
Notify.create({
type: 'positive',
message: `Operación (${index + 1}) ${data[key]} completada con éxito.`,
});
} else {
const data = JSON.parse(result.reason.config.data);
Notify.create({
type: 'negative',
message: `Operación (${index + 1}) ${data[key]} fallida: ${result.reason.message}`,
});
}
});
}