Merge branch 'dev' into 8472-unifyStileForMoreCreateDialogSlot
gitea/salix-front/pipeline/pr-dev This commit looks good Details

This commit is contained in:
Jose Antonio Tubau 2025-03-10 08:24:38 +01:00
commit a9302c8033
7 changed files with 199 additions and 163 deletions

View File

@ -5,7 +5,7 @@ import SkeletonDescriptor from 'components/ui/SkeletonDescriptor.vue';
import { useArrayData } from 'composables/useArrayData'; import { useArrayData } from 'composables/useArrayData';
import { useSummaryDialog } from 'src/composables/useSummaryDialog'; import { useSummaryDialog } from 'src/composables/useSummaryDialog';
import { useState } from 'src/composables/useState'; import { useState } from 'src/composables/useState';
import { useRoute } from 'vue-router'; import { useRoute, useRouter } from 'vue-router';
import { useClipboard } from 'src/composables/useClipboard'; import { useClipboard } from 'src/composables/useClipboard';
import VnMoreOptions from './VnMoreOptions.vue'; import VnMoreOptions from './VnMoreOptions.vue';
@ -42,6 +42,7 @@ const $props = defineProps({
const state = useState(); const state = useState();
const route = useRoute(); const route = useRoute();
const router = useRouter();
const { t } = useI18n(); const { t } = useI18n();
const { copyText } = useClipboard(); const { copyText } = useClipboard();
const { viewSummary } = useSummaryDialog(); const { viewSummary } = useSummaryDialog();
@ -50,6 +51,9 @@ let store;
let entity; let entity;
const isLoading = ref(false); const isLoading = ref(false);
const isSameDataKey = computed(() => $props.dataKey === route.meta.moduleName); const isSameDataKey = computed(() => $props.dataKey === route.meta.moduleName);
const DESCRIPTOR_PROXY = 'DescriptorProxy';
const moduleName = ref();
const isSameModuleName = route.matched[1].meta.moduleName !== moduleName.value;
defineExpose({ getData }); defineExpose({ getData });
onBeforeMount(async () => { onBeforeMount(async () => {
@ -76,15 +80,18 @@ onBeforeMount(async () => {
); );
}); });
const routeName = computed(() => { function getName() {
const DESCRIPTOR_PROXY = 'DescriptorProxy';
let name = $props.dataKey; let name = $props.dataKey;
if ($props.dataKey.includes(DESCRIPTOR_PROXY)) { if ($props.dataKey.includes(DESCRIPTOR_PROXY)) {
name = name.split(DESCRIPTOR_PROXY)[0]; name = name.split(DESCRIPTOR_PROXY)[0];
} }
return `${name}Summary`; return name;
}
const routeName = computed(() => {
let routeName = getName();
return `${routeName}Summary`;
}); });
async function getData() { async function getData() {
store.url = $props.url; store.url = $props.url;
store.filter = $props.filter ?? {}; store.filter = $props.filter ?? {};
@ -120,20 +127,35 @@ function copyIdText(id) {
const emit = defineEmits(['onFetch']); const emit = defineEmits(['onFetch']);
const iconModule = computed(() => route.matched[1].meta.icon); const iconModule = computed(() => {
const toModule = computed(() => moduleName.value = getName();
route.matched[1].path.split('/').length > 2 if (isSameModuleName) {
return router.options.routes[1].children.find((r) => r.name === moduleName.value)
?.meta?.icon;
} else {
return route.matched[1].meta.icon;
}
});
const toModule = computed(() => {
moduleName.value = getName();
if (isSameModuleName) {
return router.options.routes[1].children.find((r) => r.name === moduleName.value)
?.children[0]?.redirect;
} else {
return route.matched[1].path.split('/').length > 2
? route.matched[1].redirect ? route.matched[1].redirect
: route.matched[1].children[0].redirect, : route.matched[1].children[0].redirect;
); }
});
</script> </script>
<template> <template>
<div class="descriptor"> <div class="descriptor">
<template v-if="entity && !isLoading"> <template v-if="entity && !isLoading">
<div class="header bg-primary q-pa-sm justify-between"> <div class="header bg-primary q-pa-sm justify-between">
<slot name="header-extra-action" <slot name="header-extra-action">
><QBtn <QBtn
round round
flat flat
dense dense
@ -141,13 +163,13 @@ const toModule = computed(() =>
:icon="iconModule" :icon="iconModule"
color="white" color="white"
class="link" class="link"
:to="$attrs['to-module'] ?? toModule" :to="toModule"
> >
<QTooltip> <QTooltip>
{{ t('globals.goToModuleIndex') }} {{ t('globals.goToModuleIndex') }}
</QTooltip> </QTooltip>
</QBtn></slot </QBtn>
> </slot>
<QBtn <QBtn
@click.stop="viewSummary(entity.id, $props.summary, $props.width)" @click.stop="viewSummary(entity.id, $props.summary, $props.width)"
round round
@ -230,7 +252,6 @@ const toModule = computed(() =>
</div> </div>
<slot name="after" /> <slot name="after" />
</template> </template>
<!-- Skeleton -->
<SkeletonDescriptor v-if="!entity || isLoading" /> <SkeletonDescriptor v-if="!entity || isLoading" />
</div> </div>
<QInnerLoading <QInnerLoading

View File

@ -1,6 +1,6 @@
<script setup> <script setup>
import VnCard from 'components/common/VnCard.vue'; import VnCardBeta from 'src/components/common/VnCardBeta.vue';
</script> </script>
<template> <template>
<VnCard data-key="Wagon" url="Wagons" /> <VnCardBeta data-key="Wagon" url="Wagons" :descriptor="{}" />
</template> </template>

View File

@ -8,6 +8,7 @@ import VnTable from 'src/components/VnTable/VnTable.vue';
import { computed, ref } from 'vue'; import { computed, ref } from 'vue';
import VnSelect from 'src/components/common/VnSelect.vue'; import VnSelect from 'src/components/common/VnSelect.vue';
import VnInput from 'src/components/common/VnInput.vue'; import VnInput from 'src/components/common/VnInput.vue';
import VnSection from 'src/components/common/VnSection.vue';
const quasar = useQuasar(); const quasar = useQuasar();
const arrayData = useArrayData('WagonList'); const arrayData = useArrayData('WagonList');
@ -15,6 +16,7 @@ const store = arrayData.store;
const router = useRouter(); const router = useRouter();
const { t } = useI18n(); const { t } = useI18n();
const tableRef = ref(); const tableRef = ref();
const dataKey = 'WagonList';
const filter = { const filter = {
include: { include: {
relation: 'type', relation: 'type',
@ -75,7 +77,6 @@ function navigate(id) {
} }
async function remove(row) { async function remove(row) {
try {
await axios.delete(`Wagons/${row.id}`).then(async () => { await axios.delete(`Wagons/${row.id}`).then(async () => {
quasar.notify({ quasar.notify({
message: t('wagon.list.removeItem'), message: t('wagon.list.removeItem'),
@ -84,33 +85,40 @@ async function remove(row) {
store.data.splice(store.data.indexOf(row), 1); store.data.splice(store.data.indexOf(row), 1);
window.location.reload(); window.location.reload();
}); });
} catch (error) {
//
}
} }
</script> </script>
<template> <template>
<QPage class="column items-center q-pa-md"> <QPage class="column items-center q-pa-md">
<VnSection
:data-key="dataKey"
:columns="columns"
prefix="card"
:array-data-props="{
url: 'Wagons',
exprBuilder,
order: 'id DESC',
}"
>
<template #body>
<VnTable <VnTable
ref="tableRef" ref="tableRef"
data-key="WagonList" :data-key="dataKey"
url="Wagons"
:filter="filter"
:columns="columns"
order="id DESC"
:column-search="false"
:default-mode="'card'"
:disable-option="{ table: true }"
:create="{ :create="{
urlCreate: 'Wagons', urlCreate: 'Wagons',
title: t('Create new wagon'), title: t('Create new wagon'),
onDataSaved: () => tableRef.reload(), onDataSaved: () => tableRef.reload(),
formInitialData: {}, formInitialData: {},
}" }"
:filter="filter"
:columns="columns"
:column-search="false"
:default-mode="'card'"
:disable-option="{ table: true }"
:right-search="false"
> >
<template #more-create-dialog="{ data }"> <template #more-create-dialog="{ data }">
<VnInput <VnInput
filled
v-model="data.label" v-model="data.label"
:label="t('wagon.create.label')" :label="t('wagon.create.label')"
type="number" type="number"
@ -118,19 +126,24 @@ async function remove(row) {
:rules="[(val) => !!val || t('wagon.warnings.labelNotEmpty')]" :rules="[(val) => !!val || t('wagon.warnings.labelNotEmpty')]"
/> />
<VnInput <VnInput
filled
v-model="data.plate" v-model="data.plate"
:label="t('wagon.list.plate')" :label="t('wagon.list.plate')"
:rules="[(val) => !!val || t('wagon.warnings.plateNotEmpty')]" :rules="[(val) => !!val || t('wagon.warnings.plateNotEmpty')]"
/> />
<VnInput <VnInput
filled
v-model="data.volume" v-model="data.volume"
:label="t('wagon.list.volume')" :label="t('wagon.list.volume')"
type="number" type="number"
min="0" min="0"
:rules="[(val) => !!val || t('wagon.warnings.volumeNotEmpty')]" :rules="[
(val) => !!val || t('wagon.warnings.volumeNotEmpty'),
]"
/> />
<VnSelect <VnSelect
url="WagonTypes" url="WagonTypes"
filled
v-model="data.typeFk" v-model="data.typeFk"
use-input use-input
fill-input fill-input
@ -162,6 +175,8 @@ async function remove(row) {
</VnSelect> </VnSelect>
</template> </template>
</VnTable> </VnTable>
</template>
</VnSection>
</QPage> </QPage>
</template> </template>

View File

@ -1,46 +1,16 @@
import { RouterView } from 'vue-router'; import { RouterView } from 'vue-router';
export default { const wagonCard = {
path: '/wagon', name: 'WagonCard',
name: 'Wagon', path: ':id',
component: () => import('src/pages/Wagon/Card/WagonCard.vue'),
redirect: { name: 'WagonEdit' },
meta: { meta: {
title: 'wagons', menu: ['WagonEdit'],
icon: 'vn:trolley',
moduleName: 'Wagon',
},
component: RouterView,
redirect: { name: 'WagonMain' },
menus: {
main: ['WagonList', 'WagonTypeList', 'WagonCounter', 'WagonTray'],
card: [],
}, },
children: [ children: [
{ {
path: '/wagon', path: 'edit',
name: 'WagonMain',
component: () => import('src/components/common/VnModule.vue'),
redirect: { name: 'WagonList' },
children: [
{
path: 'list',
name: 'WagonList',
meta: {
title: 'list',
icon: 'vn:trolley',
},
component: () => import('src/pages/Wagon/WagonList.vue'),
},
{
path: 'create',
name: 'WagonCreate',
meta: {
title: 'wagonCreate',
icon: 'create',
},
component: () => import('src/pages/Wagon/WagonCreate.vue'),
},
{
path: ':id/edit',
name: 'WagonEdit', name: 'WagonEdit',
meta: { meta: {
title: 'wagonEdit', title: 'wagonEdit',
@ -48,6 +18,44 @@ export default {
}, },
component: () => import('src/pages/Wagon/WagonCreate.vue'), component: () => import('src/pages/Wagon/WagonCreate.vue'),
}, },
],
};
export default {
name: 'Wagon',
path: '/wagon',
meta: {
title: 'wagons',
icon: 'vn:trolley',
moduleName: 'Wagon',
menu: ['WagonList', 'WagonTypeList', 'WagonCounter'],
},
component: RouterView,
redirect: { name: 'WagonMain' },
children: [
{
path: '',
name: 'WagonMain',
component: () => import('src/components/common/VnModule.vue'),
redirect: { name: 'WagonIndexMain' },
children: [
{
path: '',
name: 'WagonIndexMain',
redirect: { name: 'WagonList' },
component: () => import('src/pages/Wagon/WagonList.vue'),
children: [
{
name: 'WagonList',
path: 'list',
meta: {
title: 'list',
icon: 'view_list',
},
},
wagonCard,
],
},
{ {
path: 'counter', path: 'counter',
name: 'WagonCounter', name: 'WagonCounter',
@ -57,12 +65,9 @@ export default {
}, },
component: () => import('src/pages/Wagon/WagonCounter.vue'), component: () => import('src/pages/Wagon/WagonCounter.vue'),
}, },
],
},
{ {
path: '/wagon/type', path: 'type',
name: 'WagonTypeMain', name: 'WagonTypeMain',
component: () => import('src/components/common/VnModule.vue'),
redirect: { name: 'WagonTypeList' }, redirect: { name: 'WagonTypeList' },
children: [ children: [
{ {
@ -72,16 +77,8 @@ export default {
title: 'typesList', title: 'typesList',
icon: 'view_list', icon: 'view_list',
}, },
component: () => import('src/pages/Wagon/Type/WagonTypeList.vue'), component: () =>
}, import('src/pages/Wagon/Type/WagonTypeList.vue'),
{
path: 'create',
name: 'WagonTypeCreate',
meta: {
title: 'typeCreate',
icon: 'create',
},
component: () => import('src/pages/Wagon/Type/WagonTypeList.vue'),
}, },
{ {
path: ':id/edit', path: ':id/edit',
@ -90,7 +87,10 @@ export default {
title: 'typeEdit', title: 'typeEdit',
icon: 'edit', icon: 'edit',
}, },
component: () => import('src/pages/Wagon/Type/WagonTypeEdit.vue'), component: () =>
import('src/pages/Wagon/Type/WagonTypeEdit.vue'),
},
],
}, },
], ],
}, },

View File

@ -1,4 +1,4 @@
describe('Route extended list', () => { describe.skip('Route extended list', () => {
const getSelector = (colField) => `tr:last-child > [data-col-field="${colField}"]`; const getSelector = (colField) => `tr:last-child > [data-col-field="${colField}"]`;
const selectors = { const selectors = {

View File

@ -1,4 +1,4 @@
describe.skip('WagonCreate', () => { describe('WagonCreate', () => {
beforeEach(() => { beforeEach(() => {
cy.viewport(1280, 720); cy.viewport(1280, 720);
cy.login('developer'); cy.login('developer');
@ -17,7 +17,7 @@ describe.skip('WagonCreate', () => {
'.grid-create > [label="Volume"] > .q-field > .q-field__inner > .q-field__control > .q-field__control-container > [data-cy="Volume_input"]', '.grid-create > [label="Volume"] > .q-field > .q-field__inner > .q-field__control > .q-field__control-container > [data-cy="Volume_input"]',
).type('100'); ).type('100');
cy.selectOption('[data-cy="Type_select"]', '1'); cy.selectOption('[data-cy="Type_select"]', '1');
cy.dataCy('FormModelPopup_save').click();
cy.get('[title="Remove"] > .q-btn__content > .q-icon').first().click(); cy.get('[title="Remove"] > .q-btn__content > .q-icon').first().click();
}); });
}); });

View File

@ -2,7 +2,7 @@ describe('WagonTypeCreate', () => {
beforeEach(() => { beforeEach(() => {
cy.viewport(1920, 1080); cy.viewport(1920, 1080);
cy.login('developer'); cy.login('developer');
cy.visit('/#/wagon/type/create'); cy.visit('/#/wagon/type/list');
cy.waitForElement('.q-page', 6000); cy.waitForElement('.q-page', 6000);
}); });