-
-
-
-
- #{{ scope.opt?.id }}
- {{ scope.opt?.nickname }}
-
-
-
-
-
O
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ #{{ scope.opt?.id }} -
+ {{ scope.opt?.name }}
+
+
+
+
+
+
+
+
+
+
+
+
+ {{
+ `${
+ !scope.opt?.isActive
+ ? t('inactive')
+ : ''
+ } `
+ }}
+ {{
+ scope.opt?.nickname
+ }}
+
+ , {{ scope.opt?.street }},
+ {{ scope.opt?.city }},
+ {{
+ scope.opt?.province?.name
+ }}
+ -
+ {{
+ scope.opt?.agencyMode
+ ?.name
+ }}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {{ scope.opt?.code }} -
+ {{ scope.opt?.description }}
+
+
+
+
+
+
+
+
+
+
+
+
-
-
-
-
-
+
+
-en:
- searchInvoice: Search issued invoice
- fileDenied: Browser denied file download...
- fileAllowed: Successful download of CSV file
- youCanSearchByInvoiceReference: You can search by invoice reference
- createInvoice: Make invoice
- Create manual invoice: Create manual invoice
-es:
- searchInvoice: Buscar factura emitida
- fileDenied: El navegador denegó la descarga de archivos...
- fileAllowed: Descarga exitosa de archivo CSV
- youCanSearchByInvoiceReference: Puedes buscar por referencia de la factura
- createInvoice: Crear factura
- Create manual invoice: Crear factura manual
+ en:
+ invoiceId: Invoice ID
+ youCanSearchByInvoiceReference: You can search by invoice reference
+ createManualInvoice: Create Manual Invoice
+ inactive: (Inactive)
+
+ es:
+ invoiceId: ID de factura
+ youCanSearchByInvoiceReference: Puedes buscar por referencia de la factura
+ createManualInvoice: Crear factura manual
+ inactive: (Inactivo)
diff --git a/src/pages/InvoiceOut/locale/en.yml b/src/pages/InvoiceOut/locale/en.yml
index 5ad92ed09..8cefe8bdc 100644
--- a/src/pages/InvoiceOut/locale/en.yml
+++ b/src/pages/InvoiceOut/locale/en.yml
@@ -2,6 +2,7 @@ invoiceOutModule:
customer: Client
amount: Amount
company: Company
+ address: Address
invoiceOutList:
tableVisibleColumns:
id: ID
@@ -15,11 +16,11 @@ invoiceOutList:
DownloadPdf: Download PDF
InvoiceOutSummary: Summary
negativeBases:
- country: Country
- clientId: Client ID
- base: Base
- ticketId: Ticket
- active: Active
- hasToInvoice: Has to invoice
- verifiedData: Verified data
- commercial: Commercial
\ No newline at end of file
+ country: Country
+ clientId: Client ID
+ base: Base
+ ticketId: Ticket
+ active: Active
+ hasToInvoice: Has to invoice
+ verifiedData: Verified data
+ commercial: Commercial
diff --git a/src/pages/InvoiceOut/locale/es.yml b/src/pages/InvoiceOut/locale/es.yml
index 192f5b26f..bf5126641 100644
--- a/src/pages/InvoiceOut/locale/es.yml
+++ b/src/pages/InvoiceOut/locale/es.yml
@@ -4,6 +4,7 @@ invoiceOutModule:
customer: Cliente
amount: Importe
company: Empresa
+ address: Consignatario
invoiceOutList:
tableVisibleColumns:
id: ID
diff --git a/src/pages/Monitor/Ticket/MonitorTickets.vue b/src/pages/Monitor/Ticket/MonitorTickets.vue
index 253316a09..e5fea3003 100644
--- a/src/pages/Monitor/Ticket/MonitorTickets.vue
+++ b/src/pages/Monitor/Ticket/MonitorTickets.vue
@@ -15,6 +15,7 @@ import { toCurrency, dateRange, dashIfEmpty } from 'src/filters';
import RightMenu from 'src/components/common/RightMenu.vue';
import MonitorTicketSearchbar from './MonitorTicketSearchbar.vue';
import MonitorTicketFilter from './MonitorTicketFilter.vue';
+import TicketProblems from 'src/components/TicketProblems.vue';
const DEFAULT_AUTO_REFRESH = 2 * 60 * 1000; // 2min in ms
const { t } = useI18n();
@@ -23,9 +24,15 @@ const tableRef = ref(null);
const provinceOpts = ref([]);
const stateOpts = ref([]);
const zoneOpts = ref([]);
-const visibleColumns = ref([]);
const { viewSummary } = useSummaryDialog();
+
const [from, to] = dateRange(Date.vnNew());
+const stateColors = {
+ notice: 'info',
+ success: 'positive',
+ warning: 'warning',
+ alert: 'negative',
+};
function exprBuilder(param, value) {
switch (param) {
@@ -220,7 +227,7 @@ const columns = computed(() => [
{
title: t('salesTicketsTable.goToLines'),
icon: 'vn:lines',
- color: 'priamry',
+ color: 'primary',
action: (row) => openTab(row.id),
isPrimary: true,
attrs: {
@@ -231,7 +238,7 @@ const columns = computed(() => [
{
title: t('salesTicketsTable.preview'),
icon: 'preview',
- color: 'priamry',
+ color: 'primary',
action: (row) => viewSummary(row.id, TicketSummary),
isPrimary: true,
attrs: {
@@ -249,10 +256,10 @@ const getBadgeAttrs = (date) => {
let timeTicket = new Date(date);
timeTicket.setHours(0, 0, 0, 0);
- let comparation = today - timeTicket;
+ let timeDiff = today - timeTicket;
- if (comparation == 0) return { color: 'warning', 'text-color': 'black' };
- if (comparation < 0) return { color: 'success', 'text-color': 'black' };
+ if (timeDiff == 0) return { color: 'warning', 'text-color': 'black' };
+ if (timeDiff < 0) return { color: 'success', 'text-color': 'black' };
return { color: 'transparent', 'text-color': 'white' };
};
@@ -267,13 +274,6 @@ const autoRefreshHandler = (value) => {
}
};
-const stateColors = {
- notice: 'info',
- success: 'positive',
- warning: 'warning',
- alert: 'negative',
-};
-
const totalPriceColor = (ticket) => {
const total = parseInt(ticket.totalWithVat);
if (total > 0 && total < 50) return 'warning';
@@ -281,10 +281,10 @@ const totalPriceColor = (ticket) => {
const formatShippedDate = (date) => {
if (!date) return '-';
- const split1 = date.split('T');
- const [year, month, day] = split1[0].split('-');
- const _date = new Date(year, month - 1, day);
- return toDateFormat(_date);
+ const dateSplit = date.split('T');
+ const [year, month, day] = dateSplit[0].split('-');
+ const newDate = new Date(year, month - 1, day);
+ return toDateFormat(newDate);
};
const openTab = (id) =>
@@ -332,7 +332,6 @@ const openTab = (id) =>
:expr-builder="exprBuilder"
:offset="50"
:columns="columns"
- :visible-columns="visibleColumns"
:right-search="false"
default-mode="table"
auto-load
@@ -362,61 +361,7 @@ const openTab = (id) =>
-
-
- {{ $t('salesTicketsTable.noVerifiedData') }}
-
-
- {{ $t('salesTicketsTable.purchaseRequest') }}
-
-
- {{ $t('salesTicketsTable.notVisible') }}
-
-
- {{ $t('salesTicketsTable.clientFrozen') }}
-
-
- {{ $t('salesTicketsTable.risk') }}: {{ row.risk }}
-
-
- {{ $t('salesTicketsTable.componentLack') }}
-
-
- {{ $t('salesTicketsTable.tooLittle') }}
-
-
+
@@ -471,7 +416,7 @@ const openTab = (id) =>
-
+
{{ row.zoneName }}
diff --git a/src/pages/Order/Card/CatalogFilterValueDialog.vue b/src/pages/Order/Card/CatalogFilterValueDialog.vue
index 53bb87f8d..b91e7d229 100644
--- a/src/pages/Order/Card/CatalogFilterValueDialog.vue
+++ b/src/pages/Order/Card/CatalogFilterValueDialog.vue
@@ -49,7 +49,7 @@ const getSelectedTagValues = async (tag) => {
-
+
{
:emit-value="false"
use-input
@update:model-value="getSelectedTagValues"
+ data-cy="catalogFilterValueDialogTagSelect"
/>
{
:disable="!value"
is-outlined
class="col"
+ data-cy="catalogFilterValueDialogValueInput"
/>
-
+
diff --git a/src/pages/Order/Card/OrderCatalogFilter.vue b/src/pages/Order/Card/OrderCatalogFilter.vue
index 6202a6f90..1dd569fb5 100644
--- a/src/pages/Order/Card/OrderCatalogFilter.vue
+++ b/src/pages/Order/Card/OrderCatalogFilter.vue
@@ -178,6 +178,7 @@ function addOrder(value, field, params) {
? resetCategory(params, searchFn)
: removeTagGroupParam(params, searchFn, valIndex)
"
+ data-cy="catalogFilterCustomTag"
>
{{
@@ -211,6 +212,7 @@ function addOrder(value, field, params) {
:name="category.icon"
class="category-icon"
@click="selectCategory(params, category, searchFn)"
+ data-cy="catalogFilterCategory"
>
{{ t(category.name) }}
@@ -234,6 +236,7 @@ function addOrder(value, field, params) {
sort-by="name ASC"
:disable="!params.categoryFk"
@update:model-value="searchFn()"
+ data-cy="catalogFilterType"
>
@@ -285,6 +288,7 @@ function addOrder(value, field, params) {
:is-clearable="false"
v-model="searchByTag"
@keyup.enter="(val) => onSearchByTag(val, params)"
+ data-cy="catalogFilterValueInput"
>
@@ -297,6 +301,7 @@ function addOrder(value, field, params) {
color="primary"
size="md"
dense
+ data-cy="catalogFilterValueDialogBtn"
/>
import { useRouter } from 'vue-router';
-import { onMounted, ref } from 'vue';
+import { reactive, onMounted, ref } from 'vue';
import { useI18n } from 'vue-i18n';
import axios from 'axios';
import { useState } from 'composables/useState';
@@ -9,7 +9,6 @@ import VnRow from 'components/ui/VnRow.vue';
import VnSelect from 'components/common/VnSelect.vue';
import VnInputDate from 'components/common/VnInputDate.vue';
import { useDialogPluginComponent } from 'quasar';
-import { reactive } from 'vue';
const { t } = useI18n();
const state = useState();
@@ -48,10 +47,6 @@ const fetchAgencyList = async (landed, addressFk) => {
agencyList.value = data;
};
-// const fetchOrderDetails = (order) => {
-// fetchAddressList(order?.addressFk);
-// fetchAgencyList(order?.landed, order?.addressFk);
-// };
const $props = defineProps({
clientFk: {
type: Number,
@@ -63,39 +58,6 @@ const initialFormState = reactive({
addressId: null,
clientFk: $props.clientFk,
});
-// const orderMapper = (order) => {
-// return {
-// addressId: order.addressFk,
-// agencyModeId: order.agencyModeFk,
-// landed: new Date(order.landed).toISOString(),
-// };
-// };
-// const orderFilter = {
-// include: [
-// { relation: 'agencyMode', scope: { fields: ['name'] } },
-// {
-// relation: 'address',
-// scope: { fields: ['nickname'] },
-// },
-// { relation: 'rows', scope: { fields: ['id'] } },
-// {
-// relation: 'client',
-// scope: {
-// fields: [
-// 'salesPersonFk',
-// 'name',
-// 'isActive',
-// 'isFreezed',
-// 'isTaxDataChecked',
-// ],
-// include: {
-// relation: 'salesPersonUser',
-// scope: { fields: ['id', 'name'] },
-// },
-// },
-// },
-// ],
-// };
const onClientChange = async (clientId = $props.clientFk) => {
const { data } = await axios.get(`Clients/${clientId}`);
diff --git a/src/pages/Supplier/Card/SupplierFiscalData.vue b/src/pages/Supplier/Card/SupplierFiscalData.vue
index 547842960..1a79be8bc 100644
--- a/src/pages/Supplier/Card/SupplierFiscalData.vue
+++ b/src/pages/Supplier/Card/SupplierFiscalData.vue
@@ -9,6 +9,7 @@ import VnRow from 'components/ui/VnRow.vue';
import VnInput from 'src/components/common/VnInput.vue';
import VnSelect from 'src/components/common/VnSelect.vue';
import VnLocation from 'src/components/common/VnLocation.vue';
+import VnAccountNumber from 'src/components/common/VnAccountNumber.vue';
const route = useRoute();
const { t } = useI18n();
@@ -100,10 +101,13 @@ function handleLocation(data, location) {
/>
-
{{ entity.ticketState.state.name }}
@@ -174,7 +175,7 @@ function ticketFilter(ticket) {
{{ t('Client Frozen') }}
{
dense
style="width: 50%"
@click="save()"
+ data-cy="saveManaBtn"
>
{{ t('globals.save') }}
diff --git a/src/pages/Ticket/Card/TicketNotes.vue b/src/pages/Ticket/Card/TicketNotes.vue
index 6861cf000..f558b71cc 100644
--- a/src/pages/Ticket/Card/TicketNotes.vue
+++ b/src/pages/Ticket/Card/TicketNotes.vue
@@ -80,12 +80,14 @@ async function handleSave() {
option-value="id"
v-model="row.observationTypeFk"
:disable="!!row.id"
+ data-cy="ticketNotesObservationType"
/>
{{ t('ticketNotes.removeNote') }}
@@ -107,6 +110,7 @@ async function handleSave() {
class="fill-icon-on-hover q-ml-md"
color="primary"
@click="ticketNotesCrudRef.insert()"
+ data-cy="ticketNotesAddNoteBtn"
>
{{ t('ticketNotes.addNote') }}
diff --git a/src/pages/Ticket/Card/TicketSale.vue b/src/pages/Ticket/Card/TicketSale.vue
index 95ef3118e..b534170c9 100644
--- a/src/pages/Ticket/Card/TicketSale.vue
+++ b/src/pages/Ticket/Card/TicketSale.vue
@@ -555,6 +555,7 @@ watch(
color="primary"
:disable="!isTicketEditable || ticketState === 'OK'"
@click="changeTicketState('OK')"
+ data-cy="ticketSaleOkStateBtn"
>
{{ t(`Change ticket state to 'Ok'`) }}
@@ -563,6 +564,7 @@ watch(
color="primary"
:label="t('ticketList.state')"
:disable="!isTicketEditable"
+ data-cy="ticketSaleStateDropdown"
>
{{ t('Transfer lines') }}
-
+
{{ t('ticketSale.reserved') }}
@@ -832,7 +842,14 @@ watch(
-
+
{{ t('Add item to basket') }}
diff --git a/src/pages/Ticket/Card/TicketSaleMoreActions.vue b/src/pages/Ticket/Card/TicketSaleMoreActions.vue
index 87e1d2a48..bd2099756 100644
--- a/src/pages/Ticket/Card/TicketSaleMoreActions.vue
+++ b/src/pages/Ticket/Card/TicketSaleMoreActions.vue
@@ -175,6 +175,7 @@ const createRefund = async (withWarehouse) => {
color="primary"
:label="t('ticketSale.more')"
:disable="disable"
+ data-cy="ticketSaleMoreActionsDropdown"
>
{{ t('Select lines to see the options') }}
@@ -186,6 +187,7 @@ const createRefund = async (withWarehouse) => {
v-close-popup
v-ripple
@click="showSmsDialog('productNotAvailable')"
+ data-cy="sendShortageSMSItem"
>
{{ t('Send shortage SMS') }}
@@ -197,12 +199,18 @@ const createRefund = async (withWarehouse) => {
v-close-popup
v-ripple
@click="calculateSalePrice()"
+ data-cy="recalculatePriceItem"
>
{{ t('Recalculate price') }}
-
+
{{ t('Update discount') }}
@@ -211,6 +219,7 @@ const createRefund = async (withWarehouse) => {
v-model.number="newDiscount"
:label="t('ticketSale.discount')"
type="number"
+ data-cy="ticketSaleDiscountInput"
/>
@@ -220,6 +229,7 @@ const createRefund = async (withWarehouse) => {
v-close-popup
v-ripple
@click="createClaim()"
+ data-cy="createClaimItem"
>
{{ t('Add claim') }}
@@ -231,6 +241,7 @@ const createRefund = async (withWarehouse) => {
v-close-popup
v-ripple
@click="setReserved(true)"
+ data-cy="markAsReservedItem"
>
{{ t('Mark as reserved') }}
@@ -242,12 +253,13 @@ const createRefund = async (withWarehouse) => {
v-close-popup
v-ripple
@click="setReserved(false)"
+ data-cy="unmarkAsReservedItem"
>
{{ t('Unmark as reserved') }}
-
+
{{ t('Refund') }}
@@ -256,12 +268,22 @@ const createRefund = async (withWarehouse) => {
-
+
{{ t('with warehouse') }}
-
+
{{ t('without warehouse') }}
diff --git a/src/pages/Ticket/Card/TicketSummary.vue b/src/pages/Ticket/Card/TicketSummary.vue
index 5bda60cba..2f5f69e1c 100644
--- a/src/pages/Ticket/Card/TicketSummary.vue
+++ b/src/pages/Ticket/Card/TicketSummary.vue
@@ -96,6 +96,7 @@ function toTicketUrl(section) {
ref="summaryRef"
:url="`Tickets/${entityId}/summary`"
data-key="TicketSummary"
+ data-cy="ticketSummary"
>
(_transfer.value = $props.transfer));
-
+
@@ -72,6 +74,7 @@ defineExpose({ transferSales });
color="primary"
class="full-width q-my-lg"
@click="transferSales()"
+ data-cy="ticketTransferNewTicketBtn"
/>
diff --git a/src/pages/Ticket/TicketAdvance.vue b/src/pages/Ticket/TicketAdvance.vue
index 8de602b37..71e3926ac 100644
--- a/src/pages/Ticket/TicketAdvance.vue
+++ b/src/pages/Ticket/TicketAdvance.vue
@@ -215,7 +215,7 @@ const requestComponentUpdate = async (ticket, isWithoutNegatives) => {
if (!newLanded) {
notify(t('advanceTickets.noDeliveryZone'), 'negative');
- return;
+ throw new Error(t('advanceTickets.noDeliveryZone'));
}
ticket.landed = newLanded.landed;
@@ -299,10 +299,10 @@ const splitTickets = async () => {
const { query, params } = await requestComponentUpdate(ticket, true);
await axios.post(query, params);
progressAdd(ticket.futureId);
- } catch (error) {
+ } catch (e) {
splitErrors.value.push({
id: ticket.futureId,
- reason: error.response?.data?.error?.message,
+ reason: e.message || e.response?.data?.error?.message,
});
progressAdd(ticket.futureId);
}
diff --git a/src/pages/Ticket/TicketList.vue b/src/pages/Ticket/TicketList.vue
index 6f6c556ca..2fe4fcddc 100644
--- a/src/pages/Ticket/TicketList.vue
+++ b/src/pages/Ticket/TicketList.vue
@@ -22,6 +22,7 @@ import CustomerDescriptorProxy from 'src/pages/Customer/Card/CustomerDescriptorP
import ZoneDescriptorProxy from 'src/pages/Zone/Card/ZoneDescriptorProxy.vue';
import { toTimeFormat } from 'src/filters/date';
import InvoiceOutDescriptorProxy from 'src/pages/InvoiceOut/Card/InvoiceOutDescriptorProxy.vue';
+import TicketProblems from 'src/components/TicketProblems.vue';
const route = useRoute();
const router = useRouter();
@@ -455,6 +456,7 @@ function setReference(data) {
data-key="TicketList"
:label="t('Search ticket')"
:info="t('You can search by ticket id or alias')"
+ data-cy="ticketListSearchBar"
/>
@@ -482,68 +484,10 @@ function setReference(data) {
'row-key': 'id',
selection: 'multiple',
}"
+ data-cy="ticketListTable"
>
-
-
-
- {{ t('No verified data') }}
-
-
-
-
- {{ t('Purchase request') }}
-
-
-
-
- {{ t('Not visible') }}
-
-
-
-
- {{ t('Client frozen') }}
-
-
-
- {{ t('Risk') }}: {{ row.risk }}
-
-
-
- {{ t('Component lack') }}
-
-
-
-
- {{ t('Rounding') }}
-
-
-
+
diff --git a/src/pages/Travel/Card/TravelSummary.vue b/src/pages/Travel/Card/TravelSummary.vue
index f4331ccb2..be1a12406 100644
--- a/src/pages/Travel/Card/TravelSummary.vue
+++ b/src/pages/Travel/Card/TravelSummary.vue
@@ -300,10 +300,6 @@ const getLink = (param) => `#/travel/${entityId.value}/${param}`;
-
diff --git a/src/pages/Worker/Card/WorkerPit.vue b/src/pages/Worker/Card/WorkerPit.vue
new file mode 100644
index 000000000..c58196c7b
--- /dev/null
+++ b/src/pages/Worker/Card/WorkerPit.vue
@@ -0,0 +1,263 @@
+
+
+
+ (disabilityGradesOptions = data)"
+ auto-load
+ />
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+es:
+ familySituation: Situación familiar
+ disabilityGrades: Discapacidad
+ geographicMobilityDate: Movilidad geografica
+ childPension: Pensión hijos
+ spousePension: Pensión cónyuge
+ isDependend: Ayuda / Movilidad reducida
+ spouseNif: NIF cónyuge
+ hasHousingPaymentBefore: Pagos vivienda anterior 2011
+ hasHousingPaymentAfter: Pagos vivienda posterior 2011
+ hasExtendedWorking: Prolongación actividad laboral
+ isDescendant: Descen/Ascen
+ disabilityGradeFk: Discapacidad
+ birthed: Año nacimiento
+ adoptionYear: Año adopción
+ isJointCustody: Computo por entero
+ Relatives: Relacionados
+en:
+ familySituation: Family Situation
+ disabilityGrades: Disability Grades
+ geographicMobilityDate: Geographic Mobility Date
+ childPension: Child Pension
+ spousePension: Spouse Pension
+ isDependend: Dependent Suport / Reduced Mobility
+ spouseNif: Spouse NIF (Tax ID)
+ hasHousingPaymentBefore: Housing Payments Before 2011
+ hasHousingPaymentAfter: Housing Payments After 2011
+ hasExtendedWorking: Extended Work Activity
+ isDescendant: Descendant/Ascendant
+ disabilityGradeFk: Disability Grade
+ birthed: Birth Year
+ adoptionYear: Adoption Year
+ isJointCustody: Joint custody
+ Relatives: Relatives
+
diff --git a/src/router/modules/Supplier.js b/src/router/modules/Supplier.js
index 143d7c824..c08fb5961 100644
--- a/src/router/modules/Supplier.js
+++ b/src/router/modules/Supplier.js
@@ -113,7 +113,7 @@ export default {
name: 'SupplierAccounts',
meta: {
title: 'accounts',
- icon: 'vn:account',
+ icon: 'vn:credit',
},
component: () =>
import('src/pages/Supplier/Card/SupplierAccounts.vue'),
diff --git a/src/router/modules/worker.js b/src/router/modules/worker.js
index d1feff23d..925019734 100644
--- a/src/router/modules/worker.js
+++ b/src/router/modules/worker.js
@@ -24,6 +24,7 @@ export default {
'WorkerDms',
'WorkerTimeControl',
'WorkerLocker',
+ 'WorkerPit',
'WorkerBalance',
'WorkerFormation',
'WorkerMedical',
@@ -216,6 +217,15 @@ export default {
},
component: () => import('src/pages/Worker/Card/WorkerMedical.vue'),
},
+ {
+ name: 'WorkerPit',
+ path: 'pit',
+ meta: {
+ title: 'pit',
+ icon: 'lock',
+ },
+ component: () => import('src/pages/Worker/Card/WorkerPit.vue'),
+ },
{
name: 'WorkerOperator',
path: 'operator',
diff --git a/src/stores/invoiceOutGlobal.js b/src/stores/invoiceOutGlobal.js
index 35f834f3d..332494aa8 100644
--- a/src/stores/invoiceOutGlobal.js
+++ b/src/stores/invoiceOutGlobal.js
@@ -162,6 +162,15 @@ export const useInvoiceOutGlobalStore = defineStore({
);
throw new Error('Invalid Serial Type');
}
+
+ if (clientsToInvoice === 'all' && params.serialType !== 'global') {
+ notify(
+ 'invoiceOut.globalInvoices.errors.invalidSerialTypeForAll',
+ 'negative'
+ );
+ throw new Error('For "all" clients, the serialType must be "global"');
+ }
+
if (!params.companyFk) {
notify('invoiceOut.globalInvoices.errors.chooseValidCompany', 'negative');
throw new Error('Invalid company');
diff --git a/test/cypress/integration/Order/orderCatalog.spec.js b/test/cypress/integration/Order/orderCatalog.spec.js
new file mode 100644
index 000000000..45eda6f1f
--- /dev/null
+++ b/test/cypress/integration/Order/orderCatalog.spec.js
@@ -0,0 +1,112 @@
+///
+describe('OrderCatalog', () => {
+ beforeEach(() => {
+ cy.login('developer');
+ cy.viewport(1920, 720);
+ cy.visit('/#/order/8/catalog');
+ });
+
+ const checkCustomFilterTag = (filterName = 'Plant') => {
+ cy.dataCy('catalogFilterCustomTag').should('exist');
+ cy.dataCy('catalogFilterCustomTag').contains(filterName);
+ };
+
+ const checkFilterTag = (filterName = 'Plant') => {
+ cy.dataCy('vnFilterPanelChip').should('exist');
+ cy.dataCy('vnFilterPanelChip').contains(filterName);
+ };
+
+ const selectCategory = (categoryIndex = 1, categoryName = 'Plant') => {
+ cy.get(
+ `div.q-page-container div:nth-of-type(${categoryIndex}) > [data-cy='catalogFilterCategory']`
+ ).should('exist');
+ cy.get(
+ `div.q-page-container div:nth-of-type(${categoryIndex}) > [data-cy='catalogFilterCategory']`
+ ).click();
+ checkCustomFilterTag(categoryName);
+ };
+
+ const searchByCustomTagInput = (option) => {
+ cy.dataCy('catalogFilterValueInput').find('input').last().focus();
+ cy.dataCy('catalogFilterValueInput').find('input').last().type(option);
+ cy.dataCy('catalogFilterValueInput').find('input').last().type('{enter}');
+ checkCustomFilterTag(option);
+ };
+
+ const selectTypeFilter = (option) => {
+ cy.selectOption(
+ 'div.q-page-container div.list > div:nth-of-type(2) div:nth-of-type(3)',
+ option
+ );
+ checkFilterTag(option);
+ };
+
+ it('Shows empty state', () => {
+ cy.dataCy('orderCatalogPage').should('exist');
+ cy.dataCy('orderCatalogPage').contains('No data to display');
+ });
+
+ it('filter by category', () => {
+ selectCategory();
+ cy.dataCy('orderCatalogItem').should('exist');
+ });
+
+ it('filters by type', () => {
+ selectCategory();
+ selectTypeFilter('Anthurium');
+ });
+
+ it('filters by custom value select', () => {
+ selectCategory();
+ searchByCustomTagInput('Silver');
+ });
+
+ it('filters by custom value dialog', () => {
+ Cypress.on('uncaught:exception', (err) => {
+ if (err.message.includes('canceled')) {
+ return false;
+ }
+ });
+ selectCategory();
+ cy.dataCy('catalogFilterValueDialogBtn').should('exist');
+ cy.dataCy('catalogFilterValueDialogBtn').last().click();
+ cy.dataCy('catalogFilterValueDialogTagSelect').should('exist');
+ cy.selectOption("[data-cy='catalogFilterValueDialogTagSelect']", 'Tallos');
+ cy.dataCy('catalogFilterValueDialogValueInput').find('input').focus();
+ cy.dataCy('catalogFilterValueDialogValueInput').find('input').type('2');
+ cy.dataCy('catalogFilterValueDialogValueInput').find('input').type('{enter}');
+ checkCustomFilterTag('2');
+ });
+
+ it('removes a secondary tag', () => {
+ selectCategory();
+ selectTypeFilter('Anthurium');
+ cy.dataCy('vnFilterPanelChip').should('exist');
+ cy.get(
+ "div.q-page-container [data-cy='vnFilterPanelChip'] > i.q-chip__icon--remove"
+ )
+ .contains('cancel')
+ .should('exist');
+ cy.get(
+ "div.q-page-container [data-cy='vnFilterPanelChip'] > i.q-chip__icon--remove"
+ )
+ .contains('cancel')
+ .click();
+ cy.dataCy('vnFilterPanelChip').should('not.exist');
+ });
+
+ it('Removes category tag', () => {
+ selectCategory();
+ cy.get(
+ "div.q-page-container [data-cy='catalogFilterCustomTag'] > i.q-chip__icon--remove"
+ )
+ .contains('cancel')
+ .should('exist');
+ cy.get(
+ "div.q-page-container [data-cy='catalogFilterCustomTag'] > i.q-chip__icon--remove"
+ )
+ .contains('cancel')
+ .click();
+ cy.dataCy('catalogFilterCustomTag').should('not.exist');
+ });
+});
diff --git a/test/cypress/integration/item/01_summary.spec.js b/test/cypress/integration/item/01_summary.spec.js
new file mode 100644
index 000000000..c44f4d047
--- /dev/null
+++ b/test/cypress/integration/item/01_summary.spec.js
@@ -0,0 +1,25 @@
+describe.skip('Item summary path', () => {
+ beforeEach(() => {
+ const itemId = 1;
+ cy.viewport(1280, 720);
+ cy.login('developer');
+ cy.visit(`/#/item/${itemId}`);
+ });
+ it('should search for an item', async () => {});
+ it(`should check the item summary preview shows fields from basic data`, async () => {});
+ it(`should check the item summary preview shows fields from tags`, async () => {});
+ it(`should check the item summary preview shows fields from botanical`, async () => {});
+ it(`should check the item summary preview shows fields from barcode`, async () => {});
+ it(`should close the summary popup`, async () => {});
+ it('should search for other item', async () => {});
+ it(`should now check the item summary preview shows fields from basic data`, async () => {});
+ it(`should now check the item summary preview shows fields from tags`, async () => {});
+ it(`should now check the item summary preview shows fields from botanical`, async () => {});
+ it(`should now close the summary popup`, async () => {});
+ it(`should navigate to one of the items detailed section`, async () => {});
+ it(`should check the descritor edit button is not visible for employee`, async () => {});
+ it(`should check the item summary shows fields from basic data section`, async () => {});
+ it(`should check the item summary shows fields from tags section`, async () => {});
+ it(`should check the item summary shows fields from botanical section`, async () => {});
+ it(`should check the item summary shows fields from barcodes section`, async () => {});
+});
diff --git a/test/cypress/integration/item/02_basic_data.spec.js b/test/cypress/integration/item/02_basic_data.spec.js
new file mode 100644
index 000000000..ada9ef57c
--- /dev/null
+++ b/test/cypress/integration/item/02_basic_data.spec.js
@@ -0,0 +1,10 @@
+describe.skip('Item Edit basic data path', () => {
+ beforeEach(() => {
+ const itemId = 1;
+ cy.viewport(1280, 720);
+ cy.login('developer');
+ cy.visit(`/#/item/${itemId}`);
+ });
+ it(`should edit the item basic data and confirm the item data was edited`, async () => {});
+ it(`should create a new intrastat and save it`, async () => {});
+});
diff --git a/test/cypress/integration/item/03_tax.spec.js b/test/cypress/integration/item/03_tax.spec.js
new file mode 100644
index 000000000..593dbfb36
--- /dev/null
+++ b/test/cypress/integration/item/03_tax.spec.js
@@ -0,0 +1,12 @@
+describe.skip('Item edit tax path', () => {
+ beforeEach(() => {
+ const itemId = 1;
+ cy.viewport(1280, 720);
+ cy.login('developer');
+ cy.visit(`/#/item/${itemId}`);
+ });
+ it(`should add the item tax to all countries`, async () => {});
+ it(`should confirm the first item tax class was edited`, async () => {});
+ it(`should confirm the second item tax class was edited`, async () => {});
+ it(`should edit the first class without saving the form`, async () => {});
+});
diff --git a/test/cypress/integration/item/04_tags.spec.js b/test/cypress/integration/item/04_tags.spec.js
new file mode 100644
index 000000000..720d19a89
--- /dev/null
+++ b/test/cypress/integration/item/04_tags.spec.js
@@ -0,0 +1,12 @@
+describe.skip('Item create tags path', () => {
+ beforeEach(() => {
+ const itemId = 1;
+ cy.viewport(1280, 720);
+ cy.login('developer');
+ cy.visit(`/#/item/${itemId}`);
+ });
+ it('should create a new tag and delete a former one', async () => {});
+ it('should confirm the fourth row data is the expected one', async () => {});
+ it('should confirm the fifth row data is the expected one', async () => {});
+ it('should confirm the sixth row data is the expected one', async () => {});
+});
diff --git a/test/cypress/integration/item/05_botanical.spec.js b/test/cypress/integration/item/05_botanical.spec.js
new file mode 100644
index 000000000..72e7f3fee
--- /dev/null
+++ b/test/cypress/integration/item/05_botanical.spec.js
@@ -0,0 +1,14 @@
+describe.skip('Item Create botanical path', () => {
+ beforeEach(() => {
+ const itemId = 1;
+ cy.viewport(1280, 720);
+ cy.login('developer');
+ cy.visit(`/#/item/${itemId}`);
+ });
+ it(`should create a new botanical for the item`, async () => {});
+ it(`should confirm the Genus for the item was created`, async () => {});
+ it(`should confirm the Species for the item was created`, async () => {});
+ it(`should edit botanical for the item`, async () => {});
+ it(`should confirm the Genus for the item was edited`, async () => {});
+ it(`should confirm the Species for the item was edited`, async () => {});
+});
diff --git a/test/cypress/integration/item/06_barcode.spec.js b/test/cypress/integration/item/06_barcode.spec.js
new file mode 100644
index 000000000..cea3c13f2
--- /dev/null
+++ b/test/cypress/integration/item/06_barcode.spec.js
@@ -0,0 +1,10 @@
+describe.skip('Item Create barcodes path', () => {
+ beforeEach(() => {
+ const itemId = 1;
+ cy.viewport(1280, 720);
+ cy.login('developer');
+ cy.visit(`/#/item/${itemId}`);
+ });
+ it(`should click create a new code and delete a former one`, async () => {});
+ it(`should confirm the barcode 5 is created and it is now the third barcode as the first was deleted`, async () => {});
+});
diff --git a/test/cypress/integration/item/07_create.spec.js b/test/cypress/integration/item/07_create.spec.js
new file mode 100644
index 000000000..7555d6927
--- /dev/null
+++ b/test/cypress/integration/item/07_create.spec.js
@@ -0,0 +1,13 @@
+describe.skip('Item Create', () => {
+ beforeEach(() => {
+ const itemId = 1;
+ cy.viewport(1280, 720);
+ cy.login('developer');
+ cy.visit(`/#/item/${itemId}`);
+ });
+ it('should access to the create item view by clicking the create floating button', async () => {});
+ it('should return to the item index by clickig the cancel button', async () => {});
+ it('should now access to the create item view by clicking the create floating button', async () => {});
+ it('should throw an error when insert an invalid priority', async () => {});
+ it('should create the Infinity Gauntlet item', async () => {});
+});
diff --git a/test/cypress/integration/item/08_regularize.spec.js b/test/cypress/integration/item/08_regularize.spec.js
new file mode 100644
index 000000000..74d12846f
--- /dev/null
+++ b/test/cypress/integration/item/08_regularize.spec.js
@@ -0,0 +1,24 @@
+describe.skip('Item regularize path', () => {
+ beforeEach(() => {
+ const itemId = 1;
+ cy.viewport(1280, 720);
+ cy.login('developer');
+ cy.visit(`/#/item/${itemId}`);
+ });
+ it('should edit the user local warehouse', async () => {});
+ it('should check the local settings were saved', async () => {});
+ it('should search for a specific item', async () => {});
+ it('should open the regularize dialog and check the warehouse matches the local user settings', async () => {});
+ it('should regularize the item', async () => {});
+ it('should click on the Tickets button of the top bar menu', async () => {});
+ it('should clear the user local settings now', async () => {});
+ it('should search for the ticket with alias missing', async () => {});
+ it(`should check the ticket sale quantity is showing a negative value`, async () => {});
+ it(`should check the ticket sale discount is 100%`, async () => {});
+ it('should now click on the Items button of the top bar menu', async () => {});
+ it('should search for the item once again', async () => {});
+ it('should regularize the item once more', async () => {});
+ it('should again click on the Tickets button of the top bar menu', async () => {});
+ it('should search for the ticket missing once again', async () => {});
+ it(`should check the ticket contains now two sales`, async () => {});
+});
diff --git a/test/cypress/integration/item/09_index.spec.js b/test/cypress/integration/item/09_index.spec.js
new file mode 100644
index 000000000..5648a8b5f
--- /dev/null
+++ b/test/cypress/integration/item/09_index.spec.js
@@ -0,0 +1,14 @@
+describe.skip('Item index path', () => {
+ beforeEach(() => {
+ const itemId = 1;
+ cy.viewport(1280, 720);
+ cy.login('developer');
+ cy.visit(`/#/item/${itemId}`);
+ });
+ it('should click on the fields to show button to open the list of columns to show', async () => {});
+ it('should unmark all checkboxes except the first and the last ones', async () => {});
+ it('should navigate forth and back to see the images column is still visible', async () => {});
+ it('should check the ids column is not visible', async () => {});
+ it('should mark all unchecked boxes to leave the index as it was', async () => {});
+ it('should now navigate forth and back to see the ids column is now visible', async () => {});
+});
diff --git a/test/cypress/integration/item/10_item_log.spec.js b/test/cypress/integration/item/10_item_log.spec.js
new file mode 100644
index 000000000..3cc64dbbb
--- /dev/null
+++ b/test/cypress/integration/item/10_item_log.spec.js
@@ -0,0 +1,12 @@
+describe.skip('Item log path', () => {
+ beforeEach(() => {
+ const itemId = 1;
+ cy.viewport(1280, 720);
+ cy.login('developer');
+ cy.visit(`/#/item/${itemId}`);
+ });
+ it(`should search for the Knowledge artifact to confirm it isn't created yet`, async () => {});
+ it('should access to the create item view by clicking the create floating button', async () => {});
+ it('should create the Knowledge artifact item', async () => {});
+ it('should return to the items index by clicking the return to items button', async () => {});
+});
diff --git a/test/cypress/integration/item/11_descriptor.spec.js b/test/cypress/integration/item/11_descriptor.spec.js
new file mode 100644
index 000000000..6ea89c365
--- /dev/null
+++ b/test/cypress/integration/item/11_descriptor.spec.js
@@ -0,0 +1,11 @@
+describe.skip('Item descriptor path', () => {
+ beforeEach(() => {
+ const itemId = 1;
+ cy.viewport(1280, 720);
+ cy.login('developer');
+ cy.visit(`/#/item/${itemId}`);
+ });
+ it('should set the item to inactive', async () => {});
+ it('should reload the section and check the inactive icon is visible', async () => {});
+ it('should set the item back to active', async () => {});
+});
diff --git a/test/cypress/integration/item/12_request.spec.js b/test/cypress/integration/item/12_request.spec.js
new file mode 100644
index 000000000..f4e26f421
--- /dev/null
+++ b/test/cypress/integration/item/12_request.spec.js
@@ -0,0 +1,12 @@
+describe.skip('Item request path', () => {
+ beforeEach(() => {
+ const itemId = 1;
+ cy.viewport(1280, 720);
+ cy.login('developer');
+ cy.visit(`/#/item/${itemId}`);
+ });
+ it('should reach the item request section', async () => {});
+ it('should fill the id and quantity then check the concept was updated', async () => {});
+ it('should check the status of the request should now be accepted', async () => {});
+ it('should now click on the second declain request icon then type the reason', async () => {});
+});
diff --git a/test/cypress/integration/item/13_fixedPrice.spec.js b/test/cypress/integration/item/13_fixedPrice.spec.js
new file mode 100644
index 000000000..44fdfde46
--- /dev/null
+++ b/test/cypress/integration/item/13_fixedPrice.spec.js
@@ -0,0 +1,12 @@
+describe.skip('Item fixed prices path', () => {
+ beforeEach(() => {
+ const itemId = 1;
+ cy.viewport(1280, 720);
+ cy.login('developer');
+ cy.visit(`/#/item/${itemId}`);
+ });
+ it('should filter using all the fields', async () => {});
+ it('should click on the add new fixed price button', async () => {});
+ it('should fill the fixed price data', async () => {});
+ it('should reload the section and check the created price has the expected ID', async () => {});
+});
diff --git a/test/cypress/integration/order/01_summary.spec.js b/test/cypress/integration/order/01_summary.spec.js
new file mode 100644
index 000000000..1c51ead8b
--- /dev/null
+++ b/test/cypress/integration/order/01_summary.spec.js
@@ -0,0 +1,9 @@
+describe.skip('Order summary path', () => {
+ beforeEach(() => {
+ const orderId = 1;
+ cy.viewport(1280, 720);
+ cy.login('developer');
+ cy.visit(`/#/order/${orderId}`);
+ });
+ it('should reach the order summary section and check data', async () => {});
+});
diff --git a/test/cypress/integration/order/02_basic_data.spec.js b/test/cypress/integration/order/02_basic_data.spec.js
new file mode 100644
index 000000000..381a8435f
--- /dev/null
+++ b/test/cypress/integration/order/02_basic_data.spec.js
@@ -0,0 +1,12 @@
+describe.skip('Order edit basic data path', () => {
+ beforeEach(() => {
+ const orderId = 1;
+ cy.viewport(1280, 720);
+ cy.login('developer');
+ cy.visit(`/#/order/${orderId}`);
+ });
+ describe('when confirmed order', () => {});
+ it('should not be able to change the client', async () => {});
+ describe('when new order', () => {});
+ it('should create an order and edit its basic data', async () => {});
+});
diff --git a/test/cypress/integration/order/03_lines.spec.js b/test/cypress/integration/order/03_lines.spec.js
new file mode 100644
index 000000000..483d09d0d
--- /dev/null
+++ b/test/cypress/integration/order/03_lines.spec.js
@@ -0,0 +1,12 @@
+describe.skip('Order lines', () => {
+ beforeEach(() => {
+ const orderId = 1;
+ cy.viewport(1280, 720);
+ cy.login('developer');
+ cy.visit(`/#/order/${orderId}`);
+ });
+ it('should check the order subtotal', async () => {});
+ it('should delete the first line in the order', async () => {});
+ it('should confirm the order subtotal has changed', async () => {});
+ it('should confirm the whole order and redirect to ticket index filtered by clientFk', async () => {});
+});
diff --git a/test/cypress/integration/order/04_catalog.spec.js b/test/cypress/integration/order/04_catalog.spec.js
new file mode 100644
index 000000000..6115f34ce
--- /dev/null
+++ b/test/cypress/integration/order/04_catalog.spec.js
@@ -0,0 +1,17 @@
+describe.skip('Order catalog', () => {
+ beforeEach(() => {
+ const orderId = 1;
+ cy.viewport(1280, 720);
+ cy.login('developer');
+ cy.visit(`/#/order/${orderId}`);
+ });
+ it('should open the create new order form', async () => {});
+ it('should create a new order', async () => {});
+ it('should add the realm and type filters and obtain results', async () => {});
+ it('should perfom an "OR" search for the item tag colors silver and brown', async () => {});
+ it('should perfom an "OR" search for the item tag tallos 2 and 9', async () => {});
+ it('should perform a general search for category', async () => {});
+ it('should perfom an "AND" search for the item tag tallos 2', async () => {});
+ it('should remove the tag filters and have 4 results', async () => {});
+ it('should search for an item by id', async () => {});
+});
diff --git a/test/cypress/integration/order/05_index.spec.js b/test/cypress/integration/order/05_index.spec.js
new file mode 100644
index 000000000..2e9117338
--- /dev/null
+++ b/test/cypress/integration/order/05_index.spec.js
@@ -0,0 +1,10 @@
+describe.skip('Order Index', () => {
+ beforeEach(() => {
+ const orderId = 1;
+ cy.viewport(1280, 720);
+ cy.login('developer');
+ cy.visit(`/#/order/${orderId}`);
+ });
+ it(`should check the second search result doesn't contain a total of 0€`, async () => {});
+ it('should search including empty orders', async () => {});
+});
diff --git a/test/cypress/integration/ticket/ticketList.spec.js b/test/cypress/integration/ticket/ticketList.spec.js
new file mode 100644
index 000000000..bbdbcea92
--- /dev/null
+++ b/test/cypress/integration/ticket/ticketList.spec.js
@@ -0,0 +1,54 @@
+///
+describe('TicketList', () => {
+ const firstRow = 'tbody > :nth-child(1)';
+
+ beforeEach(() => {
+ cy.login('developer');
+ cy.viewport(1920, 1080);
+ cy.visit('/#/ticket/list');
+ });
+
+ const searchResults = (search) => {
+ cy.dataCy('vnSearchBar').find('input').focus();
+ if (search) cy.dataCy('vnSearchBar').find('input').type(search);
+ cy.dataCy('vnSearchBar').find('input').type('{enter}');
+ cy.dataCy('ticketListTable').should('exist');
+ cy.get(firstRow).should('exist');
+ };
+
+ it('should search results', () => {
+ cy.dataCy('ticketListTable').should('not.exist');
+ cy.get('.q-field__control').should('exist');
+ searchResults();
+ });
+
+ it('should open ticket sales', () => {
+ searchResults();
+ cy.window().then((win) => {
+ cy.stub(win, 'open').as('windowOpen');
+ });
+ cy.get(firstRow).find('.q-btn:first').click();
+ cy.get('@windowOpen').should('be.calledWithMatch', /\/ticket\/\d+\/sale/);
+ });
+
+ it('should open ticket summary', () => {
+ searchResults();
+ cy.get(firstRow).find('.q-btn:last').click();
+ cy.dataCy('ticketSummary').should('exist');
+ });
+
+ it('Client list create new client', () => {
+ cy.dataCy('vnTableCreateBtn').should('exist');
+ cy.dataCy('vnTableCreateBtn').click();
+ const data = {
+ Customer: { val: 1, type: 'select' },
+ Warehouse: { val: 'Warehouse One', type: 'select' },
+ Address: { val: 'employee', type: 'select' },
+ Landed: { val: '01-01-2024', type: 'date' },
+ };
+ cy.fillInForm(data);
+ cy.get('.q-mt-lg > .q-btn--standard').click();
+ cy.checkNotification('Data created');
+ cy.url().should('match', /\/ticket\/\d+\/summary/);
+ });
+});
diff --git a/test/cypress/integration/ticket/ticketNotes.spec.js b/test/cypress/integration/ticket/ticketNotes.spec.js
new file mode 100644
index 000000000..ef196c783
--- /dev/null
+++ b/test/cypress/integration/ticket/ticketNotes.spec.js
@@ -0,0 +1,25 @@
+///
+describe('TicketRequest', () => {
+ beforeEach(() => {
+ cy.login('developer');
+ cy.viewport(1920, 1080);
+ cy.visit('/#/ticket/31/observation');
+ });
+
+ it('Creates and deletes a note', () => {
+ cy.dataCy('ticketNotesAddNoteBtn').should('exist');
+ cy.dataCy('ticketNotesAddNoteBtn').click();
+ cy.dataCy('ticketNotesObservationType').should('exist');
+ cy.selectOption('[data-cy="ticketNotesObservationType"]:last', 'Weight');
+ cy.dataCy('ticketNotesDescription').should('exist');
+ cy.get('[data-cy="ticketNotesDescription"]:last').type(
+ 'This is a note description'
+ );
+ cy.dataCy('crudModelDefaultSaveBtn').click();
+ cy.checkNotification('Data saved');
+ cy.dataCy('ticketNotesRemoveNoteBtn').should('exist');
+ cy.dataCy('ticketNotesRemoveNoteBtn').click();
+ cy.dataCy('VnConfirm_confirm').click();
+ cy.checkNotification('Data saved');
+ });
+});
diff --git a/test/cypress/integration/ticket/ticketRequest.spec.js b/test/cypress/integration/ticket/ticketRequest.spec.js
new file mode 100644
index 000000000..b9dc509ef
--- /dev/null
+++ b/test/cypress/integration/ticket/ticketRequest.spec.js
@@ -0,0 +1,22 @@
+///
+describe('TicketRequest', () => {
+ beforeEach(() => {
+ cy.login('developer');
+ cy.viewport(1920, 1080);
+ cy.visit('/#/ticket/31/request');
+ });
+
+ it('Creates a new request', () => {
+ cy.dataCy('vnTableCreateBtn').should('exist');
+ cy.dataCy('vnTableCreateBtn').click();
+ const data = {
+ Description: { val: 'Purchase description' },
+ Atender: { val: 'buyerNick', type: 'select' },
+ Quantity: { val: 2 },
+ Price: { val: 123 },
+ };
+ cy.fillInForm(data);
+ cy.get('.q-mt-lg > .q-btn--standard').click();
+ cy.checkNotification('Data created');
+ });
+});
diff --git a/test/cypress/integration/ticket/ticketSale.spec.js b/test/cypress/integration/ticket/ticketSale.spec.js
new file mode 100644
index 000000000..60f31dbf6
--- /dev/null
+++ b/test/cypress/integration/ticket/ticketSale.spec.js
@@ -0,0 +1,131 @@
+///
+
+const c = require('croppie');
+
+describe('TicketSale', () => {
+ beforeEach(() => {
+ cy.login('developer');
+ cy.viewport(1920, 1080);
+ cy.visit('/#/ticket/31/sale');
+ });
+
+ const firstRow = 'tbody > :nth-child(1)';
+
+ const selectFirstRow = () => {
+ cy.waitForElement(firstRow);
+ cy.get(firstRow).find('.q-checkbox__inner').click();
+ };
+
+ it('it should add item to basket', () => {
+ cy.window().then((win) => {
+ cy.stub(win, 'open').as('windowOpen');
+ });
+ cy.dataCy('ticketSaleAddToBasketBtn').should('exist');
+ cy.dataCy('ticketSaleAddToBasketBtn').click();
+ cy.get('@windowOpen').should('be.calledWithMatch', /\/order\/\d+\/catalog/);
+ });
+
+ it('should send SMS', () => {
+ selectFirstRow();
+ cy.dataCy('ticketSaleMoreActionsDropdown').click();
+ cy.waitForElement('[data-cy="sendShortageSMSItem"]');
+ cy.dataCy('sendShortageSMSItem').should('exist');
+ cy.dataCy('sendShortageSMSItem').click();
+ cy.dataCy('vnSmsDialog').should('exist');
+ cy.dataCy('sendSmsBtn').click();
+ cy.checkNotification('SMS sent');
+ });
+
+ it('should recalculate price when "Recalculate price" is clicked', () => {
+ cy.intercept('POST', '**/recalculatePrice').as('recalculatePrice');
+ selectFirstRow();
+ cy.dataCy('ticketSaleMoreActionsDropdown').click();
+ cy.waitForElement('[data-cy="recalculatePriceItem"]');
+ cy.dataCy('recalculatePriceItem').should('exist');
+ cy.dataCy('recalculatePriceItem').click();
+ cy.wait('@recalculatePrice').its('response.statusCode').should('eq', 200);
+ cy.checkNotification('Data saved');
+ });
+
+ it('should update discount when "Update discount" is clicked', () => {
+ selectFirstRow();
+ cy.dataCy('ticketSaleMoreActionsDropdown').click();
+ cy.waitForElement('[data-cy="updateDiscountItem"]');
+ cy.dataCy('updateDiscountItem').should('exist');
+ cy.dataCy('updateDiscountItem').click();
+ cy.waitForElement('[data-cy="ticketSaleDiscountInput"]');
+ cy.dataCy('ticketSaleDiscountInput').find('input').focus();
+ cy.dataCy('ticketSaleDiscountInput').find('input').type('10');
+ cy.dataCy('saveManaBtn').click();
+ cy.waitForElement('.q-notification__message');
+ cy.checkNotification('Data saved');
+ });
+
+ it('adds claim', () => {
+ selectFirstRow();
+ cy.dataCy('ticketSaleMoreActionsDropdown').click();
+ cy.dataCy('createClaimItem').click();
+ cy.dataCy('VnConfirm_confirm').click();
+ cy.url().should('match', /\/claim\/\d+\/basic-data/);
+ // Delete created claim to avoid cluttering the database
+ cy.dataCy('descriptor-more-opts').click();
+ cy.dataCy('deleteClaim').click();
+ cy.dataCy('VnConfirm_confirm').click();
+ cy.checkNotification('Data deleted');
+ });
+
+ it('marks row as reserved', () => {
+ selectFirstRow();
+ cy.dataCy('ticketSaleMoreActionsDropdown').click();
+ cy.waitForElement('[data-cy="markAsReservedItem"]');
+ cy.dataCy('markAsReservedItem').click();
+ cy.dataCy('ticketSaleReservedIcon').should('exist');
+ });
+
+ it('unmarks row as reserved', () => {
+ selectFirstRow();
+ cy.dataCy('ticketSaleMoreActionsDropdown').click();
+ cy.waitForElement('[data-cy="unmarkAsReservedItem"]');
+ cy.dataCy('unmarkAsReservedItem').click();
+ cy.dataCy('ticketSaleReservedIcon').should('not.exist');
+ });
+
+ it('refunds row with warehouse', () => {
+ selectFirstRow();
+ cy.dataCy('ticketSaleMoreActionsDropdown').click();
+ cy.dataCy('ticketSaleRefundItem').click();
+ cy.dataCy('ticketSaleRefundWithWarehouse').click();
+ cy.checkNotification('The following refund ticket have been created');
+ });
+
+ it('refunds row without warehouse', () => {
+ selectFirstRow();
+ cy.dataCy('ticketSaleMoreActionsDropdown').click();
+ cy.dataCy('ticketSaleRefundItem').click();
+ cy.dataCy('ticketSaleRefundWithoutWarehouse').click();
+ cy.checkNotification('The following refund ticket have been created');
+ });
+
+ it('transfers ticket', () => {
+ cy.visit('/#/ticket/32/sale');
+ selectFirstRow();
+ cy.dataCy('ticketSaleTransferBtn').click();
+ cy.dataCy('ticketTransferPopup').should('exist');
+ cy.dataCy('ticketTransferNewTicketBtn').click();
+ // existen 3 elementos "tbody" necesito checkear que el segundo elemento tbody tenga una row sola
+ cy.get('tbody').eq(1).find('tr').should('have.length', 1);
+ selectFirstRow();
+ cy.dataCy('ticketSaleTransferBtn').click();
+ cy.dataCy('ticketTransferPopup').should('exist');
+ cy.dataCy('ticketTransferDestinationTicketInput').find('input').focus();
+ cy.dataCy('ticketTransferDestinationTicketInput').find('input').type('32');
+ cy.dataCy('ticketTransferTransferBtn').click();
+ // checkear que la url contenga /ticket/1000002/sale
+ cy.url().should('match', /\/ticket\/32\/sale/);
+ });
+
+ it('should redirect to ticket logs', () => {
+ cy.get(firstRow).find('.q-btn:last').click();
+ cy.url().should('match', /\/ticket\/31\/log/);
+ });
+});
diff --git a/test/cypress/integration/vnComponent/VnAccountNumber.spec.js b/test/cypress/integration/vnComponent/VnAccountNumber.spec.js
new file mode 100644
index 000000000..000c2151d
--- /dev/null
+++ b/test/cypress/integration/vnComponent/VnAccountNumber.spec.js
@@ -0,0 +1,39 @@
+describe('VnInput Component', () => {
+ beforeEach(() => {
+ cy.login('developer');
+ cy.viewport(1920, 1080);
+ cy.visit('/#/supplier/1/fiscal-data');
+ cy.domContentLoad();
+ });
+
+ it('should replace character at cursor position in insert mode', () => {
+ // Simula escribir en el input
+ cy.dataCy('supplierFiscalDataAccount').clear();
+ cy.dataCy('supplierFiscalDataAccount').type('4100000001');
+ // Coloca el cursor en la posición 0
+ cy.dataCy('supplierFiscalDataAccount').type('{movetostart}');
+ // Escribe un número y verifica que se reemplace correctamente
+ cy.dataCy('supplierFiscalDataAccount').type('999');
+ cy.dataCy('supplierFiscalDataAccount')
+ .should('have.value', '9990000001');
+ });
+
+ it('should replace character at cursor position in insert mode', () => {
+ // Simula escribir en el input
+ cy.dataCy('supplierFiscalDataAccount').clear();
+ cy.dataCy('supplierFiscalDataAccount').type('4100000001');
+ // Coloca el cursor en la posición 0
+ cy.dataCy('supplierFiscalDataAccount').type('{movetostart}');
+ // Escribe un número y verifica que se reemplace correctamente en la posicion incial
+ cy.dataCy('supplierFiscalDataAccount').type('999');
+ cy.dataCy('supplierFiscalDataAccount')
+ .should('have.value', '9990000001');
+ });
+
+ it('should respect maxlength prop', () => {
+ cy.dataCy('supplierFiscalDataAccount').clear();
+ cy.dataCy('supplierFiscalDataAccount').type('123456789012345');
+ cy.dataCy('supplierFiscalDataAccount')
+ .should('have.value', '1234567890'); // asumiendo que maxlength es 10
+ });
+});
diff --git a/test/cypress/integration/vnComponent/vnBreadcrumbs.spec.js b/test/cypress/integration/vnComponent/VnBreadcrumbs.spec.js
similarity index 100%
rename from test/cypress/integration/vnComponent/vnBreadcrumbs.spec.js
rename to test/cypress/integration/vnComponent/VnBreadcrumbs.spec.js
diff --git a/test/cypress/integration/vnComponent/vnLocation.spec.js b/test/cypress/integration/vnComponent/VnLocation.spec.js
similarity index 71%
rename from test/cypress/integration/vnComponent/vnLocation.spec.js
rename to test/cypress/integration/vnComponent/VnLocation.spec.js
index 06c23f4ee..b98d42fdd 100644
--- a/test/cypress/integration/vnComponent/vnLocation.spec.js
+++ b/test/cypress/integration/vnComponent/VnLocation.spec.js
@@ -1,3 +1,5 @@
+const { randomNumber, randomString } = require('../../support');
+
describe('VnLocation', () => {
const locationOptions = '[role="listbox"] > div.q-virtual-scroll__content > .q-item';
const dialogInputs = '.q-dialog label input';
@@ -99,7 +101,7 @@ describe('VnLocation', () => {
});
it('Create postCode', () => {
- const postCode = '1234475';
+ const postCode = Math.floor(100000 + Math.random() * 900000);
const province = 'Valencia';
cy.get(createLocationButton).click();
cy.get('.q-card > h1').should('have.text', 'New postcode');
@@ -115,9 +117,10 @@ describe('VnLocation', () => {
checkVnLocation(postCode, province);
});
- it('Create city', () => {
- const postCode = '9011';
- const province = 'Saskatchew';
+
+ it('Create city without country', () => {
+ const postCode = randomNumber();
+ const province = randomString({ length: 4 });
cy.get(createLocationButton).click();
cy.get(dialogInputs).eq(0).type(postCode);
cy.get(
@@ -131,6 +134,58 @@ describe('VnLocation', () => {
checkVnLocation(postCode, province);
});
+ it('Create province without country', () => {
+ const provinceName = 'Saskatchew'.concat(Math.random(1 * 100));
+ cy.get(createLocationButton).click();
+ cy.get(
+ `${createForm.prefix} > :nth-child(5) > .q-select > ${createForm.sufix} > :nth-child(2) `
+ )
+ .eq(0)
+ .click();
+ cy.selectOption('#q-portal--dialog--3 .q-select', 'one');
+ cy.countSelectOptions('#q-portal--dialog--3 .q-select', 4);
+ cy.get('#q-portal--dialog--3 .q-input').type(provinceName);
+
+ cy.get('#q-portal--dialog--3 .q-btn--standard').click();
+ });
+
+ it('Create city with country', () => {
+ const cityName = 'Saskatchew'.concat(Math.random(1 * 100));
+ cy.get(createLocationButton).click();
+ cy.selectOption(
+ `${createForm.prefix} > :nth-child(5) > :nth-child(3) `,
+ 'Italia'
+ );
+ cy.get(
+ `${createForm.prefix} > :nth-child(4) > .q-select > ${createForm.sufix} > :nth-child(2) > .q-icon`
+ ).click();
+ cy.selectOption('#q-portal--dialog--4 .q-select', 'Province four');
+ cy.countSelectOptions('#q-portal--dialog--4 .q-select', 1);
+
+ cy.get('#q-portal--dialog--4 .q-input').type(cityName);
+ cy.get('#q-portal--dialog--4 .q-btn--standard').click();
+ });
+
+ it('Create province with country', () => {
+ const provinceName = 'Saskatchew'.concat(Math.random(1 * 100));
+ cy.get(createLocationButton).click();
+ cy.selectOption(
+ `${createForm.prefix} > :nth-child(5) > :nth-child(3) `,
+ 'España'
+ );
+ cy.get(
+ `${createForm.prefix} > :nth-child(5) > .q-select > ${createForm.sufix} > :nth-child(2) `
+ )
+ .eq(0)
+ .click();
+
+ cy.selectOption('#q-portal--dialog--4 .q-select', 'one');
+ cy.countSelectOptions('#q-portal--dialog--4 .q-select', 2);
+
+ cy.get('#q-portal--dialog--4 .q-input').type(provinceName);
+ cy.get('#q-portal--dialog--4 .q-btn--standard').click();
+ });
+
function checkVnLocation(postCode, province) {
cy.get(`${createForm.prefix}`).should('not.exist');
cy.get('.q-form > .q-card > .vn-row:nth-child(6)')
diff --git a/test/cypress/integration/vnComponent/vnLog.spec.js b/test/cypress/integration/vnComponent/VnLog.spec.js
similarity index 100%
rename from test/cypress/integration/vnComponent/vnLog.spec.js
rename to test/cypress/integration/vnComponent/VnLog.spec.js
diff --git a/test/cypress/integration/vnComponent/vnSearchBar.spec.js b/test/cypress/integration/vnComponent/VnSearchBar.spec.js
similarity index 100%
rename from test/cypress/integration/vnComponent/vnSearchBar.spec.js
rename to test/cypress/integration/vnComponent/VnSearchBar.spec.js
diff --git a/test/cypress/integration/worker/workerPit.spec.js b/test/cypress/integration/worker/workerPit.spec.js
new file mode 100644
index 000000000..cc3a87637
--- /dev/null
+++ b/test/cypress/integration/worker/workerPit.spec.js
@@ -0,0 +1,40 @@
+describe('WorkerPit', () => {
+ const familySituationInput = '[data-cy="Family Situation_input"]';
+ const familySituation = '1';
+ const childPensionInput = '[data-cy="Child Pension_input"]';
+ const childPension = '120';
+ const spouseNifInput = '[data-cy="Spouse Pension_input"]';
+ const spouseNif = '65117125P';
+ const spousePensionInput = '[data-cy="Spouse Pension_input"]';
+ const spousePension = '120';
+ const addRelative = '[data-cy="addRelative"]';
+ const isDescendantSelect = '[data-cy="Descendant/Ascendant_select"]';
+ const birthedInput = '[data-cy="Birth Year_input"]';
+ const birthed = '2002';
+ const adoptionYearInput = '[data-cy="Adoption Year_input"]';
+ const adoptionYear = '2004';
+ const saveRelative = '[data-cy="workerPitRelativeSaveBtn"]';
+ const savePIT = '#st-actions > .q-btn-group > .q-btn--standard';
+
+ beforeEach(() => {
+ cy.viewport(1920, 1080);
+ cy.login('developer');
+ cy.visit(`/#/worker/1107/pit`);
+ });
+
+ it('complete PIT', () => {
+ cy.get(familySituationInput).type(familySituation);
+ cy.get(childPensionInput).type(childPension);
+ cy.get(spouseNifInput).type(spouseNif);
+ cy.get(spousePensionInput).type(spousePension);
+ cy.get(savePIT).click();
+ });
+
+ it('complete relative', () => {
+ cy.get(addRelative).click();
+ cy.get(isDescendantSelect).type('{downArrow}{downArrow}{enter}');
+ cy.get(birthedInput).type(birthed);
+ cy.get(adoptionYearInput).type(adoptionYear);
+ cy.get(saveRelative).click();
+ });
+});
diff --git a/test/cypress/integration/zone/01_basic-data.spec.js b/test/cypress/integration/zone/01_basic-data.spec.js
new file mode 100644
index 000000000..111f2495c
--- /dev/null
+++ b/test/cypress/integration/zone/01_basic-data.spec.js
@@ -0,0 +1,20 @@
+describe.skip('Zone basic data path', () => {
+ beforeEach(() => {
+ const zoneId = 1;
+ cy.viewport(1280, 720);
+ cy.login('developer');
+ cy.visit(`/#/zone/${zoneId}`);
+ });
+ it('should reach the basic data section', async () => {});
+ it('should edit de form and then save', async () => {});
+ it('should now reload the section', async () => {});
+ it('should confirm the name was updated', async () => {});
+ it('should confirm the agency was updated', async () => {});
+ it('should confirm the max volume was updated', async () => {});
+ it('should confirm the traveling days were updated', async () => {});
+ it('should confirm the closing hour was updated', async () => {});
+ it('should confirm the price was updated', async () => {});
+ it('should confirm the bonus was updated', async () => {});
+ it('should confirm the inflation was updated', async () => {});
+ it('should confirm the volumetric checkbox was checked', async () => {});
+});
diff --git a/test/cypress/integration/zone/02_descriptor.spec.js b/test/cypress/integration/zone/02_descriptor.spec.js
new file mode 100644
index 000000000..02f2d30ce
--- /dev/null
+++ b/test/cypress/integration/zone/02_descriptor.spec.js
@@ -0,0 +1,10 @@
+describe('Zone descriptor path', () => {
+ beforeEach(() => {
+ const zoneId = 1;
+ cy.viewport(1280, 720);
+ cy.login('developer');
+ cy.visit(`/#/zone/${zoneId}`);
+ });
+ it('should eliminate the zone using the descriptor option', async () => {});
+ it('should search for the deleted zone to find no results', async () => {});
+});
diff --git a/test/cypress/support/commands.js b/test/cypress/support/commands.js
index cb1d3de44..21121d9df 100755
--- a/test/cypress/support/commands.js
+++ b/test/cypress/support/commands.js
@@ -86,11 +86,17 @@ Cypress.Commands.add('getValue', (selector) => {
});
// Fill Inputs
-Cypress.Commands.add('selectOption', (selector, option) => {
+Cypress.Commands.add('selectOption', (selector, option, timeout) => {
cy.waitForElement(selector);
cy.get(selector).click();
+ cy.wait(timeout || 1000);
cy.get('.q-menu .q-item').contains(option).click();
});
+Cypress.Commands.add('countSelectOptions', (selector, option) => {
+ cy.waitForElement(selector);
+ cy.get(selector).click();
+ cy.get('.q-menu .q-item').should('have.length', option);
+});
Cypress.Commands.add('fillInForm', (obj, form = '.q-form > .q-card') => {
cy.waitForElement('.q-form > .q-card');
diff --git a/test/cypress/support/index.js b/test/cypress/support/index.js
index 4385698ec..c57c1a303 100644
--- a/test/cypress/support/index.js
+++ b/test/cypress/support/index.js
@@ -15,3 +15,19 @@
import './commands';
+function randomString(options = { length: 10 }) {
+ let possible = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz';
+ return randomizeValue(possible, options);
+}
+
+function randomNumber(options = { length: 10 }) {
+ let possible = '0123456789';
+ return randomizeValue(possible, options);
+}
+
+function randomizeValue(characterSet, options) {
+ return Array.from({ length: options.length }, () =>
+ characterSet.charAt(Math.floor(Math.random() * characterSet.length))
+ ).join('');
+}
+export { randomString, randomNumber, randomizeValue };
diff --git a/test/vitest/__tests__/components/common/VnLinkPhone.spec.js b/test/vitest/__tests__/components/common/VnLinkPhone.spec.js
index e460ab2fc..e31bff4a8 100644
--- a/test/vitest/__tests__/components/common/VnLinkPhone.spec.js
+++ b/test/vitest/__tests__/components/common/VnLinkPhone.spec.js
@@ -2,6 +2,14 @@ import { describe, it, expect } from 'vitest';
import parsePhone from 'src/filters/parsePhone';
describe('parsePhone filter', () => {
+ it('no phone', () => {
+ const resultado = parsePhone(null, '34');
+ expect(resultado).toBe(undefined);
+ });
+ it('no phone and no prefix', () => {
+ const resultado = parsePhone(null, null);
+ expect(resultado).toBe(undefined);
+ });
it("adds prefix +34 if it doesn't have one", () => {
const resultado = parsePhone('123456789', '34');
expect(resultado).toBe('34123456789');
diff --git a/test/vitest/__tests__/composables/downloadFile.spec.js b/test/vitest/__tests__/composables/downloadFile.spec.js
index f611479bf..f53b56b3e 100644
--- a/test/vitest/__tests__/composables/downloadFile.spec.js
+++ b/test/vitest/__tests__/composables/downloadFile.spec.js
@@ -1,25 +1,36 @@
-import { vi, describe, expect, it } from 'vitest';
+import { vi, describe, expect, it, beforeAll, afterAll } from 'vitest';
import { axios } from 'app/test/vitest/helper';
import { downloadFile } from 'src/composables/downloadFile';
import { useSession } from 'src/composables/useSession';
-
const session = useSession();
const token = session.getToken();
describe('downloadFile', () => {
+ const baseUrl = 'http://localhost:9000';
+ let defaulCreateObjectURL;
+
+ beforeAll(() => {
+ defaulCreateObjectURL = window.URL.createObjectURL;
+ window.URL.createObjectURL = vi.fn(() => 'blob:http://localhost:9000/blob-id');
+ });
+
+ afterAll(() => (window.URL.createObjectURL = defaulCreateObjectURL));
+
it('should open a new window to download the file', async () => {
- const url = 'http://localhost:9000';
-
- vi.spyOn(axios, 'get').mockResolvedValueOnce({ data: url });
-
- const mockWindowOpen = vi.spyOn(window, 'open');
+ const res = {
+ data: new Blob(['file content'], { type: 'application/octet-stream' }),
+ headers: { 'content-disposition': 'attachment; filename="test-file.txt"' },
+ };
+ vi.spyOn(axios, 'get').mockImplementation((url) => {
+ if (url == 'Urls/getUrl') return Promise.resolve({ data: baseUrl });
+ else if (url.includes('downloadFile')) return Promise.resolve(res);
+ });
await downloadFile(1);
- expect(mockWindowOpen).toHaveBeenCalledWith(
- `${url}/api/dms/1/downloadFile?access_token=${token}`
+ expect(axios.get).toHaveBeenCalledWith(
+ `${baseUrl}/api/dms/1/downloadFile?access_token=${token}`,
+ { responseType: 'blob' }
);
-
- mockWindowOpen.mockRestore();
});
});