feature/Address-view-refactor #123
|
@ -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",
|
||||||
|
|||||||
"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/",
|
||||||
|
|
|
@ -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
jsegarra
commented
porque lo quitamos? porque lo quitamos?
wbuezas
commented
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'],
|
||||||
|
|
|
@ -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'
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -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>
|
|
@ -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>
|
||||||
|
|
|
@ -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([]);
|
||||||
|
|
|
@ -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
|
||||||
|
};
|
||||||
|
}
|
|
@ -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 s’han 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',
|
||||||
|
|
|
@ -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',
|
||||||
|
|
|
@ -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',
|
||||||
|
|
|
@ -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',
|
||||||
|
|
|
@ -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',
|
||||||
|
|
|
@ -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
|
||||||
jsegarra
commented
y si le pasamos required tal cual, no deberia ir? 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?
wbuezas
commented
Recuerdo que en un momento en lilium se hacia algo asi por algo en especial. Pero parece que aca pasando solo Commit: 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>
|
||||||
|
|
||||||
|
|
|
@ -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)
|
||||||
)
|
)
|
||||||
"
|
"
|
||||||
>
|
>
|
||||||
|
|
|
@ -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);
|
||||||
});
|
});
|
||||||
|
|
|
@ -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');
|
||||||
});
|
});
|
||||||
|
|
Loading…
Reference in New Issue
esto es necesario?
Si, ya que aparentemente hay un bug en cypresss que en la version terminal fallan los tests.