-
+ |
{{ item.warehouse }}
|
-
+ |
{
diff --git a/src/pages/Order/Card/OrderDescriptor.vue b/src/pages/Order/Card/OrderDescriptor.vue
index 4d84a32fc22..a035971b0e4 100644
--- a/src/pages/Order/Card/OrderDescriptor.vue
+++ b/src/pages/Order/Card/OrderDescriptor.vue
@@ -4,13 +4,13 @@ import { useRoute } from 'vue-router';
import { useI18n } from 'vue-i18n';
import { toCurrency, toDate } from 'src/filters';
import { useState } from 'src/composables/useState';
-
import useCardDescription from 'src/composables/useCardDescription';
-import WorkerDescriptorProxy from 'src/pages/Worker/Card/WorkerDescriptorProxy.vue';
+
import CardDescriptor from 'components/ui/CardDescriptor.vue';
import VnLv from 'src/components/ui/VnLv.vue';
-import OrderDescriptorMenu from 'pages/Order/Card/OrderDescriptorMenu.vue';
import FetchData from 'components/FetchData.vue';
+import OrderDescriptorMenu from 'pages/Order/Card/OrderDescriptorMenu.vue';
+import WorkerDescriptorProxy from 'src/pages/Worker/Card/WorkerDescriptorProxy.vue';
const DEFAULT_ITEMS = 0;
@@ -25,6 +25,8 @@ const $props = defineProps({
const route = useRoute();
const state = useState();
const { t } = useI18n();
+const data = ref(useCardDescription());
+const getTotalRef = ref();
const entityId = computed(() => {
return $props.id || route.params.id;
@@ -57,11 +59,11 @@ const filter = {
],
};
-const data = ref(useCardDescription());
const setData = (entity) => {
if (!entity) return;
+ getTotalRef.value && getTotalRef.value.fetch();
data.value = useCardDescription(entity?.client?.name, entity?.id);
- state.set('OrderDescriptor', entity);
+ state.set('orderData', entity);
};
const getConfirmationValue = (isConfirmed) => {
@@ -69,13 +71,17 @@ const getConfirmationValue = (isConfirmed) => {
};
const total = ref(null);
+
+function ticketFilter(order) {
+ return JSON.stringify({ id: order.id });
+}
(total = response)"
- auto-load
/>
{{ t('order.summary.orderTicketList') }}
diff --git a/src/pages/Order/Card/OrderFilter.vue b/src/pages/Order/Card/OrderFilter.vue
index 347affb0423..be47eed547b 100644
--- a/src/pages/Order/Card/OrderFilter.vue
+++ b/src/pages/Order/Card/OrderFilter.vue
@@ -21,15 +21,13 @@ const salesPersonFilter = {
fields: ['id', 'nickname'],
};
const salesPersonList = ref(null);
-const sourceFilter = { fields: ['value'] };
-const sourceList = ref(null);
+const sourceList = ref([]);
(agencyList = data)"
@@ -37,7 +35,6 @@ const sourceList = ref(null);
(salesPersonList = data)"
:params="{ departmentCodes: ['VT'] }"
@@ -45,8 +42,7 @@ const sourceList = ref(null);
/>
(sourceList = data)"
auto-load
@@ -59,148 +55,88 @@ const sourceList = ref(null);
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- {{ opt.name }}
-
- {{ opt.nickname }},{{ opt.code }}
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
+
+
+
+
+
+
+
+ {{ opt.name }}
+
+ {{ opt.nickname }},{{ opt.code }}
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/src/pages/Order/OrderLines.vue b/src/pages/Order/Card/OrderLines.vue
similarity index 71%
rename from src/pages/Order/OrderLines.vue
rename to src/pages/Order/Card/OrderLines.vue
index 9814eaf3450..dab4a959ca1 100644
--- a/src/pages/Order/OrderLines.vue
+++ b/src/pages/Order/Card/OrderLines.vue
@@ -1,23 +1,26 @@
@@ -204,78 +235,80 @@ async function confirmOrder() {
auto-load
/>
(orderSummary.total = data)"
auto-load
/>
(orderSummary.vat = data)"
auto-load
/>
-
+
-
-
+
+
-
-
-
- {{ t('globals.noResults') }}
-
-
-
-
-
-
-
-
-
- {{ row?.item?.name }}
-
- {{ row?.item?.subName.toUpperCase() }}
-
-
-
-
-
-
-
-
-
- {{ t('confirm') }}
-
-
-
+
+
+
+
+
+
+
+
+
+ {{ row?.item?.name }}
+
+ {{ row?.item?.subName.toUpperCase() }}
+
+
+
+
+
+ {{ toCurrency(row.quantity * row.price) }}
+
+
+
+
+
+
+
+
+ {{ t('confirm') }}
+
+
+
+en:
+ summary: Summary
+ total: Total
+ boxes: Boxes
+ item: Item
+ quantity: Quantity
+ volume: m³ per quantity
+es:
+ summary: Resumen
+ total: Total
+ boxes: Cajas
+ item: Artículo
+ quantity: Cantidad
+ volume: m³ por cantidad
+
diff --git a/src/pages/Order/OrderList.vue b/src/pages/Order/OrderList.vue
index 945f61f3bb5..9870be9b300 100644
--- a/src/pages/Order/OrderList.vue
+++ b/src/pages/Order/OrderList.vue
@@ -3,19 +3,21 @@ import axios from 'axios';
import { useI18n } from 'vue-i18n';
import { computed, ref } from 'vue';
import { dashIfEmpty, toCurrency, toDate } from 'src/filters';
-import VnSearchbar from 'src/components/ui/VnSearchbar.vue';
import OrderSummary from 'pages/Order/Card/OrderSummary.vue';
import { useSummaryDialog } from 'src/composables/useSummaryDialog';
import VnTable from 'src/components/VnTable/VnTable.vue';
import VnInputDate from 'src/components/common/VnInputDate.vue';
import VnSelect from 'src/components/common/VnSelect.vue';
+import OrderSearchbar from './Card/OrderSearchbar.vue';
+import RightMenu from 'src/components/common/RightMenu.vue';
+import OrderFilter from './Card/OrderFilter.vue';
const { t } = useI18n();
const { viewSummary } = useSummaryDialog();
const tableRef = ref();
-const clientList = ref([]);
const agencyList = ref([]);
-const selectedAddress = ref();
+const addressesList = ref([]);
+const clientId = ref();
const columns = computed(() => [
{
@@ -29,7 +31,7 @@ const columns = computed(() => [
},
{
align: 'left',
- name: 'clientName',
+ name: 'clientFk',
label: t('module.customer'),
isTitle: true,
cardVisible: true,
@@ -41,20 +43,26 @@ const columns = computed(() => [
columnField: {
component: null,
},
+ format: (row) => row?.clientName,
},
{
align: 'left',
- name: 'name',
+ name: 'salesPersonFk',
label: t('module.salesPerson'),
- component: 'select',
- attrs: {
- url: 'Workers/activeWithInheritedRole',
- fields: ['id', 'name'],
- where: { role: 'salesPerson' },
- },
- columnField: {
- component: null,
+ columnFilter: {
+ component: 'select',
+ inWhere: true,
+ attrs: {
+ url: 'Workers/activeWithInheritedRole',
+ fields: ['id', 'name'],
+ where: { role: 'salesPerson' },
+ useLike: false,
+ optionValue: 'id',
+ optionLabel: 'name',
+ optionFilter: 'firstName',
+ },
},
+ format: (row) => row?.name,
},
{
align: 'left',
@@ -92,17 +100,21 @@ const columns = computed(() => [
},
{
align: 'left',
- name: 'agencyName',
+ name: 'agencyModeFk',
label: t('module.agency'),
- component: 'select',
+ format: (row) => row?.agencyName,
+ columnFilter: {
+ component: 'select',
+ attrs: {
+ url: 'agencyModes',
+ fields: ['id', 'name'],
+ find: {
+ value: 'agencyModeFk',
+ label: 'agencyName',
+ },
+ },
+ },
cardVisible: true,
- attrs: {
- url: 'Agencies',
- fields: ['id', 'name'],
- },
- columnField: {
- component: null,
- },
},
{
align: 'left',
@@ -125,22 +137,36 @@ const columns = computed(() => [
},
]);
-async function fetchClientAddress(id, data) {
- const clientData = await axios.get(`Clients/${id}`);
- selectedAddress.value = clientData.data.defaultAddressFk;
- data.addressId = selectedAddress.value;
+async function fetchClientAddress(id, formData) {
+ const { data } = await axios.get(`Clients/${id}`, {
+ params: { filter: { include: { relation: 'addresses' } } },
+ });
+ addressesList.value = data.addresses;
+ formData.addressId = data.defaultAddressFk;
+ fetchAgencies(formData);
+}
+
+async function fetchAgencies({ landed, addressId }) {
+ if (!landed || !addressId) return (agencyList.value = []);
+
+ const { data } = await axios.get('Agencies/landsThatDay', {
+ params: { addressFk: addressId, landed },
+ });
+ agencyList.value = data;
}
-
+
+
+
+
+
+
fetchClientAddress(id, data)"
/>
fetchAgencies(data)"
+ >
+
+
+
+
+ {{ scope.opt?.nickname }}: {{ scope.opt?.street }},
+ {{ scope.opt?.city }}
+
+
+
+
+ fetchAgencies(data)"
/>
-
diff --git a/src/pages/Order/OrderVolume.vue b/src/pages/Order/OrderVolume.vue
deleted file mode 100644
index 35d68de16fe..00000000000
--- a/src/pages/Order/OrderVolume.vue
+++ /dev/null
@@ -1,156 +0,0 @@
-
-
-
- (volumeSummary = data)"
- auto-load
- />
-
-
-
- {{ t('globals.noResults') }}
-
-
-
-
-
-
- loadVolumes(data)"
- >
-
-
-
-
-
-
-
-
-
-
-
- {{ row.item.subName }}
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-en:
- summary: Summary
- total: Total
- boxes: Boxes
- item: Item
- subName: Subname
- quantity: Quantity
- volume: m³ per quantity
-es:
- summary: Resumen
- total: Total
- boxes: Cajas
- item: Artículo
- subName: Subname
- quantity: Cantidad
- volume: m³ por cantidad
-
diff --git a/src/pages/Shelving/Card/ShelvingFilter.vue b/src/pages/Shelving/Card/ShelvingFilter.vue
index abc91373b56..0056ffaece0 100644
--- a/src/pages/Shelving/Card/ShelvingFilter.vue
+++ b/src/pages/Shelving/Card/ShelvingFilter.vue
@@ -3,6 +3,7 @@ import { ref } from 'vue';
import { useI18n } from 'vue-i18n';
import FetchData from 'components/FetchData.vue';
import VnFilterPanel from 'components/ui/VnFilterPanel.vue';
+import VnSelect from 'src/components/common/VnSelect.vue';
const { t } = useI18n();
const props = defineProps({
@@ -15,25 +16,13 @@ const props = defineProps({
const emit = defineEmits(['search']);
const workers = ref();
-const parkings = ref();
function setWorkers(data) {
workers.value = data;
}
-
-function setParkings(data) {
- parkings.value = data;
-}
-
-
-
-
-
-
+
-
-
-
-
-
+
diff --git a/src/pages/Shelving/Card/ShelvingForm.vue b/src/pages/Shelving/Card/ShelvingForm.vue
index dd1c4e4a2f2..aee6f7f3a93 100644
--- a/src/pages/Shelving/Card/ShelvingForm.vue
+++ b/src/pages/Shelving/Card/ShelvingForm.vue
@@ -1,12 +1,11 @@
-
{
:form-initial-data="defaultInitialData"
@on-data-saved="onSave"
>
-
+
- filter(value, update, parkingSelectFilter)
- "
:rules="validate('Shelving.parkingFk')"
- :input-debounce="0"
/>
diff --git a/src/pages/Ticket/Card/BasicData/TicketBasicDataForm.vue b/src/pages/Ticket/Card/BasicData/TicketBasicDataForm.vue
index 2d937346af0..28c6fcf157a 100644
--- a/src/pages/Ticket/Card/BasicData/TicketBasicDataForm.vue
+++ b/src/pages/Ticket/Card/BasicData/TicketBasicDataForm.vue
@@ -30,7 +30,6 @@ const { t } = useI18n();
const agencyFetchRef = ref(null);
const zonesFetchRef = ref(null);
-const clientsOptions = ref([]);
const warehousesOptions = ref([]);
const companiesOptions = ref([]);
const agenciesOptions = ref([]);
@@ -273,15 +272,6 @@ const redirectToCustomerAddress = () => {
onMounted(() => onFormModelInit());
- (clientsOptions = data)"
- auto-load
- />
(warehousesOptions = data)"
@@ -317,7 +307,9 @@ onMounted(() => onFormModelInit());
v-model="clientId"
option-value="id"
option-label="name"
- :options="clientsOptions"
+ url="Clients"
+ :fields="['id', 'name']"
+ sort-by="id"
hide-selected
map-options
:required="true"
diff --git a/src/pages/Ticket/Card/TicketCreateTracking.vue b/src/pages/Ticket/Card/TicketCreateTracking.vue
index 87ca8dd3f27..d692f550df9 100644
--- a/src/pages/Ticket/Card/TicketCreateTracking.vue
+++ b/src/pages/Ticket/Card/TicketCreateTracking.vue
@@ -17,9 +17,7 @@ const { t } = useI18n();
const state = useState();
const user = state.getUser();
const stateFetchDataRef = ref(null);
-
const statesOptions = ref([]);
-const workersOptions = ref([]);
const onStateFkChange = (formData) => (formData.userFk = user.value.id);
@@ -30,12 +28,6 @@ const onStateFkChange = (formData) => (formData.userFk = user.value.id);
auto-load
@on-fetch="(data) => (statesOptions = data)"
/>
- (workersOptions = data)"
- />
(formData.userFk = user.value.id);
{
switch (param) {
@@ -139,7 +137,9 @@ const columns = computed(() => [
filterValue: null,
event: getInputEvents,
attrs: {
- options: itemsOptions.value,
+ url: 'Items',
+ fields: ['id', 'name'],
+ 'sort-by': 'name ASC',
'option-value': 'id',
'option-label': 'name',
dense: true,
@@ -268,19 +268,12 @@ onMounted(async () => {
stateStore.rightDrawer = true;
const filteredColumns = columns.value.filter((col) => col.name !== 'history');
allColumnNames.value = filteredColumns.map((col) => col.name);
- // await expeditionsArrayData.fetch({ append: false });
});
onUnmounted(() => (stateStore.rightDrawer = false));
- (itemsOptions = data)"
- />
(stateStore.rightDrawer = false));
auto-load
@on-fetch="(data) => (isLocked = data)"
/>
- (itemsWithNameOptions = data)"
- />
(stateStore.rightDrawer = false));
diff --git a/src/pages/Ticket/Card/TicketSaleMoreActions.vue b/src/pages/Ticket/Card/TicketSaleMoreActions.vue
index 9ec6b303a41..f04a13c4e30 100644
--- a/src/pages/Ticket/Card/TicketSaleMoreActions.vue
+++ b/src/pages/Ticket/Card/TicketSaleMoreActions.vue
@@ -153,14 +153,22 @@ const setReserved = async (reserved) => {
};
const createRefund = async (withWarehouse) => {
- if (!props.sales) return;
+ if (!props.ticket) return;
- const salesIds = props.sales.map((sale) => sale.id);
- const params = { salesIds: salesIds, withWarehouse: withWarehouse, negative: true };
- const { data } = await axios.post('Sales/clone', params);
- const [refundTicket] = data;
- notify(t('refundTicketCreated', { ticketId: refundTicket.id }), 'positive');
- router.push({ name: 'TicketSale', params: { id: refundTicket.id } });
+ const params = {
+ ticketsIds: [props.ticket.id],
+ withWarehouse: withWarehouse,
+ negative: true,
+ };
+
+ try {
+ const { data } = await axios.post('Tickets/cloneAll', params);
+ const [refundTicket] = data;
+ notify(t('refundTicketCreated', { ticketId: refundTicket.id }), 'positive');
+ router.push({ name: 'TicketSale', params: { id: refundTicket.id } });
+ } catch (error) {
+ console.error(error);
+ }
};
@@ -244,7 +252,7 @@ const createRefund = async (withWarehouse) => {
- {{ t('Refund...') }}
+ {{ t('Refund') }}
@@ -279,7 +287,7 @@ es:
Add claim: Crear reclamación
Mark as reserved: Marcar como reservado
Unmark as reserved: Desmarcar como reservado
- Refund...: Abono...
+ Refund: Abono
with warehouse: con almacén
without warehouse: sin almacén
Claim out of time: Reclamación fuera de plazo
diff --git a/src/pages/Ticket/Card/TicketSaleTracking.vue b/src/pages/Ticket/Card/TicketSaleTracking.vue
index d519bc2c601..e699d2bfdf7 100644
--- a/src/pages/Ticket/Card/TicketSaleTracking.vue
+++ b/src/pages/Ticket/Card/TicketSaleTracking.vue
@@ -26,8 +26,6 @@ const saleTrackingFetchDataRef = ref(null);
const sales = ref([]);
const saleTrackings = ref([]);
const itemShelvignsSales = ref([]);
-const shelvingsOptions = ref([]);
-const parkingsOptions = ref([]);
const saleTrackingUrl = computed(() => `SaleTrackings/${route.params.id}/filter`);
const oldQuantity = ref(null);
@@ -330,12 +328,6 @@ const qCheckBoxController = (sale, action) => {
auto-load
@on-fetch="(data) => (sales = data)"
/>
- (shelvingsOptions = data)"
- />
- (parkingsOptions = data)" />
{
{
{
const handleCancelProgress = () => (cancelProgress.value = true);
onMounted(async () => {
- let today = Date.vnNew();
+ let today = Date.vnNew().toISOString();
const tomorrow = new Date(today);
tomorrow.setDate(tomorrow.getDate() + 1);
userParams.dateFuture = tomorrow;
diff --git a/src/pages/Ticket/TicketAdvanceFilter.vue b/src/pages/Ticket/TicketAdvanceFilter.vue
index c4548763a3d..ed61d94478a 100644
--- a/src/pages/Ticket/TicketAdvanceFilter.vue
+++ b/src/pages/Ticket/TicketAdvanceFilter.vue
@@ -55,7 +55,7 @@ onMounted(async () => await getItemPackingTypes());
:data-key="props.dataKey"
:search-button="true"
:hidden-tags="['search']"
- :unremovable-params="['warehouseFk', 'dateFuture', 'dateToAdvance']"
+ :un-removable-params="['warehouseFk', 'dateFuture', 'dateToAdvance']"
>
@@ -70,7 +70,6 @@ onMounted(async () => await getItemPackingTypes());
v-model="params.dateFuture"
:label="t('params.dateFuture')"
is-outlined
- @update:model-value="searchFn()"
/>
@@ -80,7 +79,6 @@ onMounted(async () => await getItemPackingTypes());
v-model="params.dateToAdvance"
:label="t('params.dateToAdvance')"
is-outlined
- @update:model-value="searchFn()"
/>
diff --git a/src/pages/Ticket/TicketFuture.vue b/src/pages/Ticket/TicketFuture.vue
index 2fec6dc189f..14a4a5ac898 100644
--- a/src/pages/Ticket/TicketFuture.vue
+++ b/src/pages/Ticket/TicketFuture.vue
@@ -8,6 +8,8 @@ import VnSelect from 'src/components/common/VnSelect.vue';
import TicketDescriptorProxy from 'src/pages/Ticket/Card/TicketDescriptorProxy.vue';
import VnSubToolbar from 'src/components/ui/VnSubToolbar.vue';
import VnSearchbar from 'src/components/ui/VnSearchbar.vue';
+import RightMenu from 'src/components/common/RightMenu.vue';
+import TicketFutureFilter from './TicketFutureFilter.vue';
import { dashIfEmpty, toCurrency } from 'src/filters';
import { useVnConfirm } from 'composables/useVnConfirm';
@@ -37,9 +39,9 @@ const exprBuilder = (param, value) => {
return { liters: value };
case 'lines':
return { lines: value };
- case 'ipt':
+ case 'iptColFilter':
return { ipt: { like: `%${value}%` } };
- case 'futureIpt':
+ case 'futureIptColFilter':
return { futureIpt: { like: `%${value}%` } };
case 'totalWithVat':
return { totalWithVat: value };
@@ -47,8 +49,8 @@ const exprBuilder = (param, value) => {
};
const userParams = reactive({
- futureDated: Date.vnNew(),
- originDated: Date.vnNew(),
+ futureDated: Date.vnNew().toISOString(),
+ originDated: Date.vnNew().toISOString(),
warehouseFk: user.value.warehouseFk,
});
@@ -83,6 +85,8 @@ const getInputEvents = (col) => {
};
};
+const tickets = computed(() => store.data);
+
const ticketColumns = computed(() => [
{
label: t('futureTickets.problems'),
@@ -121,7 +125,7 @@ const ticketColumns = computed(() => [
sortable: true,
columnFilter: {
component: VnSelect,
- filterParamKey: 'ipt',
+ filterParamKey: 'iptColFilter',
type: 'select',
filterValue: null,
event: getInputEvents,
@@ -214,7 +218,7 @@ const ticketColumns = computed(() => [
sortable: true,
columnFilter: {
component: VnSelect,
- filterParamKey: 'futureIpt',
+ filterParamKey: 'futureIptColFilter',
type: 'select',
filterValue: null,
event: getInputEvents,
@@ -305,9 +309,14 @@ onMounted(async () => {
+
+
+
+
+
+import { ref } from 'vue';
+import { useI18n } from 'vue-i18n';
+
+import FetchData from 'components/FetchData.vue';
+import VnFilterPanel from 'src/components/ui/VnFilterPanel.vue';
+import VnSelect from 'components/common/VnSelect.vue';
+import VnInputDate from 'src/components/common/VnInputDate.vue';
+import VnInput from 'src/components/common/VnInput.vue';
+
+import axios from 'axios';
+import { onMounted } from 'vue';
+
+const { t } = useI18n();
+const props = defineProps({
+ dataKey: {
+ type: String,
+ required: true,
+ },
+});
+
+const warehousesOptions = ref([]);
+const itemPackingTypes = ref([]);
+const stateOptions = ref([]);
+
+const getItemPackingTypes = async () => {
+ try {
+ const filter = {
+ where: { isActive: true },
+ };
+ const { data } = await axios.get('ItemPackingTypes', {
+ params: { filter: JSON.stringify(filter) },
+ });
+ itemPackingTypes.value = data.map((ipt) => ({
+ description: t(ipt.description),
+ code: ipt.code,
+ }));
+ } catch (error) {
+ console.error(error);
+ }
+};
+
+const getGroupedStates = async () => {
+ try {
+ const { data } = await axios.get('AlertLevels');
+ stateOptions.value = data.map((state) => ({
+ id: state.id,
+ name: t(`futureTickets.${state.code}`),
+ code: state.code,
+ }));
+ } catch (error) {
+ console.error(error);
+ }
+};
+
+onMounted(async () => {
+ getItemPackingTypes();
+ getGroupedStates();
+});
+
+
+
+ (warehousesOptions = data)"
+ auto-load
+ />
+
+
+
+ {{ t(`params.${tag.label}`) }}:
+ {{ formatFn(tag.value) }}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+en:
+ iptInfo: IPT
+ params:
+ originDated: Origin date
+ futureDated: Destination date
+ futureIpt: Destination IPT
+ ipt: Origin IPT
+ warehouseFk: Warehouse
+ litersMax: Max liters
+ linesMax: Max lines
+ state: Origin grouped state
+ futureState: Destination grouped state
+ problems: With problems
+es:
+ Horizontal: Horizontal
+ Vertical: Vertical
+ iptInfo: Encajado
+ params:
+ originDated: Fecha origen
+ futureDated: Fecha destino
+ futureIpt: IPT destino
+ ipt: IPT Origen
+ warehouseFk: Almacén
+ litersMax: Litros máx.
+ linesMax: Líneas máx.
+ state: Estado agrupado origen
+ futureState: Estado agrupado destino
+ problems: Con problemas
+
diff --git a/src/pages/Ticket/locale/en.yml b/src/pages/Ticket/locale/en.yml
index 10a8e1fa4a3..305228669d5 100644
--- a/src/pages/Ticket/locale/en.yml
+++ b/src/pages/Ticket/locale/en.yml
@@ -93,6 +93,11 @@ futureTickets:
moveTicketSuccess: Tickets moved successfully!
searchInfo: Search future tickets by date
futureTicket: Future tickets
+ FREE: Free
+ ON_PREVIOUS: ON_PREVIOUS
+ ON_PREPARATION: On preparation
+ PACKED: Packed
+ DELIVERED: Delivered
expedition:
id: Expedition
item: Item
diff --git a/src/pages/Ticket/locale/es.yml b/src/pages/Ticket/locale/es.yml
index a80692bfe7e..20b4edf1687 100644
--- a/src/pages/Ticket/locale/es.yml
+++ b/src/pages/Ticket/locale/es.yml
@@ -140,6 +140,11 @@ futureTickets:
moveTicketSuccess: Tickets movidos correctamente
searchInfo: Buscar tickets por fecha
futureTicket: Tickets a futuro
+ FREE: Libre
+ ON_PREVIOUS: ON_PREVIOUS
+ ON_PREPARATION: En preparación
+ PACKED: Encajado
+ DELIVERED: Servido
ticketSale:
id: Id
visible: Visible
diff --git a/src/pages/Travel/ExtraCommunityFilter.vue b/src/pages/Travel/ExtraCommunityFilter.vue
index c51151451f0..a8bbde75bf1 100644
--- a/src/pages/Travel/ExtraCommunityFilter.vue
+++ b/src/pages/Travel/ExtraCommunityFilter.vue
@@ -20,7 +20,6 @@ const props = defineProps({
const warehousesOptions = ref([]);
const continentsOptions = ref([]);
const agenciesOptions = ref([]);
-const suppliersOptions = ref([]);
const warehousesByContinent = ref({});
const add = (paramsObj, key) => {
@@ -76,12 +75,6 @@ warehouses();
@on-fetch="(data) => (agenciesOptions = data)"
auto-load
/>
- (suppliersOptions = data)"
- auto-load
- />
-
@@ -220,7 +213,7 @@ warehouses();
{
+ showEditPhotoForm.value = !showEditPhotoForm.value;
+};
const entityId = computed(() => {
return $props.id || route.params.id;
@@ -99,7 +105,9 @@ const handleExcluded = async () => {
workerExcluded.value = !workerExcluded.value;
};
-
+const handlePhotoUpdated = (evt = false) => {
+ image.value.reload(evt);
+};
const refetch = async () => await cardDescriptorRef.value.getData();
@@ -144,27 +152,49 @@ const refetch = async () => await cardDescriptorRef.value.getData();
-
-
-
-
-
-
-
-
- {{ t('worker.imageNotFound') }}
+
+
+
+
+
+
+
+
+
+ {{ t('worker.imageNotFound') }}
+
-
-
-
+
+
+
+
+
+
+
diff --git a/src/pages/Zone/Card/ZoneCard.vue b/src/pages/Zone/Card/ZoneCard.vue
index 02ec12fe71d..b81ee903984 100644
--- a/src/pages/Zone/Card/ZoneCard.vue
+++ b/src/pages/Zone/Card/ZoneCard.vue
@@ -10,11 +10,6 @@ const { t } = useI18n();
const route = useRoute();
const routeName = computed(() => route.name);
-const customRouteRedirectName = computed(() => {
- if (routeName.value === 'ZoneLocations') return null;
- return routeName.value;
-});
-const searchbarMakeFetch = computed(() => routeName.value !== 'ZoneEvents');
const searchBarDataKeys = {
ZoneWarehouses: 'ZoneWarehouses',
ZoneSummary: 'ZoneSummary',
diff --git a/src/router/index.js b/src/router/index.js
index faa3ab5d42c..686da2dde9c 100644
--- a/src/router/index.js
+++ b/src/router/index.js
@@ -46,7 +46,7 @@ export { Router };
export default route(function (/* { store, ssrContext } */) {
Router.beforeEach(async (to, from, next) => {
const { isLoggedIn } = session;
- const outLayout = ['Login', 'TwoFactor', 'VerifyEmail'];
+ const outLayout = Router.options.routes[0].children.map((r) => r.name);
if (!isLoggedIn() && !outLayout.includes(to.name)) {
return next({ name: 'Login', query: { redirect: to.fullPath } });
}
diff --git a/src/router/modules/account.js b/src/router/modules/account.js
index 3faa00fbc5d..cfec2b95d0b 100644
--- a/src/router/modules/account.js
+++ b/src/router/modules/account.js
@@ -65,13 +65,13 @@ export default {
component: () => import('src/pages/Account/AccountAliasList.vue'),
},
{
- path: 'connections',
- name: 'AccountConnections',
+ path: 'acls',
+ name: 'AccountAcls',
meta: {
- title: 'connections',
+ title: 'acls',
icon: 'check',
},
- component: () => import('src/pages/Account/AccountConnections.vue'),
+ component: () => import('src/pages/Account/AccountAcls.vue'),
},
{
path: 'accounts',
@@ -104,13 +104,13 @@ export default {
component: () => import('src/pages/Account/AccountSamba.vue'),
},
{
- path: 'acls',
- name: 'AccountAcls',
+ path: 'connections',
+ name: 'AccountConnections',
meta: {
- title: 'acls',
- icon: 'check',
+ title: 'connections',
+ icon: 'share',
},
- component: () => import('src/pages/Account/AccountAcls.vue'),
+ component: () => import('src/pages/Account/AccountConnections.vue'),
},
{
path: 'acl-form',
diff --git a/src/router/modules/invoiceIn.js b/src/router/modules/invoiceIn.js
index 906db8a5899..64c7b23e972 100644
--- a/src/router/modules/invoiceIn.js
+++ b/src/router/modules/invoiceIn.js
@@ -1,5 +1,5 @@
import { RouterView } from 'vue-router';
-
+import { setRectificative } from 'src/pages/InvoiceIn/composables/setRectificative';
export default {
path: '/invoice-in',
name: 'InvoiceIn',
@@ -63,6 +63,10 @@ export default {
path: ':id',
component: () => import('src/pages/InvoiceIn/Card/InvoiceInCard.vue'),
redirect: { name: 'InvoiceInSummary' },
+ beforeEnter: async (to, from, next) => {
+ await setRectificative(to);
+ next();
+ },
children: [
{
name: 'InvoiceInSummary',
diff --git a/src/router/modules/order.js b/src/router/modules/order.js
index a2b874cc679..9ccdb820b88 100644
--- a/src/router/modules/order.js
+++ b/src/router/modules/order.js
@@ -63,7 +63,7 @@ export default {
title: 'basicData',
icon: 'vn:settings',
},
- component: () => import('src/pages/Order/Card/OrderForm.vue'),
+ component: () => import('src/pages/Order/Card/OrderBasicData.vue'),
},
{
name: 'OrderCatalog',
@@ -72,7 +72,7 @@ export default {
title: 'catalog',
icon: 'vn:basket',
},
- component: () => import('src/pages/Order/OrderCatalog.vue'),
+ component: () => import('src/pages/Order/Card/OrderCatalog.vue'),
},
{
name: 'OrderVolume',
@@ -81,7 +81,7 @@ export default {
title: 'volume',
icon: 'vn:volume',
},
- component: () => import('src/pages/Order/OrderVolume.vue'),
+ component: () => import('src/pages/Order/Card/OrderVolume.vue'),
},
{
name: 'OrderLines',
@@ -90,7 +90,7 @@ export default {
title: 'lines',
icon: 'vn:lines',
},
- component: () => import('src/pages/Order/OrderLines.vue'),
+ component: () => import('src/pages/Order/Card/OrderLines.vue'),
},
],
},
diff --git a/src/router/routes.js b/src/router/routes.js
index 805eefb8cba..cced308b5a3 100644
--- a/src/router/routes.js
+++ b/src/router/routes.js
@@ -46,6 +46,18 @@ const routes = [
meta: { title: 'verifyEmail' },
component: () => import('../pages/Login/VerifyEmail.vue'),
},
+ {
+ path: '/recoverPassword',
+ name: 'RecoverPassword',
+ meta: { title: 'recoverPassword' },
+ component: () => import('../pages/Login/RecoverPassword.vue'),
+ },
+ {
+ path: '/resetPassword',
+ name: 'ResetPassword',
+ meta: { title: 'resetPassword' },
+ component: () => import('../pages/Login/ResetPassword.vue'),
+ },
],
},
{
diff --git a/src/stores/invoiceOutGlobal.js b/src/stores/invoiceOutGlobal.js
index bb9a3d376c6..42acac01395 100644
--- a/src/stores/invoiceOutGlobal.js
+++ b/src/stores/invoiceOutGlobal.js
@@ -19,6 +19,7 @@ export const useInvoiceOutGlobalStore = defineStore({
maxShipped: null,
clientId: null,
printer: null,
+ serialType: null,
},
addresses: [],
minInvoicingDate: null,
@@ -100,6 +101,7 @@ export const useInvoiceOutGlobalStore = defineStore({
maxShipped: new Date(formData.maxShipped),
clientId: formData.clientId ? formData.clientId : null,
companyFk: formData.companyFk,
+ serialType: formData.serialType,
};
this.validateMakeInvoceParams(params, clientsToInvoice);
@@ -152,7 +154,13 @@ export const useInvoiceOutGlobalStore = defineStore({
);
throw new Error('Invoice date in the future');
}
-
+ if (!params.serialType) {
+ notify(
+ 'invoiceOut.globalInvoices.errors.chooseValidSerialType',
+ 'negative'
+ );
+ throw new Error('Invalid Serial Type');
+ }
if (!params.companyFk) {
notify('invoiceOut.globalInvoices.errors.chooseValidCompany', 'negative');
throw new Error('Invalid company');
@@ -180,6 +188,7 @@ export const useInvoiceOutGlobalStore = defineStore({
invoiceDate: new Date(formData.invoiceDate),
maxShipped: new Date(formData.maxShipped),
companyFk: formData.companyFk,
+ serialType: formData.serialType,
};
this.status = 'invoicing';
@@ -191,12 +200,7 @@ export const useInvoiceOutGlobalStore = defineStore({
this.addressIndex++;
this.isInvoicing = false;
} catch (err) {
- if (
- err &&
- err.response &&
- err.response.status >= 400 &&
- err.response.status < 500
- ) {
+ if (err?.response?.status >= 400 && err?.response?.status < 500) {
this.invoiceClientError(address, err.response?.data?.error?.message);
return;
} else {
@@ -243,7 +247,7 @@ export const useInvoiceOutGlobalStore = defineStore({
params,
});
- if (data.data && data.data.error) throw new Error();
+ if (data?.data?.error) throw new Error();
const status = exportFile('negativeBases.csv', data, {
encoding: 'windows-1252',
diff --git a/src/stores/useNavigationStore.js b/src/stores/useNavigationStore.js
index 961e8037718..ad671a7f9bc 100644
--- a/src/stores/useNavigationStore.js
+++ b/src/stores/useNavigationStore.js
@@ -56,6 +56,7 @@ export const useNavigationStore = defineStore('navigationStore', () => {
function addMenuItem(module, route, parent) {
const { meta } = route;
let { menuChildren = null } = meta;
+ if (meta.hidden) return;
if (menuChildren)
menuChildren = menuChildren.map(({ name, title, icon }) => ({
name,
diff --git a/test/cypress/integration/invoiceIn/invoiceInBasicData.spec.js b/test/cypress/integration/invoiceIn/invoiceInBasicData.spec.js
index 77a11969baf..e1939fe5a4a 100644
--- a/test/cypress/integration/invoiceIn/invoiceInBasicData.spec.js
+++ b/test/cypress/integration/invoiceIn/invoiceInBasicData.spec.js
@@ -36,8 +36,7 @@ describe('InvoiceInBasicData', () => {
});
it('should throw an error creating a new dms if a file is not attached', () => {
- cy.get(formInputs).eq(5).click();
- cy.get(formInputs).eq(5).type('{selectall}{backspace}');
+ cy.get(formInputs).eq(7).type('{selectall}{backspace}');
cy.get(documentBtns).eq(0).click();
cy.get(dialogActionBtns).eq(1).click();
cy.get('.q-notification__message').should(
diff --git a/test/cypress/integration/invoiceIn/invoiceInVat.spec.js b/test/cypress/integration/invoiceIn/invoiceInVat.spec.js
index 018ae7a5316..b84d743d11b 100644
--- a/test/cypress/integration/invoiceIn/invoiceInVat.spec.js
+++ b/test/cypress/integration/invoiceIn/invoiceInVat.spec.js
@@ -3,13 +3,14 @@ describe('InvoiceInVat', () => {
const thirdRow = 'tbody > :nth-child(3)';
const firstLineVat = 'tbody > :nth-child(1) > :nth-child(4)';
const dialogInputs = '.q-dialog label input';
- const dialogBtns = '.q-dialog button';
- const acrossInput = 'tbody tr:nth-child(1) td:nth-child(2) .default-icon';
+ const addBtn = 'tbody tr:nth-child(1) td:nth-child(2) .--add-icon';
const randomInt = Math.floor(Math.random() * 100);
beforeEach(() => {
cy.login('developer');
cy.visit(`/#/invoice-in/1/vat`);
+ cy.intercept('GET', '/api/InvoiceIns/1/getTotals').as('lastCall');
+ cy.wait('@lastCall');
});
it('should edit the sage iva', () => {
@@ -26,22 +27,15 @@ describe('InvoiceInVat', () => {
});
it('should remove the first line', () => {
- cy.removeRow(2);
- });
-
- it('should throw an error if there are fields undefined', () => {
- cy.get(acrossInput).click();
- cy.get(dialogBtns).eq(2).click();
- cy.get('.q-notification__message').should('have.text', "The code can't be empty");
+ cy.removeRow(1);
});
it('should correctly handle expense addition', () => {
- cy.get(acrossInput).click();
+ cy.get(addBtn).click();
cy.get(dialogInputs).eq(0).type(randomInt);
- cy.get(dialogInputs).eq(1).click();
cy.get(dialogInputs).eq(1).type('This is a dummy expense');
- cy.get(dialogBtns).eq(2).click();
- cy.get('.q-notification__message').should('have.text', 'Data saved');
+ cy.get('button[type="submit"]').click();
+ cy.get('.q-notification__message').should('have.text', 'Data created');
});
});
diff --git a/test/cypress/integration/login.spec.js b/test/cypress/integration/outLogin/login.spec.js
similarity index 100%
rename from test/cypress/integration/login.spec.js
rename to test/cypress/integration/outLogin/login.spec.js
diff --git a/test/cypress/integration/logout.spec.js b/test/cypress/integration/outLogin/logout.spec.js
similarity index 100%
rename from test/cypress/integration/logout.spec.js
rename to test/cypress/integration/outLogin/logout.spec.js
diff --git a/test/cypress/integration/outLogin/recoverPassword.spec.js b/test/cypress/integration/outLogin/recoverPassword.spec.js
new file mode 100755
index 00000000000..eec81b661ef
--- /dev/null
+++ b/test/cypress/integration/outLogin/recoverPassword.spec.js
@@ -0,0 +1,54 @@
+///
+describe('Recover Password', () => {
+ const username = 'trainee';
+ beforeEach(() => {
+ cy.visit('/#/login');
+ cy.get('#switchLanguage').click();
+ cy.get('.q-menu > :nth-child(1) > .q-item').click();
+ });
+
+ it('should go to recover password section and send notification', () => {
+ cy.get('input[aria-label="Username"]').type(username);
+ cy.get(`a[href="#/recoverPassword?user=${username}"]`).click();
+
+ cy.waitForElement('input[aria-label="User or recovery email"]');
+ cy.get('input[aria-label="User or recovery email"]').should(
+ 'have.value',
+ username
+ );
+
+ cy.get('button[type="submit"]').click();
+ cy.get('.q-notification__message').should('have.text', 'Notification sent');
+ });
+
+ it('should change password to user', () => {
+ // Get token from mail
+ cy.request(
+ `http://localhost:3000/api/Mails?filter=%7B%22where%22%3A%20%7B%22receiver%22%3A%20%22${username}%40mydomain.com%22%7D%2C%20%22order%22%3A%20%5B%22id%20DESC%22%5D%7D&access_token=DEFAULT_TOKEN`
+ ).then((response) => {
+ const regex = /access_token=([a-zA-Z0-9]+)/;
+ const [match] = response.body[0].body.match(regex);
+
+ const resetUrl = '/#/resetPassword?' + match;
+ cy.visit(resetUrl);
+ });
+
+ // Change password
+ const newPassword = 'test.1234';
+ cy.waitForElement('input[aria-label="Password"]');
+ cy.waitForElement('input[aria-label="Repeat password"]');
+ cy.get('input[aria-label="Password"]').type(newPassword);
+ cy.get('input[aria-label="Repeat password"]').type(newPassword);
+
+ cy.get('button[type="submit"]').click();
+ cy.get('.q-notification__message').should('have.text', 'Password changed');
+
+ // Try to login successfully
+ cy.get('input[aria-label="Username"]').type(username);
+ cy.get('input[aria-label="Password"]').type(newPassword);
+ cy.get('button[type="submit"]').click();
+ cy.url().should('contain', '/dashboard');
+
+ // ❗The password cannot be returned because "nightmare" does not meet the requirements
+ });
+});
diff --git a/test/cypress/integration/outLogin/twoFactor.spec.js b/test/cypress/integration/outLogin/twoFactor.spec.js
new file mode 100755
index 00000000000..4d8561f0f12
--- /dev/null
+++ b/test/cypress/integration/outLogin/twoFactor.spec.js
@@ -0,0 +1,56 @@
+///
+describe('Two Factor', () => {
+ const username = 'sysadmin';
+ const userId = 66;
+ beforeEach(() => {
+ cy.visit('/#/login');
+ cy.get('#switchLanguage').click();
+ cy.get('.q-menu > :nth-child(1) > .q-item').click();
+ });
+
+ it('should enable two factor to sysadmin', () => {
+ cy.request(
+ 'PATCH',
+ `http://localhost:3000/api/VnUsers/${userId}/update-user?access_token=DEFAULT_TOKEN`,
+ { twoFactor: 'email' }
+ );
+ });
+
+ it('should fail when login with incorrect two factor', () => {
+ cy.get('input[aria-label="Username"]').type(username);
+ cy.get('input[aria-label="Password"]').type('nightmare');
+ cy.get('button[type="submit"]').click();
+ cy.get('.q-notification__message').should(
+ 'have.text',
+ 'Two-factor verification required'
+ );
+
+ cy.get('input[type="text"]').type('123456');
+ cy.get('button[type="submit"]').click();
+ cy.url().should('contain', '/twoFactor');
+ });
+
+ it('should login with correct two factor', () => {
+ cy.get('input[aria-label="Username"]').type(username);
+ cy.get('input[aria-label="Password"]').type('nightmare');
+ cy.get('button[type="submit"]').click();
+ cy.get('.q-notification__message').should(
+ 'have.text',
+ 'Two-factor verification required'
+ );
+
+ // Get code from mail
+ cy.request(
+ `http://localhost:3000/api/Mails?filter=%7B%22where%22%3A%20%7B%22receiver%22%3A%20%22${username}%40mydomain.com%22%7D%2C%20%22order%22%3A%20%5B%22id%20DESC%22%5D%7D&access_token=DEFAULT_TOKEN`
+ ).then((response) => {
+ const tempDiv = document.createElement('div');
+ tempDiv.innerHTML = response.body[0].body;
+ const codeElement = tempDiv.querySelector('.code');
+ const code = codeElement ? codeElement.textContent.trim() : null;
+
+ cy.get('input[type="text"]').type(code);
+ cy.get('button[type="submit"]').click();
+ cy.url().should('contain', '/dashboard');
+ });
+ });
+});
diff --git a/test/cypress/integration/ticket/ticketDescriptor.spec.js b/test/cypress/integration/ticket/ticketDescriptor.spec.js
index fc920f346bd..882aab157ff 100644
--- a/test/cypress/integration/ticket/ticketDescriptor.spec.js
+++ b/test/cypress/integration/ticket/ticketDescriptor.spec.js
@@ -16,7 +16,7 @@ describe('Ticket descriptor', () => {
cy.openActionsDescriptor();
cy.get(toCloneOpt).click();
cy.clickConfirm();
- cy.get(warehouseValue).contains('-');
+ cy.get(warehouseValue).contains('Warehouse One');
cy.get(summaryHeader)
.invoke('text')
.then((text) => {
diff --git a/test/vitest/__tests__/pages/Login/Login.spec.js b/test/vitest/__tests__/pages/Login/Login.spec.js
index 5ab4cee9e98..e90a8ee535b 100644
--- a/test/vitest/__tests__/pages/Login/Login.spec.js
+++ b/test/vitest/__tests__/pages/Login/Login.spec.js
@@ -44,10 +44,6 @@ describe('Login', () => {
it('should not set the token into session if any error occurred', async () => {
vi.spyOn(axios, 'post').mockReturnValue({ data: null });
- vi.spyOn(vm.quasar, 'notify');
-
await vm.onSubmit();
-
- expect(vm.quasar.notify).not.toHaveBeenCalled();
});
});
|