feature/Address-view-refactor #123

Merged
jsegarra merged 6 commits from wbuezas/hedera-web-mindshore:feature/Address-view-refactor into beta 2025-03-21 14:40:28 +00:00
17 changed files with 491 additions and 156 deletions

View File

@ -59,7 +59,7 @@
"salix": "cd ../salix && gulp back", "salix": "cd ../salix && gulp back",
"db": "cd ../salix && gulp docker", "db": "cd ../salix && gulp docker",
"cy:open": "npm run db && cypress open", "cy:open": "npm run db && cypress open",
"test:e2e": "npm run db && cypress run", "test:e2e": "npm run db && cypress run --headed --config video=false",
Review

esto es necesario?

esto es necesario?
Review

Si, ya que aparentemente hay un bug en cypresss que en la version terminal fallan los tests.

Si, ya que aparentemente hay un bug en cypresss que en la version terminal fallan los tests.
"test:unit": "vitest", "test:unit": "vitest",
"build": "rm -rf dist/ ; quasar build", "build": "rm -rf dist/ ; quasar build",
"clean": "rm -rf dist/", "clean": "rm -rf dist/",

View File

@ -23,7 +23,7 @@ module.exports = configure(function (ctx) {
// app boot file (/src/boot) // app boot file (/src/boot)
// --> boot files are part of "main.js" // --> boot files are part of "main.js"
// https://v2.quasar.dev/quasar-cli-webpack/boot-files // https://v2.quasar.dev/quasar-cli-webpack/boot-files
boot: ['i18n', 'axios', 'vnDate', 'error-handler', 'app'], boot: ['i18n', 'axios', 'vnDate', 'app'],
wbuezas marked this conversation as resolved
Review

porque lo quitamos?

porque lo quitamos?
Review

Porque es innecesario, ya hacemos error handling en el archivo de axios, por esto aparecian duplicados los notify de errores.

Porque es innecesario, ya hacemos error handling en el archivo de axios, por esto aparecian duplicados los notify de errores.
// https://v2.quasar.dev/quasar-cli-webpack/quasar-config-js#Property%3A-css // https://v2.quasar.dev/quasar-cli-webpack/quasar-config-js#Property%3A-css
css: ['app.scss', 'width.scss', 'responsive.scss'], css: ['app.scss', 'width.scss', 'responsive.scss'],

View File

@ -1,66 +0,0 @@
export default async ({ app }) => {
/*
window.addEventListener('error',
e => onWindowError(e));
window.addEventListener('unhandledrejection',
e => onWindowRejection(e));
,onWindowError(event) {
errorHandler(event.error);
}
,onWindowRejection(event) {
errorHandler(event.reason);
}
*/
app.config.errorHandler = (err, vm, info) => {
errorHandler(err, vm)
}
function errorHandler (err, vm) {
let message
let tMessage
let res = err.response
// XXX: Compatibility with old JSON service
if (err.name === 'JsonException') {
res = {
status: err.statusCode,
data: { error: { message: err.message } }
}
}
if (res) {
const status = res.status
if (status >= 400 && status < 500) {
switch (status) {
case 401:
tMessage = 'loginFailed'
break
case 403:
tMessage = 'authenticationRequired'
vm.$router.push('/login')
break
case 404:
tMessage = 'notFound'
break
default:
message = res.data.error.message
}
} else if (status >= 500) {
tMessage = 'internalServerError'
}
} else {
tMessage = 'somethingWentWrong'
console.error(err)
}
if (tMessage) {
message = vm.$t(tMessage)
}
vm.$q.notify({
message,
type: 'negative'
})
}
}

View File

@ -0,0 +1,231 @@
<script setup>
import { ref, inject, onMounted, computed, Teleport, watch } from 'vue';
import { useI18n } from 'vue-i18n';
import { useAppStore } from 'stores/app';
import { storeToRefs } from 'pinia';
import useNotify from 'src/composables/useNotify.js';
const props = defineProps({
title: {
type: String,
default: ''
},
table: {
type: String,
default: ''
},
schema: {
type: String,
default: ''
},
// Objeto con los datos iniciales del form, si este objeto es definido, no se ejecuta la query fetch
formInitialData: {
type: Object,
default: () => {}
},
autoLoad: {
type: Boolean,
default: true
},
defaultActions: {
type: Boolean,
default: true
},
showBottomActions: {
type: Boolean,
default: false
},
saveFn: {
type: Function,
default: null
},
separationBetweenInputs: {
type: String,
default: 'xs'
},
url: {
type: String,
default: ''
},
urlUpdate: {
type: String,
default: null
},
urlCreate: {
type: String,
default: null
},
observeFormChanges: {
type: Boolean,
default: true,
description:
'Esto se usa principalmente para permitir guardar sin hacer cambios (Útil para la feature de clonar ya que en este caso queremos poder guardar de primeras)'
},
mapper: {
type: Function,
default: null
},
filter: {
type: Object,
default: null
}
});
const emit = defineEmits(['onDataSaved', 'onDataFetched']);
const api = inject('api');
const { t } = useI18n();
const { notify } = useNotify();
const appStore = useAppStore();
const { isHeaderMounted } = storeToRefs(appStore);
const isLoading = ref(false);
const formData = ref({});
const formModelRef = ref(null);
const hasChanges = ref(!props.observeFormChanges);
const isResetting = ref(false);
const originalData = ref(null);
const separationBetweenInputs = computed(() => {
return `q-gutter-y-${props.separationBetweenInputs}`;
});
const onSubmitSuccess = () => {
emit('onDataSaved');
notify(t('dataSaved'), 'positive');
};
onMounted(async () => {
if (!props.formInitialData) {
if (props.autoLoad && props.url) await fetch();
originalData.value = { ...formData.value };
} else {
formData.value = { ...props.formInitialData };
originalData.value = { ...props.formInitialData };
}
if (props.observeFormChanges) {
watch(
() => formData.value,
(newVal, oldVal) => {
if (!oldVal) return;
hasChanges.value =
!isResetting.value &&
JSON.stringify(newVal) !==
JSON.stringify(originalData.value);
isResetting.value = false;
},
{ deep: true }
);
}
});
async function fetch() {
try {
let { data } = await api.get(props.url, {
params: { filter: JSON.stringify(props.filter) }
});
if (Array.isArray(data)) data = data[0] ?? {};
formData.value = { ...data };
emit('onDataFetched', formData.value);
} catch (e) {
throw e;
}
}
const submitForm = async evt => {
const isFormValid = await formModelRef.value.validate();
if (isFormValid) await save(evt);
};
async function save() {
if (props.observeFormChanges && !hasChanges.value)
return notify('noChanges', 'negative');
isLoading.value = true;
try {
const body = props.mapper
? props.mapper(formData.value, originalData.value)
: formData.value;
const method = props.urlCreate ? 'post' : 'patch';
const url = props.urlCreate || props.urlUpdate || props.url;
await Promise.resolve(
props.saveFn ? props.saveFn(body) : api[method](url, body)
);
onSubmitSuccess();
hasChanges.value = false;
} finally {
isLoading.value = false;
}
}
defineExpose({
formData,
submitForm
});
</script>
<template>
<QCard class="form-container" v-bind="$attrs">
<QForm ref="formModelRef" class="form" :class="separationBetweenInputs">
<span v-if="title" class="text-h6 text-bold">
{{ title }}
</span>
<slot name="form" :data="formData" />
<slot name="extraForm" :data="formData" />
<component
v-if="isHeaderMounted"
:is="showBottomActions ? 'div' : Teleport"
to="#actions"
class="flex row justify-end q-gutter-x-sm"
:class="{ 'q-mt-md': showBottomActions }"
>
<QBtn
v-if="defaultActions && showBottomActions"
:label="t('cancel')"
:icon="showBottomActions ? undefined : 'check'"
rounded
no-caps
flat
v-close-popup
>
<QTooltip>{{ t('cancel') }}</QTooltip>
</QBtn>
<QBtn
v-if="defaultActions"
:label="t('save')"
:icon="showBottomActions ? undefined : 'check'"
rounded
no-caps
flat
:loading="isLoading"
:disabled="!showBottomActions && !hasChanges"
@click="submitForm()"
data-cy="formModelDefaultSaveButton"
>
<QTooltip>{{ t('save') }}</QTooltip>
</QBtn>
<slot name="actions" :data="formData" />
</component>
</QForm>
</QCard>
</template>
<style lang="scss" scoped>
.form-container {
width: 100%;
height: max-content;
padding: 32px;
max-width: 544px;
display: flex;
justify-content: center;
}
.form {
display: flex;
flex-direction: column;
width: 100%;
}
</style>

View File

@ -1,15 +1,20 @@
<script setup> <script setup>
import { computed, ref } from 'vue'; import { computed, ref, useAttrs, nextTick } from 'vue';
import { useI18n } from 'vue-i18n'; import { useI18n } from 'vue-i18n';
import { useRequired } from 'src/composables/useRequired';
const $attrs = useAttrs();
const { isRequired, requiredFieldRule } = useRequired($attrs);
const { t } = useI18n();
const emit = defineEmits([ const emit = defineEmits([
'update:modelValue', 'update:modelValue',
'update:options', 'update:options',
'keyup.enter', 'keyup.enter',
'remove' 'remove',
'blur'
]); ]);
const props = defineProps({ const $props = defineProps({
modelValue: { modelValue: {
type: [String, Number], type: [String, Number],
default: null default: null
@ -25,24 +30,43 @@ const props = defineProps({
clearable: { clearable: {
type: Boolean, type: Boolean,
default: true default: true
},
emptyToNull: {
type: Boolean,
default: true
},
insertable: {
type: Boolean,
default: false
},
maxlength: {
type: Number,
default: null
},
uppercase: {
type: Boolean,
default: false
} }
}); });
const { t } = useI18n();
const requiredFieldRule = val => !!val || t('globals.fieldRequired');
const vnInputRef = ref(null); const vnInputRef = ref(null);
const value = computed({ const value = computed({
get() { get() {
return props.modelValue; return $props.modelValue;
}, },
set(value) { set(value) {
if ($props.emptyToNull && value === '') value = null;
emit('update:modelValue', value); emit('update:modelValue', value);
} }
}); });
const hover = ref(false); const hover = ref(false);
const styleAttrs = computed(() => { const styleAttrs = computed(() => {
return props.isOutlined return $props.isOutlined
? { dense: true, outlined: true, rounded: true } ? {
dense: true,
outlined: true,
rounded: true
}
: {}; : {};
}); });
@ -51,66 +75,119 @@ const focus = () => {
}; };
defineExpose({ defineExpose({
focus focus,
vnInputRef
}); });
const inputRules = [ const mixinRules = [
requiredFieldRule,
...($attrs.rules ?? []),
val => { val => {
const { maxlength } = vnInputRef.value;
if (maxlength && +val.length > maxlength)
return t(`maxLength`, { value: maxlength });
const { min, max } = vnInputRef.value.$attrs; const { min, max } = vnInputRef.value.$attrs;
if (min >= 0) { if (!min) return null;
if (min >= 0)
if (Math.floor(val) < min) return t('inputMin', { value: min }); if (Math.floor(val) < min) return t('inputMin', { value: min });
} if (!max) return null;
if (max > 0) { if (max > 0) {
if (Math.floor(val) > max) return t('inputMax', { value: max }); if (Math.floor(val) > max) return t('inputMax', { value: max });
} }
} }
]; ];
const handleKeydown = e => {
if (e.key === 'Backspace') return;
if ($props.insertable && e.key.match(/[0-9]/)) {
handleInsertMode(e);
}
};
const handleInsertMode = e => {
e.preventDefault();
const input = e.target;
const cursorPos = input.selectionStart;
const { maxlength } = vnInputRef.value;
let currentValue = value.value;
if (!currentValue) currentValue = e.key;
const newValue = e.key;
if (newValue && !isNaN(newValue) && cursorPos < maxlength) {
value.value =
currentValue.substring(0, cursorPos) +
newValue +
currentValue.substring(cursorPos + 1);
}
nextTick(() => {
input.setSelectionRange(cursorPos + 1, cursorPos + 1);
});
};
const handleUppercase = () => {
value.value = value.value?.toUpperCase() || '';
};
</script> </script>
<template> <template>
<div <div @mouseover="hover = true" @mouseleave="hover = false">
:rules="$attrs.required ? [requiredFieldRule] : null"
@mouseover="hover = true"
@mouseleave="hover = false"
>
<QInput <QInput
ref="vnInputRef" ref="vnInputRef"
v-model="value" v-model="value"
v-bind="{ ...$attrs, ...styleAttrs }" v-bind="{ ...$attrs, ...styleAttrs }"
:type="$attrs.type" :type="$attrs.type"
:class="{ required: $attrs.required }" :class="{ required: isRequired }"
@keyup.enter="emit('keyup.enter')"
@blur="emit('blur')"
@keydown="handleKeydown"
:clearable="false" :clearable="false"
:rules="inputRules" :rules="mixinRules"
:lazy-rules="true" :lazy-rules="true"
hide-bottom-space hide-bottom-space
@keyup.enter="emit('keyup.enter')" :data-cy="$attrs['data-cy'] ?? $attrs.label + '_input'"
> >
<template <template #prepend v-if="$slots.prepend">
v-if="$slots.prepend"
#prepend
>
<slot name="prepend" /> <slot name="prepend" />
</template> </template>
<template #append> <template #append>
<slot
v-if="$slots.append && !$attrs.disabled"
name="append"
/>
<QIcon <QIcon
v-if="hover && value && !$attrs.disabled && props.clearable"
name="close" name="close"
size="xs" size="xs"
:style="{
visibility:
hover &&
value &&
!$attrs.disabled &&
!$attrs.readonly &&
$props.clearable
? 'visible'
: 'hidden'
}"
@click=" @click="
() => { () => {
value = null; value = null;
vnInputRef.focus();
emit('remove'); emit('remove');
} }
" "
/> ></QIcon>
<QIcon <QIcon
v-if="info" name="match_case"
name="info" size="xs"
v-if="
!$attrs.disabled && !$attrs.readonly && $props.uppercase
"
@click="handleUppercase"
class="uppercase-icon"
> >
<QTooltip>
{{ t('Convert to uppercase') }}
</QTooltip>
</QIcon>
<slot name="append" v-if="$slots.append && !$attrs.disabled" />
<QIcon v-if="info" name="info">
<QTooltip max-width="350px"> <QTooltip max-width="350px">
{{ info }} {{ info }}
</QTooltip> </QTooltip>
@ -120,20 +197,44 @@ const inputRules = [
</div> </div>
</template> </template>
<style>
.uppercase-icon {
transition:
color 0.3s,
transform 0.2s;
cursor: pointer;
}
.uppercase-icon:hover {
color: #ed9937;
transform: scale(1.2);
}
</style>
<i18n lang="yaml"> <i18n lang="yaml">
en-US: en-US:
inputMin: Must be more than {value} inputMin: Must be more than {value}
maxLength: The value exceeds {value} characters
inputMax: Must be less than {value} inputMax: Must be less than {value}
Convert to uppercase: Convert to uppercase
es-ES: es-ES:
inputMin: Debe ser mayor a {value} inputMin: Debe ser mayor a {value}
maxLength: El valor excede los {value} carácteres
inputMax: Debe ser menor a {value} inputMax: Debe ser menor a {value}
Convert to uppercase: Convertir a mayúsculas
ca-ES: ca-ES:
inputMin: Ha de ser més gran que {value} inputMin: Ha de ser més gran que {value}
inputMax: Ha de ser menys que {value} maxLength: El valor excedeix els {value} caràcters
inputMax: Ha de ser menor que {value}
Convert to uppercase: Convertir a majúscules
fr-FR: fr-FR:
inputMin: Doit être supérieur à {value} inputMin: Doit être supérieur à {value}
inputMax: Doit être supérieur à {value} maxLength: La valeur dépasse {value} caractères
inputMax: Doit être inférieur à {value}
Convert to uppercase: Convertir en majuscules
pt-PT: pt-PT:
inputMin: Deve ser maior que {value} inputMin: Deve ser maior que {value}
inputMax: Deve ser maior que {value} maxLength: O valor excede {value} caracteres
inputMax: Deve ser menor que {value}
Convert to uppercase: Converter para maiúsculas
</i18n> </i18n>

View File

@ -67,7 +67,7 @@ const $props = defineProps({
}); });
const { t } = useI18n(); const { t } = useI18n();
const requiredFieldRule = val => val ?? t('globals.fieldRequired'); const requiredFieldRule = val => val ?? t('fieldRequired');
const { optionLabel, optionValue, options } = toRefs($props); const { optionLabel, optionValue, options } = toRefs($props);
const myOptions = ref([]); const myOptions = ref([]);

View File

@ -0,0 +1,17 @@
import { useI18n } from 'vue-i18n';
export function useRequired($attrs) {
const { t } = useI18n();
const isRequired =
typeof $attrs['required'] === 'boolean'
? $attrs['required']
: Object.keys($attrs).includes('required');
const requiredFieldRule = val =>
isRequired ? !!val || t('fieldRequired') : null;
return {
isRequired,
requiredFieldRule
};
}

View File

@ -134,6 +134,8 @@ export default {
introduceSearchTerm: 'Introdueix un terme de cerca', introduceSearchTerm: 'Introdueix un terme de cerca',
noOrdersFound: `No s'han trobat comandes`, noOrdersFound: `No s'han trobat comandes`,
send: 'Enviar', send: 'Enviar',
fieldRequired: 'Aquest camp és obligatori',
noChanges: 'No shan fet canvis',
// Image related translations // Image related translations
'Cant lock cache': 'No es pot bloquejar la memòria cau', 'Cant lock cache': 'No es pot bloquejar la memòria cau',
'Bad file format': 'Format de fitxer no reconegut', 'Bad file format': 'Format de fitxer no reconegut',

View File

@ -167,6 +167,8 @@ export default {
introduceSearchTerm: 'Enter a search term', introduceSearchTerm: 'Enter a search term',
noOrdersFound: 'No orders found', noOrdersFound: 'No orders found',
send: 'Send', send: 'Send',
fieldRequired: 'Field required',
noChanges: 'No changes',
// Image related translations // Image related translations
'Cant lock cache': 'The cache could not be blocked', 'Cant lock cache': 'The cache could not be blocked',
'Bad file format': 'Unrecognized file format', 'Bad file format': 'Unrecognized file format',

View File

@ -166,6 +166,8 @@ export default {
introduceSearchTerm: 'Introduce un término de búsqueda', introduceSearchTerm: 'Introduce un término de búsqueda',
noOrdersFound: 'No se encontrado pedidos', noOrdersFound: 'No se encontrado pedidos',
send: 'Enviar', send: 'Enviar',
fieldRequired: 'Campo requerido',
noChanges: 'No se han hecho cambios',
// Image related translations // Image related translations
'Cant lock cache': 'La caché no pudo ser bloqueada', 'Cant lock cache': 'La caché no pudo ser bloqueada',
'Bad file format': 'Formato de archivo no reconocido', 'Bad file format': 'Formato de archivo no reconocido',

View File

@ -134,6 +134,8 @@ export default {
introduceSearchTerm: 'Entrez un terme de recherche', introduceSearchTerm: 'Entrez un terme de recherche',
noOrdersFound: 'Aucune commande trouvée', noOrdersFound: 'Aucune commande trouvée',
send: 'Envoyer', send: 'Envoyer',
fieldRequired: 'Champ obligatoire',
noChanges: 'Aucun changement',
// Image related translations // Image related translations
'Cant lock cache': "Le cache n'a pas pu être verrouillé", 'Cant lock cache': "Le cache n'a pas pu être verrouillé",
'Bad file format': 'Format de fichier non reconnu', 'Bad file format': 'Format de fichier non reconnu',

View File

@ -133,6 +133,8 @@ export default {
introduceSearchTerm: 'Digite um termo de pesquisa', introduceSearchTerm: 'Digite um termo de pesquisa',
noOrdersFound: 'Nenhum pedido encontrado', noOrdersFound: 'Nenhum pedido encontrado',
send: 'Enviar', send: 'Enviar',
fieldRequired: 'Campo obrigatório',
noChanges: 'Sem alterações',
// Image related translations // Image related translations
'Cant lock cache': 'O cache não pôde ser bloqueado', 'Cant lock cache': 'O cache não pôde ser bloqueado',
'Bad file format': 'Formato de arquivo inválido', 'Bad file format': 'Formato de arquivo inválido',

View File

@ -5,32 +5,26 @@ import { useRouter, useRoute } from 'vue-router';
import VnInput from 'src/components/common/VnInput.vue'; import VnInput from 'src/components/common/VnInput.vue';
import VnSelect from 'src/components/common/VnSelect.vue'; import VnSelect from 'src/components/common/VnSelect.vue';
import VnForm from 'src/components/common/VnForm.vue'; import FormModel from 'src/components/common/FormModel.vue';
import { useUserStore } from 'stores/user';
import { useAppStore } from 'stores/app'; import { useAppStore } from 'stores/app';
import { storeToRefs } from 'pinia'; import { storeToRefs } from 'pinia';
const router = useRouter(); const router = useRouter();
const route = useRoute(); const route = useRoute();
const { t } = useI18n(); const { t } = useI18n();
const jApi = inject('jApi'); const api = inject('api');
const appStore = useAppStore(); const appStore = useAppStore();
const userStore = useUserStore();
const { isHeaderMounted } = storeToRefs(appStore); const { isHeaderMounted } = storeToRefs(appStore);
const vnFormRef = ref(null); const vnFormRef = ref(null);
const countriesOptions = ref([]); const countriesOptions = ref([]);
const provincesOptions = ref([]); const provincesOptions = ref([]);
const pks = { id: route.params.id };
const isEditMode = route.params.id !== '0'; const isEditMode = route.params.id !== '0';
const fetchAddressDataSql = { const editAddressData = ref(null);
query: ` const showForm = ref(false);
SELECT a.id, a.street, a.nickname, a.city, a.postalCode, a.provinceFk, p.countryFk
FROM myAddress a
LEFT JOIN vn.province p ON p.id = a.provinceFk
WHERE a.id = #address
`,
params: { address: route.params.id }
};
watch( watch(
() => vnFormRef?.value?.formData?.countryFk, () => vnFormRef?.value?.formData?.countryFk,
@ -40,23 +34,49 @@ watch(
const goBack = () => router.push({ name: 'addressesList' }); const goBack = () => router.push({ name: 'addressesList' });
const getCountries = async () => { const getCountries = async () => {
countriesOptions.value = await jApi.query( const filter = { fields: ['id', 'name'], order: 'name' };
`SELECT id, name FROM vn.country const { data } = await api.get('Countries', {
ORDER BY name` params: { filter: JSON.stringify(filter) }
); });
countriesOptions.value = data;
}; };
const getProvinces = async countryFk => { const getProvinces = async countryFk => {
if (!countryFk) return; if (!countryFk) return;
provincesOptions.value = await jApi.query(
`SELECT id, name FROM vn.province const filter = {
WHERE countryFk = #id where: { countryFk },
ORDER BY name`, fields: ['id', 'name'],
{ id: countryFk } order: 'name'
); };
const { data } = await api.get('Provinces', {
params: { filter: JSON.stringify(filter) }
});
provincesOptions.value = data;
}; };
onMounted(() => getCountries()); const getAddressDetails = async () => {
const { data } = await api.get(`Addresses/${route.params.id}`);
if (!data) return;
const { nickname, street, city, postalCode, province, provinceFk } = data;
editAddressData.value = {
nickname,
street,
city,
postalCode,
countryFk: province?.countryFk,
provinceFk
};
};
onMounted(async () => {
if (isEditMode) {
await getAddressDetails();
}
getCountries();
showForm.value = true;
});
</script> </script>
<template> <template>
@ -74,42 +94,49 @@ onMounted(() => getCountries());
</QTooltip> </QTooltip>
</QBtn> </QBtn>
</Teleport> </Teleport>
<VnForm <FormModel
v-if="showForm"
ref="vnFormRef" ref="vnFormRef"
:fetch-form-data-sql="fetchAddressDataSql" :url-create="
:columns-to-ignore-update="['countryFk']" !isEditMode
:create-model-default="{ ? `Clients/${userStore?.user?.id}/createAddress`
field: 'clientFk', : ''
value: 'account.myUser_getId()' "
}" :url-update="
:pks="pks" isEditMode
:is-edit-mode="isEditMode" ? `Clients/${userStore?.user?.id}/updateAddress/${route?.params?.id}`
: ''
"
:form-initial-data="editAddressData"
:title="t(isEditMode ? 'editAddress' : 'addAddress')" :title="t(isEditMode ? 'editAddress' : 'addAddress')"
table="myAddress"
schema="hedera"
@on-data-saved="goBack()" @on-data-saved="goBack()"
:show-bottom-actions="false"
> >
<template #form="{ data }"> <template #form="{ data }">
<VnInput <VnInput
v-model="data.nickname" v-model="data.nickname"
:label="t('name')" :label="t('name')"
data-cy="addressFormNickname" data-cy="addressFormNickname"
required
/> />
<VnInput <VnInput
v-model="data.street" v-model="data.street"
:label="t('address')" :label="t('address')"
data-cy="addressFormStreet" data-cy="addressFormStreet"
required
/> />
<VnInput <VnInput
v-model="data.city" v-model="data.city"
:label="t('city')" :label="t('city')"
data-cy="addressFormCity" data-cy="addressFormCity"
required
/> />
<VnInput <VnInput
v-model="data.postalCode" v-model="data.postalCode"
type="number" type="number"
:label="t('postalCode')" :label="t('postalCode')"
data-cy="addressFormPostcode" data-cy="addressFormPostcode"
required
/> />
<VnSelect <VnSelect
v-model="data.countryFk" v-model="data.countryFk"
@ -117,15 +144,17 @@ onMounted(() => getCountries());
:options="countriesOptions" :options="countriesOptions"
@update:model-value="data.provinceFk = null" @update:model-value="data.provinceFk = null"
data-cy="addressFormCountry" data-cy="addressFormCountry"
required
/> />
<VnSelect <VnSelect
v-model="data.provinceFk" v-model="data.provinceFk"
:label="t('province')" :label="t('province')"
:options="provincesOptions" :options="provincesOptions"
data-cy="addressFormProvince" data-cy="addressFormProvince"
required

y si le pasamos required tal cual, no deberia ir?
O es porque no estamos comporbando que la key de required exista en los $attrs?

y si le pasamos required tal cual, no deberia ir? O es porque no estamos comporbando que la key de required exista en los $attrs?

Recuerdo que en un momento en lilium se hacia algo asi por algo en especial. Pero parece que aca pasando solo required funciona igual

Commit: 5e3387fb6d

Recuerdo que en un momento en lilium se hacia algo asi por algo en especial. Pero parece que aca pasando solo `required` funciona igual Commit: https://gitea.verdnatura.es/verdnatura/hedera-web/commit/5e3387fb6d1af7ebc97b730e41cbbb6a3db58600
/> />
</template> </template>
</VnForm> </FormModel>
</QPage> </QPage>
</template> </template>

View File

@ -15,6 +15,7 @@ import { useUserStore } from 'stores/user';
const router = useRouter(); const router = useRouter();
const jApi = inject('jApi'); const jApi = inject('jApi');
const api = inject('api');
const { notify } = useNotify(); const { notify } = useNotify();
const { t } = useI18n(); const { t } = useI18n();
const { openConfirmationModal } = useVnConfirm(); const { openConfirmationModal } = useVnConfirm();
@ -56,16 +57,13 @@ const changeDefaultAddress = async () => {
notify(t('defaultAddressModified'), 'positive'); notify(t('defaultAddressModified'), 'positive');
}; };
const removeAddress = async id => { async function removeAddress(address) {
try { try {
await jApi.execQuery( await api.patch(
`START TRANSACTION; `/Clients/${userStore?.user?.id}/updateAddress/${address.id}`,
UPDATE hedera.myAddress SET isActive = FALSE
WHERE ((id = #id));
SELECT isActive FROM hedera.myAddress WHERE ((id = #id));
COMMIT`,
{ {
id ...address,
isActive: false
} }
); );
fetchAddressesRef.value.fetch(); fetchAddressesRef.value.fetch();
@ -73,7 +71,7 @@ const removeAddress = async id => {
} catch (error) { } catch (error) {
console.error('Error removing address:', error); console.error('Error removing address:', error);
} }
}; }
onMounted(async () => { onMounted(async () => {
getDefaultAddress(); getDefaultAddress();
@ -153,7 +151,7 @@ onMounted(async () => {
openConfirmationModal( openConfirmationModal(
null, null,
t('confirmDeleteAddress'), t('confirmDeleteAddress'),
() => removeAddress(address.id) () => removeAddress(address)
) )
" "
> >

View File

@ -46,15 +46,28 @@ describe('PendingOrders', () => {
.should('contain', data.postcode); .should('contain', data.postcode);
}; };
it('should fail if we enter a wrong postcode', () => {
cy.dataCy('newAddressBtn').should('exist');
cy.dataCy('newAddressBtn').click();
cy.dataCy('formModelDefaultSaveButton').should('exist');
cy.dataCy('formModelDefaultSaveButton').should('be.disabled');
const addressFormData = getRandomAddressFormData();
fillFormWithData(addressFormData);
cy.dataCy('formModelDefaultSaveButton').should('not.be.disabled');
cy.dataCy('formModelDefaultSaveButton').click();
cy.checkNotify('negative');
});
it('should create a new address', () => { it('should create a new address', () => {
cy.dataCy('newAddressBtn').should('exist'); cy.dataCy('newAddressBtn').should('exist');
cy.dataCy('newAddressBtn').click(); cy.dataCy('newAddressBtn').click();
cy.dataCy('formDefaultSaveButton').should('exist'); cy.dataCy('formModelDefaultSaveButton').should('exist');
cy.dataCy('formDefaultSaveButton').should('be.disabled'); cy.dataCy('formModelDefaultSaveButton').should('be.disabled');
const addressFormData = getRandomAddressFormData(); const addressFormData = getRandomAddressFormData();
addressFormData.postcode = '46460'; // Usamos un postcode válido
fillFormWithData(addressFormData); fillFormWithData(addressFormData);
cy.dataCy('formDefaultSaveButton').should('not.be.disabled'); cy.dataCy('formModelDefaultSaveButton').should('not.be.disabled');
cy.dataCy('formDefaultSaveButton').click(); cy.dataCy('formModelDefaultSaveButton').click();
cy.checkNotify('positive', 'Datos guardados'); cy.checkNotify('positive', 'Datos guardados');
verifyAddressCardData(addressFormData); verifyAddressCardData(addressFormData);
}); });
@ -71,9 +84,10 @@ describe('PendingOrders', () => {
}); });
// Fill form with new data // Fill form with new data
const addressFormData = getRandomAddressFormData(); const addressFormData = getRandomAddressFormData();
addressFormData.postcode = '46460'; // Usamos un postcode válido
fillFormWithData(addressFormData); fillFormWithData(addressFormData);
cy.dataCy('formDefaultSaveButton').should('not.be.disabled'); cy.dataCy('formModelDefaultSaveButton').should('not.be.disabled');
cy.dataCy('formDefaultSaveButton').click(); cy.dataCy('formModelDefaultSaveButton').click();
cy.checkNotify('positive', 'Datos guardados'); cy.checkNotify('positive', 'Datos guardados');
verifyAddressCardData(addressFormData); verifyAddressCardData(addressFormData);
}); });

View File

@ -90,5 +90,6 @@ Cypress.Commands.add('setConfirmDialog', () => {
}); });
Cypress.Commands.add('checkNotify', (status, content) => { Cypress.Commands.add('checkNotify', (status, content) => {
cy.dataCy(`${status}Notify`).should('contain', content); if (content) cy.dataCy(`${status}Notify`).should('contain', content);
else cy.dataCy(`${status}Notify`).should('exist');
}); });