diff --git a/src/components/common/__tests__/VnInputTime.spec.js b/src/components/common/__tests__/VnInputTime.spec.js
new file mode 100644
index 000000000..2692ac71b
--- /dev/null
+++ b/src/components/common/__tests__/VnInputTime.spec.js
@@ -0,0 +1,63 @@
+import { vi, describe, expect, it, beforeAll, afterEach } from 'vitest';
+import { createWrapper } from 'app/test/vitest/helper';
+import VnInputTime from 'components/common/VnInputTime.vue';
+
+describe('VnInputTime', () => {
+ let wrapper;
+ let vm;
+
+ beforeAll(() => {
+ wrapper = createWrapper(VnInputTime, {
+ props: {
+ isOutlined: true,
+ timeOnly: false,
+ },
+ });
+ vm = wrapper.vm;
+ wrapper = wrapper.wrapper;
+ });
+
+ afterEach(() => {
+ vi.clearAllMocks();
+ });
+
+ it('should return the correct data if isOutlined is true', () => {
+ expect(vm.isOutlined).toBe(true);
+ expect(vm.styleAttrs).toEqual({ dense: true, outlined: true, rounded: true });
+ });
+
+ it('should return the formatted data', () => {
+ expect(vm.dateToTime('2022-01-01T03:23:43')).toBe('03:23');
+ });
+
+ describe('formattedTime', () => {
+ it('should return the formatted time for a valid ISO date', () => {
+ vm.model = '2025-01-02T15:45:00';
+ expect(vm.formattedTime).toBe('15:45');
+ });
+
+ it('should handle null model value gracefully', () => {
+ vm.model = null;
+ expect(vm.formattedTime).toBe(null);
+ });
+
+ it('should handle time-only input correctly', async () => {
+ await wrapper.setProps({ timeOnly: true });
+ vm.formattedTime = '14:30';
+ expect(vm.model).toBe('14:30');
+ });
+
+ it('should pad short time values correctly', async () => {
+ await wrapper.setProps({ timeOnly: true });
+ vm.formattedTime = '9';
+ expect(vm.model).toBe('09:00');
+ });
+
+ it('should not update the model if the value is unchanged', () => {
+ vm.model = '14:30';
+ const previousModel = vm.model;
+ vm.formattedTime = '14:30';
+ expect(vm.model).toBe(previousModel);
+ });
+ });
+});
\ No newline at end of file
diff --git a/src/components/ui/CardSummary.vue b/src/components/ui/CardSummary.vue
index a1de3eee3..cf52bcd40 100644
--- a/src/components/ui/CardSummary.vue
+++ b/src/components/ui/CardSummary.vue
@@ -2,7 +2,6 @@
import { ref, computed, watch, onBeforeMount } from 'vue';
import { useRoute } from 'vue-router';
import SkeletonSummary from 'components/ui/SkeletonSummary.vue';
-import VnLv from 'src/components/ui/VnLv.vue';
import { useArrayData } from 'src/composables/useArrayData';
import { isDialogOpened } from 'src/filters';
import VnMoreOptions from './VnMoreOptions.vue';
diff --git a/src/components/ui/SkeletonSummary.vue b/src/components/ui/SkeletonSummary.vue
index e8407ee7b..659d4c53d 100644
--- a/src/components/ui/SkeletonSummary.vue
+++ b/src/components/ui/SkeletonSummary.vue
@@ -1,38 +1,49 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/src/components/ui/__tests__/CardSummary.spec.js b/src/components/ui/__tests__/CardSummary.spec.js
new file mode 100644
index 000000000..411ebf9bb
--- /dev/null
+++ b/src/components/ui/__tests__/CardSummary.spec.js
@@ -0,0 +1,78 @@
+import { vi, describe, expect, it, beforeAll, afterEach, beforeEach } from 'vitest';
+import { createWrapper, axios } from 'app/test/vitest/helper';
+import CardSummary from 'src/components/ui/CardSummary.vue';
+import * as vueRouter from 'vue-router';
+
+describe('CardSummary', () => {
+ let vm;
+ let wrapper;
+
+ beforeAll(() => {
+ vi.spyOn(axios, 'get').mockResolvedValue({ data: [] });
+ });
+
+ vi.spyOn(vueRouter, 'useRoute').mockReturnValue({
+ query: {},
+ params: {},
+ meta: { moduleName: 'mockName' },
+ path: 'mockName/1/summary',
+ name: 'CardSummary',
+ });
+
+ beforeEach(() => {
+ wrapper = createWrapper(CardSummary, {
+ propsData: {
+ dataKey: 'cardSummaryKey',
+ url: 'cardSummaryUrl',
+ filter: 'cardFilter',
+ },
+ });
+ vm = wrapper.vm;
+ wrapper = wrapper.wrapper;
+ });
+
+ afterEach(() => {
+ vi.clearAllMocks();
+ });
+
+ it('should fetch data correctly', async () => {
+ const fetchSpy = vi
+ .spyOn(vm.arrayData, 'fetch')
+ .mockResolvedValue({ data: [{ id: 1, name: 'Test Entity' }] });
+ await vm.fetch();
+
+ expect(fetchSpy).toHaveBeenCalledWith({ append: false, updateRouter: false });
+ expect(wrapper.emitted('onFetch')).toBeTruthy();
+ expect(vm.isLoading).toBe(false);
+ });
+
+ it('should set correct props to the store', () => {
+ expect(vm.store.url).toEqual('cardSummaryUrl');
+ expect(vm.store.filter).toEqual('cardFilter');
+ });
+
+ it('should compute entity correctly from store data', () => {
+ vm.store.data = [{ id: 1, name: 'Entity 1' }];
+ expect(vm.entity).toEqual({ id: 1, name: 'Entity 1' });
+ });
+
+ it('should handle empty data gracefully', () => {
+ vm.store.data = [];
+ expect(vm.entity).toBeUndefined();
+ });
+
+ it('should respond to prop changes and refetch data', async () => {
+ const newUrl = 'CardSummary/35';
+ const newKey = 'cardSummaryKey/35';
+ const fetchSpy = vi.spyOn(vm.arrayData, 'fetch');
+ await wrapper.setProps({ url: newUrl, filter: { key: newKey } });
+
+ expect(fetchSpy).toHaveBeenCalled();
+ expect(vm.store.url).toBe(newUrl);
+ expect(vm.store.filter).toEqual({ key: newKey });
+ });
+
+ it('should return true if route path ends with /summary' , () => {
+ expect(vm.isSummary).toBe(true);
+ });
+});
\ No newline at end of file
diff --git a/src/pages/Customer/components/CustomerAddressCreate.vue b/src/pages/Customer/components/CustomerAddressCreate.vue
index bc4d6a128..32b4078db 100644
--- a/src/pages/Customer/components/CustomerAddressCreate.vue
+++ b/src/pages/Customer/components/CustomerAddressCreate.vue
@@ -11,6 +11,7 @@ import VnInput from 'src/components/common/VnInput.vue';
import VnSelect from 'src/components/common/VnSelect.vue';
import VnSelectDialog from 'src/components/common/VnSelectDialog.vue';
import CustomerNewCustomsAgent from 'src/pages/Customer/components/CustomerNewCustomsAgent.vue';
+import VnInputNumber from 'src/components/common/VnInputNumber.vue';
const { t } = useI18n();
const route = useRoute();
@@ -150,6 +151,22 @@ function onAgentCreated({ id, fiscalName }, data) {
+
+
+
+
@@ -175,4 +192,6 @@ es:
Mobile: MovĂl
Incoterms: Incoterms
Customs agent: Agente de aduanas
+ Longitude: Longitud
+ Latitude: Latitud
diff --git a/src/pages/Order/Card/OrderLines.vue b/src/pages/Order/Card/OrderLines.vue
index 6093addb5..9d802c557 100644
--- a/src/pages/Order/Card/OrderLines.vue
+++ b/src/pages/Order/Card/OrderLines.vue
@@ -208,6 +208,20 @@ async function remove(item) {
async function handleConfirm() {
const result = await confirm(route.params.id);
if (result) {
+ const sale = await axios.get(`OrderRows`, {
+ params: {
+ filter: JSON.stringify({
+ where: { orderFk: route.params.id },
+ }),
+ },
+ });
+ const ticket = await axios.get(`Sales`, {
+ params: {
+ filter: JSON.stringify({
+ where: { id: sale.data[0].saleFk },
+ }),
+ },
+ });
quasar.notify({
message: t('globals.dataSaved'),
type: 'positive',
@@ -215,7 +229,7 @@ async function handleConfirm() {
router.push({
name: 'TicketSale',
query: {
- table: JSON.stringify({ id: route.params.id }),
+ table: JSON.stringify({ id: ticket.data[0].ticketFk }),
},
});
}