forked from verdnatura/salix-front
Merge branch 'dev' into 6464-fixNegativeBases
This commit is contained in:
commit
f6e0cb41ce
|
@ -6,6 +6,7 @@
|
||||||
"author": "Verdnatura",
|
"author": "Verdnatura",
|
||||||
"private": true,
|
"private": true,
|
||||||
"packageManager": "pnpm@8.15.1",
|
"packageManager": "pnpm@8.15.1",
|
||||||
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"lint": "eslint --ext .js,.vue ./",
|
"lint": "eslint --ext .js,.vue ./",
|
||||||
"format": "prettier --write \"**/*.{js,vue,scss,html,md,json}\" --ignore-path .gitignore",
|
"format": "prettier --write \"**/*.{js,vue,scss,html,md,json}\" --ignore-path .gitignore",
|
||||||
|
@ -22,7 +23,7 @@
|
||||||
"chromium": "^3.0.3",
|
"chromium": "^3.0.3",
|
||||||
"croppie": "^2.6.5",
|
"croppie": "^2.6.5",
|
||||||
"pinia": "^2.1.3",
|
"pinia": "^2.1.3",
|
||||||
"quasar": "^2.14.5",
|
"quasar": "^2.16.4",
|
||||||
"validator": "^13.9.0",
|
"validator": "^13.9.0",
|
||||||
"vue": "^3.3.4",
|
"vue": "^3.3.4",
|
||||||
"vue-i18n": "^9.2.2",
|
"vue-i18n": "^9.2.2",
|
||||||
|
@ -31,7 +32,7 @@
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@intlify/unplugin-vue-i18n": "^0.8.1",
|
"@intlify/unplugin-vue-i18n": "^0.8.1",
|
||||||
"@pinia/testing": "^0.1.2",
|
"@pinia/testing": "^0.1.2",
|
||||||
"@quasar/app-vite": "^1.7.3",
|
"@quasar/app-vite": "^1.9.3",
|
||||||
"@quasar/quasar-app-extension-qcalendar": "4.0.0-beta.15",
|
"@quasar/quasar-app-extension-qcalendar": "4.0.0-beta.15",
|
||||||
"@quasar/quasar-app-extension-testing-unit-vitest": "^0.4.0",
|
"@quasar/quasar-app-extension-testing-unit-vitest": "^0.4.0",
|
||||||
"@vue/test-utils": "^2.4.4",
|
"@vue/test-utils": "^2.4.4",
|
||||||
|
@ -43,7 +44,8 @@
|
||||||
"eslint-plugin-vue": "^9.14.1",
|
"eslint-plugin-vue": "^9.14.1",
|
||||||
"postcss": "^8.4.23",
|
"postcss": "^8.4.23",
|
||||||
"prettier": "^2.8.8",
|
"prettier": "^2.8.8",
|
||||||
"vitest": "^0.31.1"
|
"vite": "^2.5.10",
|
||||||
|
"vitest": "^0.34.0"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": "^20 || ^18 || ^16",
|
"node": "^20 || ^18 || ^16",
|
||||||
|
|
5843
pnpm-lock.yaml
5843
pnpm-lock.yaml
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,104 @@
|
||||||
|
/* eslint-env node */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This file runs in a Node context (it's NOT transpiled by Babel), so use only
|
||||||
|
* the ES6 features that are supported by your Node version. https://node.green/
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { configure } = require('quasar/wrappers');
|
||||||
|
const VueI18nPlugin = require('@intlify/unplugin-vue-i18n/vite');
|
||||||
|
const path = require('path');
|
||||||
|
|
||||||
|
module.exports = configure(function (/* ctx */) {
|
||||||
|
return {
|
||||||
|
eslint: {
|
||||||
|
warnings: true,
|
||||||
|
errors: true,
|
||||||
|
},
|
||||||
|
boot: ['i18n', 'axios', 'vnDate', 'validations', 'quasar', 'quasar.defaults'],
|
||||||
|
css: ['app.scss'],
|
||||||
|
extras: ['roboto-font', 'material-icons-outlined', 'material-symbols-outlined'],
|
||||||
|
build: {
|
||||||
|
target: {
|
||||||
|
browser: ['es2022', 'edge88', 'firefox78', 'chrome87', 'safari13.1'],
|
||||||
|
node: 'node18',
|
||||||
|
},
|
||||||
|
vueRouterMode: 'hash',
|
||||||
|
rawDefine: {
|
||||||
|
'process.env.NODE_ENV': JSON.stringify(process.env.NODE_ENV),
|
||||||
|
},
|
||||||
|
extendViteConf(viteConf) {
|
||||||
|
delete viteConf.build.polyfillModulePreload;
|
||||||
|
viteConf.build.modulePreload = {
|
||||||
|
polyfill: false,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
alias: {
|
||||||
|
composables: path.join(__dirname, './src/composables'),
|
||||||
|
filters: path.join(__dirname, './src/filters'),
|
||||||
|
},
|
||||||
|
vitePlugins: [
|
||||||
|
[
|
||||||
|
VueI18nPlugin({
|
||||||
|
runtimeOnly: false,
|
||||||
|
include: [
|
||||||
|
path.resolve(__dirname, './src/i18n/locale/**'),
|
||||||
|
path.resolve(__dirname, './src/pages/**/locale/**'),
|
||||||
|
],
|
||||||
|
}),
|
||||||
|
],
|
||||||
|
],
|
||||||
|
},
|
||||||
|
devServer: {
|
||||||
|
server: {
|
||||||
|
type: 'http',
|
||||||
|
},
|
||||||
|
proxy: {
|
||||||
|
'/api': {
|
||||||
|
target: 'http://0.0.0.0:3000',
|
||||||
|
logLevel: 'debug',
|
||||||
|
changeOrigin: true,
|
||||||
|
secure: false,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
open: false,
|
||||||
|
},
|
||||||
|
framework: {
|
||||||
|
config: {
|
||||||
|
dark: 'auto',
|
||||||
|
},
|
||||||
|
lang: 'en-GB',
|
||||||
|
plugins: ['Notify', 'Dialog'],
|
||||||
|
all: 'auto',
|
||||||
|
autoImportComponentCase: 'pascal',
|
||||||
|
},
|
||||||
|
animations: [],
|
||||||
|
ssr: {
|
||||||
|
pwa: false,
|
||||||
|
prodPort: 3000,
|
||||||
|
middlewares: ['render'],
|
||||||
|
},
|
||||||
|
pwa: {
|
||||||
|
workboxMode: 'generateSW',
|
||||||
|
injectPwaMetaTags: true,
|
||||||
|
swFilename: 'sw.js',
|
||||||
|
manifestFilename: 'manifest.json',
|
||||||
|
useCredentialsForManifestTag: false,
|
||||||
|
},
|
||||||
|
cordova: {},
|
||||||
|
capacitor: {
|
||||||
|
hideSplashscreen: true,
|
||||||
|
},
|
||||||
|
electron: {
|
||||||
|
inspectPort: 5858,
|
||||||
|
bundler: 'packager',
|
||||||
|
packager: {},
|
||||||
|
builder: {
|
||||||
|
appId: 'salix-frontend',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
bex: {
|
||||||
|
contentScripts: ['my-content-script'],
|
||||||
|
},
|
||||||
|
};
|
||||||
|
});
|
242
quasar.config.js
242
quasar.config.js
|
@ -1,242 +0,0 @@
|
||||||
/* eslint-env node */
|
|
||||||
|
|
||||||
/*
|
|
||||||
* This file runs in a Node context (it's NOT transpiled by Babel), so use only
|
|
||||||
* the ES6 features that are supported by your Node version. https://node.green/
|
|
||||||
*/
|
|
||||||
|
|
||||||
// Configuration for your app
|
|
||||||
// https://v2.quasar.dev/quasar-cli-vite/quasar-config-js
|
|
||||||
|
|
||||||
const { configure } = require('quasar/wrappers');
|
|
||||||
const VueI18nPlugin = require('@intlify/unplugin-vue-i18n/vite');
|
|
||||||
const path = require('path');
|
|
||||||
|
|
||||||
module.exports = configure(function (/* ctx */) {
|
|
||||||
return {
|
|
||||||
eslint: {
|
|
||||||
// fix: true,
|
|
||||||
// include = [],
|
|
||||||
// exclude = [],
|
|
||||||
// rawOptions = {},
|
|
||||||
warnings: true,
|
|
||||||
errors: true,
|
|
||||||
},
|
|
||||||
|
|
||||||
// https://v2.quasar.dev/quasar-cli/prefetch-feature
|
|
||||||
// preFetch: true,
|
|
||||||
|
|
||||||
// app boot file (/src/boot)
|
|
||||||
// --> boot files are part of "main.js"
|
|
||||||
// https://v2.quasar.dev/quasar-cli/boot-files
|
|
||||||
boot: ['i18n', 'axios', 'vnDate', 'validations', 'quasar', 'quasar.defaults'],
|
|
||||||
|
|
||||||
// https://v2.quasar.dev/quasar-cli-vite/quasar-config-js#css
|
|
||||||
css: ['app.scss'],
|
|
||||||
|
|
||||||
// https://github.com/quasarframework/quasar/tree/dev/extras
|
|
||||||
extras: [
|
|
||||||
// 'ionicons-v4',
|
|
||||||
// 'mdi-v5',
|
|
||||||
// 'fontawesome-v6',
|
|
||||||
// 'eva-icons',
|
|
||||||
// 'themify',
|
|
||||||
// 'line-awesome',
|
|
||||||
// 'roboto-font-latin-ext', // this or either 'roboto-font', NEVER both!
|
|
||||||
|
|
||||||
'roboto-font',
|
|
||||||
'material-icons-outlined',
|
|
||||||
'material-symbols-outlined',
|
|
||||||
],
|
|
||||||
|
|
||||||
// Full list of options: https://v2.quasar.dev/quasar-cli-vite/quasar-config-js#build
|
|
||||||
build: {
|
|
||||||
target: {
|
|
||||||
browser: ['es2022', 'edge88', 'firefox78', 'chrome87', 'safari13.1'],
|
|
||||||
node: 'node18',
|
|
||||||
},
|
|
||||||
|
|
||||||
vueRouterMode: 'hash', // available values: 'hash', 'history'
|
|
||||||
// vueRouterBase,
|
|
||||||
// vueDevtools,
|
|
||||||
// vueOptionsAPI: false,
|
|
||||||
|
|
||||||
// rebuildCache: true, // rebuilds Vite/linter/etc cache on startup
|
|
||||||
|
|
||||||
// publicPath: '/',
|
|
||||||
// analyze: true,
|
|
||||||
// env: {},
|
|
||||||
rawDefine: {
|
|
||||||
'process.env.NODE_ENV': JSON.stringify(process.env.NODE_ENV),
|
|
||||||
},
|
|
||||||
// ignorePublicFolder: true,
|
|
||||||
// minify: false,
|
|
||||||
// polyfillModulePreload: true,
|
|
||||||
// distDir
|
|
||||||
|
|
||||||
extendViteConf(viteConf) {
|
|
||||||
// FIXME: Delete deprecated property polyfillModulePreload
|
|
||||||
// that is set by Quasar by default
|
|
||||||
delete viteConf.build.polyfillModulePreload;
|
|
||||||
viteConf.build.modulePreload = {
|
|
||||||
polyfill: false,
|
|
||||||
};
|
|
||||||
},
|
|
||||||
// viteVuePluginOptions: {},
|
|
||||||
|
|
||||||
alias: {
|
|
||||||
composables: path.join(__dirname, './src/composables'),
|
|
||||||
filters: path.join(__dirname, './src/filters'),
|
|
||||||
},
|
|
||||||
|
|
||||||
vitePlugins: [
|
|
||||||
[
|
|
||||||
VueI18nPlugin({
|
|
||||||
runtimeOnly: false,
|
|
||||||
include: [
|
|
||||||
path.resolve(__dirname, './src/i18n/locale/**'),
|
|
||||||
path.resolve(__dirname, './src/pages/**/locale/**'),
|
|
||||||
],
|
|
||||||
}),
|
|
||||||
],
|
|
||||||
],
|
|
||||||
},
|
|
||||||
|
|
||||||
// Full list of options: https://v2.quasar.dev/quasar-cli-vite/quasar-config-js#devServer
|
|
||||||
devServer: {
|
|
||||||
server: {
|
|
||||||
type: 'http',
|
|
||||||
},
|
|
||||||
proxy: {
|
|
||||||
'/api': {
|
|
||||||
target: 'http://0.0.0.0:3000',
|
|
||||||
logLevel: 'debug',
|
|
||||||
changeOrigin: true,
|
|
||||||
secure: false,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
open: false,
|
|
||||||
},
|
|
||||||
|
|
||||||
// https://v2.quasar.dev/quasar-cli-vite/quasar-config-js#framework
|
|
||||||
framework: {
|
|
||||||
config: {
|
|
||||||
config: {
|
|
||||||
dark: 'auto',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
lang: 'en-GB',
|
|
||||||
|
|
||||||
// iconSet: 'material-icons', // Quasar icon set
|
|
||||||
// lang: 'en-US', // Quasar language pack
|
|
||||||
|
|
||||||
// For special cases outside of where the auto-import strategy can have an impact
|
|
||||||
// (like functional components as one of the examples),
|
|
||||||
// you can manually specify Quasar components/directives to be available everywhere:
|
|
||||||
//
|
|
||||||
// components: [],
|
|
||||||
// directives: [],
|
|
||||||
|
|
||||||
// Quasar plugins
|
|
||||||
plugins: ['Notify', 'Dialog'],
|
|
||||||
all: 'auto',
|
|
||||||
autoImportComponentCase: 'pascal',
|
|
||||||
},
|
|
||||||
|
|
||||||
// animations: 'all', // --- includes all animations
|
|
||||||
// https://v2.quasar.dev/options/animations
|
|
||||||
animations: [],
|
|
||||||
|
|
||||||
// https://v2.quasar.dev/quasar-cli-vite/quasar-config-js#property-sourcefiles
|
|
||||||
// sourceFiles: {
|
|
||||||
// rootComponent: 'src/App.vue',
|
|
||||||
// router: 'src/router/index',
|
|
||||||
// store: 'src/store/index',
|
|
||||||
// registerServiceWorker: 'src-pwa/register-service-worker',
|
|
||||||
// serviceWorker: 'src-pwa/custom-service-worker',
|
|
||||||
// pwaManifestFile: 'src-pwa/manifest.json',
|
|
||||||
// electronMain: 'src-electron/electron-main',
|
|
||||||
// electronPreload: 'src-electron/electron-preload'
|
|
||||||
// },
|
|
||||||
|
|
||||||
// https://v2.quasar.dev/quasar-cli/developing-ssr/configuring-ssr
|
|
||||||
ssr: {
|
|
||||||
// ssrPwaHtmlFilename: 'offline.html', // do NOT use index.html as name!
|
|
||||||
// will mess up SSR
|
|
||||||
|
|
||||||
// extendSSRWebserverConf (esbuildConf) {},
|
|
||||||
// extendPackageJson (json) {},
|
|
||||||
|
|
||||||
pwa: false,
|
|
||||||
|
|
||||||
// manualStoreHydration: true,
|
|
||||||
// manualPostHydrationTrigger: true,
|
|
||||||
|
|
||||||
prodPort: 3000, // The default port that the production server should use
|
|
||||||
// (gets superseded if process.env.PORT is specified at runtime)
|
|
||||||
|
|
||||||
middlewares: [
|
|
||||||
'render', // keep this as last one
|
|
||||||
],
|
|
||||||
},
|
|
||||||
|
|
||||||
// https://v2.quasar.dev/quasar-cli/developing-pwa/configuring-pwa
|
|
||||||
pwa: {
|
|
||||||
workboxMode: 'generateSW', // or 'injectManifest'
|
|
||||||
injectPwaMetaTags: true,
|
|
||||||
swFilename: 'sw.js',
|
|
||||||
manifestFilename: 'manifest.json',
|
|
||||||
useCredentialsForManifestTag: false,
|
|
||||||
// useFilenameHashes: true,
|
|
||||||
// extendGenerateSWOptions (cfg) {}
|
|
||||||
// extendInjectManifestOptions (cfg) {},
|
|
||||||
// extendManifestJson (json) {}
|
|
||||||
// extendPWACustomSWConf (esbuildConf) {}
|
|
||||||
},
|
|
||||||
|
|
||||||
// Full list of options: https://v2.quasar.dev/quasar-cli/developing-cordova-apps/configuring-cordova
|
|
||||||
cordova: {
|
|
||||||
// noIosLegacyBuildFlag: true, // uncomment only if you know what you are doing
|
|
||||||
},
|
|
||||||
|
|
||||||
// Full list of options: https://v2.quasar.dev/quasar-cli/developing-capacitor-apps/configuring-capacitor
|
|
||||||
capacitor: {
|
|
||||||
hideSplashscreen: true,
|
|
||||||
},
|
|
||||||
|
|
||||||
// Full list of options: https://v2.quasar.dev/quasar-cli/developing-electron-apps/configuring-electron
|
|
||||||
electron: {
|
|
||||||
// extendElectronMainConf (esbuildConf)
|
|
||||||
// extendElectronPreloadConf (esbuildConf)
|
|
||||||
|
|
||||||
inspectPort: 5858,
|
|
||||||
|
|
||||||
bundler: 'packager', // 'packager' or 'builder'
|
|
||||||
|
|
||||||
packager: {
|
|
||||||
// https://github.com/electron-userland/electron-packager/blob/master/docs/api.md#options
|
|
||||||
// OS X / Mac App Store
|
|
||||||
// appBundleId: '',
|
|
||||||
// appCategoryType: '',
|
|
||||||
// osxSign: '',
|
|
||||||
// protocol: 'myapp://path',
|
|
||||||
// Windows only
|
|
||||||
// win32metadata: { ... }
|
|
||||||
},
|
|
||||||
|
|
||||||
builder: {
|
|
||||||
// https://www.electron.build/configuration/configuration
|
|
||||||
|
|
||||||
appId: 'salix-frontend',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
// Full list of options: https://v2.quasar.dev/quasar-cli-vite/developing-browser-extensions/configuring-bex
|
|
||||||
bex: {
|
|
||||||
contentScripts: ['my-content-script'],
|
|
||||||
|
|
||||||
// extendBexScriptsConf (esbuildConf) {}
|
|
||||||
// extendBexManifestJson (json) {}
|
|
||||||
},
|
|
||||||
};
|
|
||||||
});
|
|
|
@ -1,5 +1,6 @@
|
||||||
<script setup>
|
<script setup>
|
||||||
import { computed, ref } from 'vue';
|
import { computed, ref } from 'vue';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
import isValidDate from 'filters/isValidDate';
|
import isValidDate from 'filters/isValidDate';
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
|
@ -24,6 +25,9 @@ const hover = ref(false);
|
||||||
|
|
||||||
const emit = defineEmits(['update:modelValue']);
|
const emit = defineEmits(['update:modelValue']);
|
||||||
|
|
||||||
|
const { t } = useI18n();
|
||||||
|
const requiredFieldRule = (val) => !!val || t('globals.fieldRequired');
|
||||||
|
|
||||||
const joinDateAndTime = (date, time) => {
|
const joinDateAndTime = (date, time) => {
|
||||||
if (!date) {
|
if (!date) {
|
||||||
return null;
|
return null;
|
||||||
|
@ -91,6 +95,8 @@ const styleAttrs = computed(() => {
|
||||||
readonly
|
readonly
|
||||||
:model-value="displayDate(value)"
|
:model-value="displayDate(value)"
|
||||||
v-bind="{ ...$attrs, ...styleAttrs }"
|
v-bind="{ ...$attrs, ...styleAttrs }"
|
||||||
|
:class="{ required: $attrs.required }"
|
||||||
|
:rules="$attrs.required ? [requiredFieldRule] : null"
|
||||||
@click="isPopupOpen = true"
|
@click="isPopupOpen = true"
|
||||||
>
|
>
|
||||||
<template #append>
|
<template #append>
|
||||||
|
|
|
@ -17,8 +17,9 @@ const props = defineProps({
|
||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
const { t } = useI18n();
|
|
||||||
const emit = defineEmits(['update:modelValue']);
|
const emit = defineEmits(['update:modelValue']);
|
||||||
|
const { t } = useI18n();
|
||||||
|
const requiredFieldRule = (val) => !!val || t('globals.fieldRequired');
|
||||||
|
|
||||||
const value = computed({
|
const value = computed({
|
||||||
get() {
|
get() {
|
||||||
|
@ -71,6 +72,8 @@ const styleAttrs = computed(() => {
|
||||||
readonly
|
readonly
|
||||||
:model-value="formatTime(value)"
|
:model-value="formatTime(value)"
|
||||||
v-bind="{ ...$attrs, ...styleAttrs }"
|
v-bind="{ ...$attrs, ...styleAttrs }"
|
||||||
|
:class="{ required: $attrs.required }"
|
||||||
|
:rules="$attrs.required ? [requiredFieldRule] : null"
|
||||||
@click="isPopupOpen = true"
|
@click="isPopupOpen = true"
|
||||||
>
|
>
|
||||||
<template #append>
|
<template #append>
|
||||||
|
|
|
@ -60,7 +60,7 @@ const emit = defineEmits(['refresh', 'clear', 'search', 'init', 'remove']);
|
||||||
const arrayData = useArrayData($props.dataKey, {
|
const arrayData = useArrayData($props.dataKey, {
|
||||||
exprBuilder: $props.exprBuilder,
|
exprBuilder: $props.exprBuilder,
|
||||||
searchUrl: $props.searchUrl,
|
searchUrl: $props.searchUrl,
|
||||||
navigate: {},
|
navigate: $props.redirect ? {} : null,
|
||||||
});
|
});
|
||||||
const route = useRoute();
|
const route = useRoute();
|
||||||
const store = arrayData.store;
|
const store = arrayData.store;
|
||||||
|
|
|
@ -32,7 +32,7 @@ const url = computed(
|
||||||
() =>
|
() =>
|
||||||
`/api/${$props.storage}/${$props.collection}/${$props.size}/${$props.id}/download?access_token=${token}&${timeStamp.value}`
|
`/api/${$props.storage}/${$props.collection}/${$props.size}/${$props.id}/download?access_token=${token}&${timeStamp.value}`
|
||||||
);
|
);
|
||||||
const reload = (emit = false) => {
|
const reload = () => {
|
||||||
timeStamp.value = `timestamp=${Date.now()}`;
|
timeStamp.value = `timestamp=${Date.now()}`;
|
||||||
};
|
};
|
||||||
defineExpose({
|
defineExpose({
|
||||||
|
|
|
@ -78,6 +78,7 @@ async function insert() {
|
||||||
ref="vnPaginateRef"
|
ref="vnPaginateRef"
|
||||||
class="show"
|
class="show"
|
||||||
v-bind="$attrs"
|
v-bind="$attrs"
|
||||||
|
search-url="notes"
|
||||||
>
|
>
|
||||||
<template #body="{ rows }">
|
<template #body="{ rows }">
|
||||||
<TransitionGroup name="list" tag="div" class="column items-center full-width">
|
<TransitionGroup name="list" tag="div" class="column items-center full-width">
|
||||||
|
|
|
@ -73,7 +73,6 @@ export function useArrayData(key = useRoute().meta.moduleName, userOptions) {
|
||||||
const filter = {
|
const filter = {
|
||||||
order: store.order,
|
order: store.order,
|
||||||
limit: store.limit,
|
limit: store.limit,
|
||||||
skip: store.skip,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
let exprFilter;
|
let exprFilter;
|
||||||
|
@ -88,7 +87,7 @@ export function useArrayData(key = useRoute().meta.moduleName, userOptions) {
|
||||||
}
|
}
|
||||||
|
|
||||||
Object.assign(filter, store.userFilter, exprFilter);
|
Object.assign(filter, store.userFilter, exprFilter);
|
||||||
Object.assign(store.filter, filter);
|
Object.assign(store.filter, { ...filter, skip: store.skip });
|
||||||
const params = {
|
const params = {
|
||||||
filter: JSON.stringify(store.filter),
|
filter: JSON.stringify(store.filter),
|
||||||
};
|
};
|
||||||
|
|
|
@ -446,6 +446,8 @@ ticket:
|
||||||
sale: Sale
|
sale: Sale
|
||||||
ticketAdvance: Advance tickets
|
ticketAdvance: Advance tickets
|
||||||
futureTickets: Future tickets
|
futureTickets: Future tickets
|
||||||
|
purchaseRequest: Purchase request
|
||||||
|
weeklyTickets: Weekly tickets
|
||||||
list:
|
list:
|
||||||
nickname: Nickname
|
nickname: Nickname
|
||||||
state: State
|
state: State
|
||||||
|
@ -847,6 +849,7 @@ worker:
|
||||||
calendar: Calendar
|
calendar: Calendar
|
||||||
timeControl: Time control
|
timeControl: Time control
|
||||||
locker: Locker
|
locker: Locker
|
||||||
|
balance: Balance
|
||||||
formation: Formation
|
formation: Formation
|
||||||
list:
|
list:
|
||||||
name: Name
|
name: Name
|
||||||
|
@ -928,6 +931,13 @@ worker:
|
||||||
remark: Bonficado
|
remark: Bonficado
|
||||||
hasDiploma: Diploma
|
hasDiploma: Diploma
|
||||||
imageNotFound: Image not found
|
imageNotFound: Image not found
|
||||||
|
balance:
|
||||||
|
tableVisibleColumns:
|
||||||
|
paymentDate: Date
|
||||||
|
incomeType: Type
|
||||||
|
debit: Debt
|
||||||
|
credit: Have
|
||||||
|
concept: Concept
|
||||||
wagon:
|
wagon:
|
||||||
pageTitles:
|
pageTitles:
|
||||||
wagons: Wagons
|
wagons: Wagons
|
||||||
|
|
|
@ -445,6 +445,8 @@ ticket:
|
||||||
sale: Lineas del pedido
|
sale: Lineas del pedido
|
||||||
ticketAdvance: Adelantar tickets
|
ticketAdvance: Adelantar tickets
|
||||||
futureTickets: Tickets a futuro
|
futureTickets: Tickets a futuro
|
||||||
|
purchaseRequest: Petición de compra
|
||||||
|
weeklyTickets: Tickets programados
|
||||||
list:
|
list:
|
||||||
nickname: Alias
|
nickname: Alias
|
||||||
state: Estado
|
state: Estado
|
||||||
|
@ -843,6 +845,7 @@ worker:
|
||||||
calendar: Calendario
|
calendar: Calendario
|
||||||
timeControl: Control de horario
|
timeControl: Control de horario
|
||||||
locker: Taquilla
|
locker: Taquilla
|
||||||
|
balance: Balance
|
||||||
formation: Formación
|
formation: Formación
|
||||||
list:
|
list:
|
||||||
name: Nombre
|
name: Nombre
|
||||||
|
@ -915,6 +918,13 @@ worker:
|
||||||
remark: Bonficado
|
remark: Bonficado
|
||||||
hasDiploma: Diploma
|
hasDiploma: Diploma
|
||||||
imageNotFound: No se ha encontrado la imagen
|
imageNotFound: No se ha encontrado la imagen
|
||||||
|
balance:
|
||||||
|
tableVisibleColumns:
|
||||||
|
paymentDate: Fecha
|
||||||
|
incomeType: Tipo
|
||||||
|
debit: Debe
|
||||||
|
credit: Haber
|
||||||
|
concept: Concepto
|
||||||
wagon:
|
wagon:
|
||||||
pageTitles:
|
pageTitles:
|
||||||
wagons: Vagones
|
wagons: Vagones
|
||||||
|
|
|
@ -3,7 +3,6 @@ import { useRoute } from 'vue-router';
|
||||||
import { useI18n } from 'vue-i18n';
|
import { useI18n } from 'vue-i18n';
|
||||||
import VnSelect from 'src/components/common/VnSelect.vue';
|
import VnSelect from 'src/components/common/VnSelect.vue';
|
||||||
import FormModel from 'components/FormModel.vue';
|
import FormModel from 'components/FormModel.vue';
|
||||||
import VnRow from 'components/ui/VnRow.vue';
|
|
||||||
import VnInput from 'src/components/common/VnInput.vue';
|
import VnInput from 'src/components/common/VnInput.vue';
|
||||||
import { ref, watch } from 'vue';
|
import { ref, watch } from 'vue';
|
||||||
|
|
||||||
|
|
|
@ -119,7 +119,7 @@ const departments = ref();
|
||||||
emit-value
|
emit-value
|
||||||
hide-selected
|
hide-selected
|
||||||
map-options
|
map-options
|
||||||
option-label="country"
|
option-label="name"
|
||||||
option-value="id"
|
option-value="id"
|
||||||
outlined
|
outlined
|
||||||
rounded
|
rounded
|
||||||
|
|
|
@ -219,6 +219,7 @@ const useLang = (values) => {
|
||||||
:expr-builder="exprBuilder"
|
:expr-builder="exprBuilder"
|
||||||
:custom-tags="['tagGroups']"
|
:custom-tags="['tagGroups']"
|
||||||
@remove="clearFilter"
|
@remove="clearFilter"
|
||||||
|
:redirect="false"
|
||||||
>
|
>
|
||||||
<template #tags="{ tag, formatFn }">
|
<template #tags="{ tag, formatFn }">
|
||||||
<strong v-if="tag.label === 'categoryFk'">
|
<strong v-if="tag.label === 'categoryFk'">
|
||||||
|
|
|
@ -1,5 +1,4 @@
|
||||||
<script setup>
|
<script setup>
|
||||||
import { useStateStore } from 'stores/useStateStore';
|
|
||||||
import { useI18n } from 'vue-i18n';
|
import { useI18n } from 'vue-i18n';
|
||||||
import { computed, onMounted, ref } from 'vue';
|
import { computed, onMounted, ref } from 'vue';
|
||||||
import { dashIfEmpty, toHour } from 'src/filters';
|
import { dashIfEmpty, toHour } from 'src/filters';
|
||||||
|
|
|
@ -128,8 +128,8 @@ function confirmRemove() {
|
||||||
.onOk(() => refreshKey.value++);
|
.onOk(() => refreshKey.value++);
|
||||||
}
|
}
|
||||||
|
|
||||||
function navigateToRoadmapSummary(event, row) {
|
function navigateToRoadmapSummary(_, { id }) {
|
||||||
router.push({ name: 'RoadmapSummary', params: { id: 1 } });
|
router.push({ name: 'RoadmapSummary', params: { id } });
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|
|
@ -104,10 +104,14 @@ const totalEntryPrice = (rows) => {
|
||||||
for (const row of rows) {
|
for (const row of rows) {
|
||||||
let total = 0;
|
let total = 0;
|
||||||
let quantity = 0;
|
let quantity = 0;
|
||||||
|
|
||||||
|
if (row.buys) {
|
||||||
for (const buy of row.buys) {
|
for (const buy of row.buys) {
|
||||||
total = total + buy.total;
|
total = total + buy.total;
|
||||||
quantity = quantity + buy.quantity;
|
quantity = quantity + buy.quantity;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
row.total = total;
|
row.total = total;
|
||||||
row.quantity = quantity;
|
row.quantity = quantity;
|
||||||
totalPrice = totalPrice + total;
|
totalPrice = totalPrice + total;
|
||||||
|
|
|
@ -50,7 +50,7 @@ const itemCategoriesOptions = ref([]);
|
||||||
@on-fetch="(data) => (itemCategoriesOptions = data)"
|
@on-fetch="(data) => (itemCategoriesOptions = data)"
|
||||||
auto-load
|
auto-load
|
||||||
/>
|
/>
|
||||||
<VnFilterPanel :data-key="props.dataKey" :search-button="true">
|
<VnFilterPanel :data-key="props.dataKey" :search-button="true" :redirect="false">
|
||||||
<template #tags="{ tag, formatFn }">
|
<template #tags="{ tag, formatFn }">
|
||||||
<div class="q-gutter-x-xs">
|
<div class="q-gutter-x-xs">
|
||||||
<strong>{{ t(`params.${tag.label}`) }}: </strong>
|
<strong>{{ t(`params.${tag.label}`) }}: </strong>
|
||||||
|
|
|
@ -0,0 +1,263 @@
|
||||||
|
<script setup>
|
||||||
|
import { ref, computed, onMounted, onUnmounted, watch } from 'vue';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
|
||||||
|
import ItemDescriptorProxy from 'src/pages/Item/Card/ItemDescriptorProxy.vue';
|
||||||
|
import FetchedTags from 'components/ui/FetchedTags.vue';
|
||||||
|
import RightMenu from 'src/components/common/RightMenu.vue';
|
||||||
|
import FetchData from 'components/FetchData.vue';
|
||||||
|
|
||||||
|
import { useStateStore } from 'stores/useStateStore';
|
||||||
|
import { toCurrency } from 'filters/index';
|
||||||
|
import { useRole } from 'src/composables/useRole';
|
||||||
|
|
||||||
|
const $props = defineProps({
|
||||||
|
formData: {
|
||||||
|
type: Object,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
haveNegatives: {
|
||||||
|
type: Boolean,
|
||||||
|
required: true,
|
||||||
|
default: false,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const emit = defineEmits(['updateForm', 'update:haveNegatives']);
|
||||||
|
|
||||||
|
const stateStore = useStateStore();
|
||||||
|
const { t } = useI18n();
|
||||||
|
const { hasAny } = useRole();
|
||||||
|
|
||||||
|
const _ticketData = ref($props.formData);
|
||||||
|
const ticketUpdateActions = ref(null);
|
||||||
|
const haveNegatives = computed({
|
||||||
|
get: () => $props.haveNegatives,
|
||||||
|
set: (val) => emit('update:haveNegatives', val),
|
||||||
|
});
|
||||||
|
const rows = computed(() => _ticketData.value?.sale?.items || []);
|
||||||
|
|
||||||
|
watch(
|
||||||
|
() => _ticketData.value,
|
||||||
|
(val) => emit('updateForm', val),
|
||||||
|
{ deep: true }
|
||||||
|
);
|
||||||
|
|
||||||
|
const columns = computed(() => [
|
||||||
|
{
|
||||||
|
required: true,
|
||||||
|
label: t('basicData.item'),
|
||||||
|
name: 'item',
|
||||||
|
align: 'left',
|
||||||
|
format: (val) => val.name,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
required: true,
|
||||||
|
label: t('basicData.description'),
|
||||||
|
name: 'description',
|
||||||
|
align: 'left',
|
||||||
|
hidden: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: t('basicData.movable'),
|
||||||
|
name: 'movable',
|
||||||
|
align: 'left',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
required: true,
|
||||||
|
label: t('basicData.quantity'),
|
||||||
|
name: 'quantity',
|
||||||
|
field: 'quantity',
|
||||||
|
align: 'left',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
required: true,
|
||||||
|
label: t('basicData.pricePPU'),
|
||||||
|
name: 'price',
|
||||||
|
field: 'price',
|
||||||
|
align: 'left',
|
||||||
|
format: (val) => toCurrency(val),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
required: true,
|
||||||
|
label: t('basicData.newPricePPU'),
|
||||||
|
name: 'newPrice',
|
||||||
|
field: (row) => row.component.newPrice,
|
||||||
|
align: 'left',
|
||||||
|
format: (val) => toCurrency(val),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
required: true,
|
||||||
|
label: t('basicData.difference'),
|
||||||
|
name: 'difference',
|
||||||
|
field: (row) => row.component.difference,
|
||||||
|
align: 'left',
|
||||||
|
format: (val) => toCurrency(val),
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
const loadDefaultTicketAction = () => {
|
||||||
|
const isSalesAssistant = hasAny(['salesAssistant']);
|
||||||
|
_ticketData.value.option = isSalesAssistant ? 'mana' : 'renewPrices';
|
||||||
|
};
|
||||||
|
|
||||||
|
const totalPrice = computed(() => {
|
||||||
|
return rows.value.reduce((acc, item) => acc + item.price * item.quantity, 0);
|
||||||
|
});
|
||||||
|
|
||||||
|
const totalNewPrice = computed(() => {
|
||||||
|
return rows.value.reduce(
|
||||||
|
(acc, item) => acc + item.component.newPrice * item.quantity,
|
||||||
|
0
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
const totalDifference = computed(() => {
|
||||||
|
return rows.value.reduce((acc, item) => acc + item.component?.difference || 0, 0);
|
||||||
|
});
|
||||||
|
const showMovablecolumn = computed(() => (haveDifferences.value > 0 ? ['movable'] : []));
|
||||||
|
const haveDifferences = computed(() => _ticketData.value.sale?.haveDifferences);
|
||||||
|
const ticketHaveNegatives = () => {
|
||||||
|
let _haveNegatives = false;
|
||||||
|
let haveNotNegatives = false;
|
||||||
|
_ticketData.value.withoutNegatives = false;
|
||||||
|
_ticketData.value?.sale?.items.forEach((item) => {
|
||||||
|
if (item.quantity > item.movable) _haveNegatives = true;
|
||||||
|
else haveNotNegatives = true;
|
||||||
|
});
|
||||||
|
|
||||||
|
haveNegatives.value = _haveNegatives && haveNotNegatives && haveDifferences.value;
|
||||||
|
if (haveNegatives.value) _ticketData.value.withoutNegatives = true;
|
||||||
|
};
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
stateStore.rightDrawer = true;
|
||||||
|
loadDefaultTicketAction();
|
||||||
|
ticketHaveNegatives();
|
||||||
|
});
|
||||||
|
|
||||||
|
onUnmounted(() => (stateStore.rightDrawer = false));
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<FetchData
|
||||||
|
url="TicketUpdateActions"
|
||||||
|
@on-fetch="(data) => (ticketUpdateActions = data)"
|
||||||
|
auto-load
|
||||||
|
/>
|
||||||
|
<RightMenu>
|
||||||
|
<template #right-panel>
|
||||||
|
<QCard
|
||||||
|
class="q-pa-md q-mb-md q-ma-md color-vn-text"
|
||||||
|
bordered
|
||||||
|
flat
|
||||||
|
style="border-color: black"
|
||||||
|
>
|
||||||
|
<QCardSection horizontal>
|
||||||
|
<span class="text-weight-bold text-subtitle1 text-center full-width">
|
||||||
|
{{ t('basicData.total') }}
|
||||||
|
</span>
|
||||||
|
</QCardSection>
|
||||||
|
<QCardSection class="column items-center" horizontal>
|
||||||
|
<span>
|
||||||
|
{{ t('basicData.price') }}:
|
||||||
|
{{ toCurrency(totalPrice) }}
|
||||||
|
</span>
|
||||||
|
</QCardSection>
|
||||||
|
<QCardSection class="column items-center" horizontal>
|
||||||
|
<span>
|
||||||
|
{{ t('basicData.newPrice') }}: {{ toCurrency(totalNewPrice) }}
|
||||||
|
</span>
|
||||||
|
</QCardSection>
|
||||||
|
<QCardSection class="column items-center" horizontal>
|
||||||
|
<span>
|
||||||
|
{{ t('basicData.difference') }}: {{ toCurrency(totalDifference) }}
|
||||||
|
</span>
|
||||||
|
</QCardSection>
|
||||||
|
</QCard>
|
||||||
|
<QCard
|
||||||
|
v-if="totalDifference"
|
||||||
|
class="q-pa-md q-mb-md q-ma-md color-vn-text"
|
||||||
|
bordered
|
||||||
|
flat
|
||||||
|
style="border-color: black"
|
||||||
|
>
|
||||||
|
<QCardSection horizontal>
|
||||||
|
<span class="text-weight-bold text-subtitle1 text-center full-width">
|
||||||
|
{{ t('basicData.chargeDifference') }}
|
||||||
|
</span>
|
||||||
|
</QCardSection>
|
||||||
|
<QCardSection
|
||||||
|
v-for="(action, index) in ticketUpdateActions"
|
||||||
|
:key="index"
|
||||||
|
horizontal
|
||||||
|
>
|
||||||
|
<QRadio
|
||||||
|
v-model="_ticketData.option"
|
||||||
|
:val="action.code"
|
||||||
|
:label="action.description"
|
||||||
|
dense
|
||||||
|
/>
|
||||||
|
</QCardSection>
|
||||||
|
</QCard>
|
||||||
|
<QCard
|
||||||
|
v-if="haveNegatives"
|
||||||
|
class="q-pa-md q-mb-md q-ma-md color-vn-text"
|
||||||
|
bordered
|
||||||
|
flat
|
||||||
|
style="border-color: black"
|
||||||
|
>
|
||||||
|
<QCardSection horizontal class="flex row items-center">
|
||||||
|
<QCheckbox
|
||||||
|
:label="t('basicData.withoutNegatives')"
|
||||||
|
v-model="_ticketData.withoutNegatives"
|
||||||
|
:toggle-indeterminate="false"
|
||||||
|
/>
|
||||||
|
<QIcon name="info" size="xs" class="q-ml-sm">
|
||||||
|
<QTooltip max-width="350px">
|
||||||
|
{{ t('basicData.withoutNegativesInfo') }}
|
||||||
|
</QTooltip>
|
||||||
|
</QIcon>
|
||||||
|
</QCardSection>
|
||||||
|
</QCard>
|
||||||
|
</template>
|
||||||
|
</RightMenu>
|
||||||
|
<QTable
|
||||||
|
:visible-columns="showMovablecolumn"
|
||||||
|
:rows="rows"
|
||||||
|
:columns="columns"
|
||||||
|
row-key="id"
|
||||||
|
:pagination="{ rowsPerPage: 0 }"
|
||||||
|
class="full-width q-mt-md"
|
||||||
|
:no-data-label="t('globals.noResults')"
|
||||||
|
flat
|
||||||
|
>
|
||||||
|
<template #body-cell-item="{ row }">
|
||||||
|
<QTd>
|
||||||
|
<QBtn flat color="primary">
|
||||||
|
{{ row.itemFk }}
|
||||||
|
<ItemDescriptorProxy :id="row.itemFk" />
|
||||||
|
</QBtn>
|
||||||
|
</QTd>
|
||||||
|
</template>
|
||||||
|
<template #body-cell-description="{ row }">
|
||||||
|
<QTd>
|
||||||
|
<div class="column">
|
||||||
|
<span>{{ row.item.name }}</span>
|
||||||
|
<span class="color-vn-label">{{ row.item.subName }}</span>
|
||||||
|
<FetchedTags :item="row.item" :max-length="6" />
|
||||||
|
</div>
|
||||||
|
</QTd>
|
||||||
|
</template>
|
||||||
|
<template #body-cell-movable="{ row }">
|
||||||
|
<QTd>
|
||||||
|
<QBadge
|
||||||
|
v-if="_ticketData?.sale?.haveDifferences"
|
||||||
|
:text-color="row.quantity > row.movable ? 'black' : 'white'"
|
||||||
|
:color="row.quantity > row.movable ? 'negative' : 'transparent'"
|
||||||
|
:label="row.movable"
|
||||||
|
/>
|
||||||
|
</QTd>
|
||||||
|
</template>
|
||||||
|
</QTable>
|
||||||
|
</template>
|
|
@ -0,0 +1,468 @@
|
||||||
|
<script setup>
|
||||||
|
import { ref, computed, onMounted, watch } from 'vue';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import { useRouter } from 'vue-router';
|
||||||
|
|
||||||
|
import FetchData from 'components/FetchData.vue';
|
||||||
|
import VnRow from 'components/ui/VnRow.vue';
|
||||||
|
import VnSelect from 'src/components/common/VnSelect.vue';
|
||||||
|
import VnInput from 'src/components/common/VnInput.vue';
|
||||||
|
import VnInputDate from 'src/components/common/VnInputDate.vue';
|
||||||
|
import VnInputTime from 'components/common/VnInputTime.vue';
|
||||||
|
|
||||||
|
import axios from 'axios';
|
||||||
|
import useNotify from 'src/composables/useNotify.js';
|
||||||
|
import { toTimeFormat } from 'filters/date.js';
|
||||||
|
|
||||||
|
const $props = defineProps({
|
||||||
|
formData: {
|
||||||
|
type: Object,
|
||||||
|
required: true,
|
||||||
|
default: () => ({}),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const emit = defineEmits(['updateForm']);
|
||||||
|
|
||||||
|
const { notify } = useNotify();
|
||||||
|
const router = useRouter();
|
||||||
|
const { t } = useI18n();
|
||||||
|
const agencyFetchRef = ref(null);
|
||||||
|
const zonesFetchRef = ref(null);
|
||||||
|
|
||||||
|
const clientsOptions = ref([]);
|
||||||
|
const warehousesOptions = ref([]);
|
||||||
|
const companiesOptions = ref([]);
|
||||||
|
const agenciesOptions = ref([]);
|
||||||
|
const zonesOptions = ref([]);
|
||||||
|
const addresses = ref([]);
|
||||||
|
const formData = ref($props.formData);
|
||||||
|
|
||||||
|
watch(
|
||||||
|
() => formData.value,
|
||||||
|
(val) => emit('updateForm', val),
|
||||||
|
{ deep: true }
|
||||||
|
);
|
||||||
|
|
||||||
|
const agencyByWarehouseFilter = computed(() => ({
|
||||||
|
fields: ['id', 'name'],
|
||||||
|
order: 'name ASC',
|
||||||
|
where: {
|
||||||
|
warehouseFk: warehouseId.value,
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
const zonesFilter = computed(() => ({
|
||||||
|
fields: ['id', 'name'],
|
||||||
|
order: 'name ASC',
|
||||||
|
where: formData.value?.agencyModeFk
|
||||||
|
? {
|
||||||
|
shipped: formData.value?.shipped,
|
||||||
|
addressFk: formData.value?.addressFk,
|
||||||
|
agencyModeFk: formData.value?.agencyModeFk,
|
||||||
|
warehouseFk: formData.value?.warehouseFk,
|
||||||
|
}
|
||||||
|
: {},
|
||||||
|
}));
|
||||||
|
|
||||||
|
const getLanded = async (params) => {
|
||||||
|
try {
|
||||||
|
const validParams =
|
||||||
|
shipped.value && addressId.value && agencyModeId.value && warehouseId.value;
|
||||||
|
if (!validParams) return;
|
||||||
|
|
||||||
|
formData.value.zoneFk = null;
|
||||||
|
zonesOptions.value = [];
|
||||||
|
const { data } = await axios.get(`Agencies/getLanded`, { params });
|
||||||
|
if (data) {
|
||||||
|
formData.value.zoneFk = data.zoneFk;
|
||||||
|
formData.value.landed = data.landed;
|
||||||
|
formData.value.shipped = params.shipped;
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error);
|
||||||
|
notify(t('basicData.noDeliveryZoneAvailable'), 'negative');
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const getShipped = async (params) => {
|
||||||
|
try {
|
||||||
|
const validParams =
|
||||||
|
landed.value && addressId.value && agencyModeId.value && warehouseId.value;
|
||||||
|
if (!validParams) return;
|
||||||
|
|
||||||
|
formData.value.zoneFk = null;
|
||||||
|
zonesOptions.value = [];
|
||||||
|
const { data } = await axios.get(`Agencies/getShipped`, { params });
|
||||||
|
if (data) {
|
||||||
|
formData.value.zoneFk = data.zoneFk;
|
||||||
|
formData.value.landed = params.landed;
|
||||||
|
formData.value.shipped = data.shipped;
|
||||||
|
} else {
|
||||||
|
notify(t('basicData.noDeliveryZoneAvailable'), 'negative');
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error);
|
||||||
|
notify(t('basicData.noDeliveryZoneAvailable'), 'negative');
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const onChangeZone = async (zoneId) => {
|
||||||
|
try {
|
||||||
|
formData.value.agencyModeFk = null;
|
||||||
|
const { data } = await axios.get(`Zones/${zoneId}`);
|
||||||
|
formData.value.agencyModeFk = data.agencyModeFk;
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const onChangeAddress = async (addressId) => {
|
||||||
|
try {
|
||||||
|
formData.value.nickname = null;
|
||||||
|
const { data } = await axios.get(`Addresses/${addressId}`);
|
||||||
|
formData.value.nickname = data.nickname;
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const getClientDefaultAddress = async (clientId) => {
|
||||||
|
try {
|
||||||
|
const { data } = await axios.get(`Clients/${clientId}`);
|
||||||
|
if (data) addressId.value = data.defaultAddressFk;
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const clientAddressesList = async (value) => {
|
||||||
|
let filter = {
|
||||||
|
include: [
|
||||||
|
{
|
||||||
|
relation: 'province',
|
||||||
|
scope: {
|
||||||
|
fields: ['name'],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
relation: 'agencyMode',
|
||||||
|
scope: {
|
||||||
|
fields: ['name'],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
|
const params = { filter: JSON.stringify(filter) };
|
||||||
|
const { data } = await axios.get(`Clients/${value}/addresses`, { params });
|
||||||
|
if (data) addresses.value = data;
|
||||||
|
};
|
||||||
|
|
||||||
|
const addressId = computed({
|
||||||
|
get: () => formData.value?.addressFk,
|
||||||
|
set: (val) => {
|
||||||
|
if (val != formData.value?.addressFk) {
|
||||||
|
formData.value.addressFk = val;
|
||||||
|
onChangeAddress(val);
|
||||||
|
getShipped({
|
||||||
|
landed: formData.value?.landed,
|
||||||
|
addressFk: val,
|
||||||
|
agencyModeFk: formData.value?.agencyModeFk,
|
||||||
|
warehouseFk: formData.value?.warehouseFk,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const clientId = computed({
|
||||||
|
get: () => formData.value?.clientFk,
|
||||||
|
set: (val) => {
|
||||||
|
formData.value.clientFk = val;
|
||||||
|
formData.value.addressFk = null;
|
||||||
|
if (!val) return;
|
||||||
|
getClientDefaultAddress(val);
|
||||||
|
clientAddressesList(val);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const landed = computed({
|
||||||
|
get: () => formData.value?.landed,
|
||||||
|
set: (val) => {
|
||||||
|
formData.value.landed = val;
|
||||||
|
getShipped({
|
||||||
|
landed: val,
|
||||||
|
addressFk: formData.value?.addressFk,
|
||||||
|
agencyModeFk: formData.value?.agencyModeFk,
|
||||||
|
warehouseFk: formData.value?.warehouseFk,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const agencyModeId = computed({
|
||||||
|
get: () => formData.value.agencyModeFk,
|
||||||
|
set: (val) => {
|
||||||
|
if (val != formData.value.agencyModeFk) {
|
||||||
|
formData.value.agencyModeFk = val;
|
||||||
|
if (!val) return;
|
||||||
|
const agencyMode = agenciesOptions.value.find((a) => a.id == val);
|
||||||
|
formData.value.warehouseFk = agencyMode.warehouseFk;
|
||||||
|
|
||||||
|
getLanded({
|
||||||
|
shipped: formData.value?.shipped,
|
||||||
|
addressFk: formData.value?.addressFk,
|
||||||
|
agencyModeFk: val,
|
||||||
|
warehouseFk: formData.value?.warehouseFk,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const zoneId = computed({
|
||||||
|
get: () => formData.value?.zoneFk,
|
||||||
|
set: (val) => {
|
||||||
|
if (val != formData.value?.zoneFk) {
|
||||||
|
formData.value.zoneFk = val;
|
||||||
|
onChangeZone(val);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const warehouseId = computed({
|
||||||
|
get: () => formData.value?.warehouseFk,
|
||||||
|
set: (val) => {
|
||||||
|
if (val != formData.value?.warehouseFk) {
|
||||||
|
formData.value.warehouseFk = val;
|
||||||
|
getShipped({
|
||||||
|
landed: formData.value?.landed,
|
||||||
|
addressFk: formData.value?.addressFk,
|
||||||
|
agencyModeFk: formData.value?.agencyModeFk,
|
||||||
|
warehouseFk: val,
|
||||||
|
}).then(() => {
|
||||||
|
if (zoneId.value == null) formData.value.agencyModeFk = null;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const shipped = computed({
|
||||||
|
get: () => formData.value?.shipped,
|
||||||
|
set: (val) => {
|
||||||
|
if (new Date(formData.value?.shipped).toDateString() != val.toDateString())
|
||||||
|
val.setHours(0, 0, 0, 0);
|
||||||
|
formData.value.shipped = val;
|
||||||
|
getLanded({
|
||||||
|
shipped: val,
|
||||||
|
addressFk: formData.value?.addressFk,
|
||||||
|
agencyModeFk: formData.value?.agencyModeFk,
|
||||||
|
warehouseFk: formData.value?.warehouseFk,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const onFormModelInit = () => {
|
||||||
|
if (formData.value?.clientFk) clientAddressesList(formData.value?.clientFk);
|
||||||
|
};
|
||||||
|
|
||||||
|
const redirectToCustomerAddress = () => {
|
||||||
|
router.push({
|
||||||
|
name: 'CustomerAddressEditCard',
|
||||||
|
params: { id: clientId.value, addressId: addressId.value },
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
onMounted(() => onFormModelInit());
|
||||||
|
</script>
|
||||||
|
<template>
|
||||||
|
<FetchData
|
||||||
|
url="Clients"
|
||||||
|
:filter="{
|
||||||
|
fields: ['id', 'name'],
|
||||||
|
order: 'id',
|
||||||
|
}"
|
||||||
|
@on-fetch="(data) => (clientsOptions = data)"
|
||||||
|
auto-load
|
||||||
|
/>
|
||||||
|
<FetchData
|
||||||
|
url="Warehouses"
|
||||||
|
@on-fetch="(data) => (warehousesOptions = data)"
|
||||||
|
auto-load
|
||||||
|
/>
|
||||||
|
<FetchData
|
||||||
|
url="Companies"
|
||||||
|
:filter="{
|
||||||
|
fields: ['id', 'code'],
|
||||||
|
order: 'code ASC',
|
||||||
|
}"
|
||||||
|
@on-fetch="(data) => (companiesOptions = data)"
|
||||||
|
auto-load
|
||||||
|
/>
|
||||||
|
<FetchData
|
||||||
|
ref="agencyFetchRef"
|
||||||
|
url="AgencyModes/byWarehouse"
|
||||||
|
:filter="agencyByWarehouseFilter"
|
||||||
|
@on-fetch="(data) => (agenciesOptions = data)"
|
||||||
|
auto-load
|
||||||
|
/>
|
||||||
|
<FetchData
|
||||||
|
ref="zonesFetchRef"
|
||||||
|
url="Zones/includingExpired"
|
||||||
|
:filter="zonesFilter"
|
||||||
|
@on-fetch="(data) => (zonesOptions = data)"
|
||||||
|
auto-load
|
||||||
|
/>
|
||||||
|
<QForm>
|
||||||
|
<VnRow class="row q-gutter-md q-mb-md">
|
||||||
|
<VnSelect
|
||||||
|
:label="t('basicData.client')"
|
||||||
|
v-model="clientId"
|
||||||
|
option-value="id"
|
||||||
|
option-label="name"
|
||||||
|
:options="clientsOptions"
|
||||||
|
hide-selected
|
||||||
|
map-options
|
||||||
|
:required="true"
|
||||||
|
>
|
||||||
|
<template #option="scope">
|
||||||
|
<QItem v-bind="scope.itemProps">
|
||||||
|
<QItemSection>
|
||||||
|
<QItemLabel
|
||||||
|
>#{{ scope.opt?.id }} {{ scope.opt?.name }}</QItemLabel
|
||||||
|
>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
</template>
|
||||||
|
</VnSelect>
|
||||||
|
<VnSelect
|
||||||
|
:label="t('basicData.warehouse')"
|
||||||
|
v-model="warehouseId"
|
||||||
|
option-value="id"
|
||||||
|
option-label="name"
|
||||||
|
:options="warehousesOptions"
|
||||||
|
hide-selected
|
||||||
|
map-options
|
||||||
|
:required="true"
|
||||||
|
/>
|
||||||
|
</VnRow>
|
||||||
|
<VnRow class="row q-gutter-md q-mb-md">
|
||||||
|
<VnSelect
|
||||||
|
:label="t('basicData.address')"
|
||||||
|
v-model="addressId"
|
||||||
|
option-value="id"
|
||||||
|
option-label="nickname"
|
||||||
|
:options="addresses"
|
||||||
|
hide-selected
|
||||||
|
map-options
|
||||||
|
:required="true"
|
||||||
|
>
|
||||||
|
<template #option="scope">
|
||||||
|
<QItem v-bind="scope.itemProps">
|
||||||
|
<QItemSection>
|
||||||
|
<QItemLabel
|
||||||
|
:class="{
|
||||||
|
'color-vn-label': !scope.opt?.isActive,
|
||||||
|
}"
|
||||||
|
>
|
||||||
|
{{
|
||||||
|
`${
|
||||||
|
!scope.opt?.isActive
|
||||||
|
? t('basicData.inactive')
|
||||||
|
: ''
|
||||||
|
} `
|
||||||
|
}}
|
||||||
|
<span> {{ scope.opt?.nickname }}</span>
|
||||||
|
<span
|
||||||
|
v-if="
|
||||||
|
scope.opt?.province ||
|
||||||
|
scope.opt?.city ||
|
||||||
|
scope.opt?.street
|
||||||
|
"
|
||||||
|
>, {{ scope.opt?.street }}, {{ scope.opt?.city }},
|
||||||
|
{{ scope.opt?.province?.name }} -
|
||||||
|
{{ scope.opt?.agencyMode?.name }}</span
|
||||||
|
>
|
||||||
|
</QItemLabel>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
</template>
|
||||||
|
<template #append>
|
||||||
|
<QIcon
|
||||||
|
name="edit"
|
||||||
|
color="primary"
|
||||||
|
size="sm"
|
||||||
|
class="fill-icon cursor-pointer"
|
||||||
|
@click.stop="redirectToCustomerAddress()"
|
||||||
|
>
|
||||||
|
<QTooltip>{{ t('basicData.editAddress') }}</QTooltip>
|
||||||
|
</QIcon>
|
||||||
|
</template>
|
||||||
|
</VnSelect>
|
||||||
|
<VnInput
|
||||||
|
:label="t('basicData.alias')"
|
||||||
|
v-model="formData.nickname"
|
||||||
|
:required="true"
|
||||||
|
/>
|
||||||
|
</VnRow>
|
||||||
|
<VnRow class="row q-gutter-md q-mb-md no-wrap">
|
||||||
|
<VnSelect
|
||||||
|
:label="t('basicData.company')"
|
||||||
|
v-model="formData.companyFk"
|
||||||
|
option-value="id"
|
||||||
|
option-label="code"
|
||||||
|
:options="companiesOptions"
|
||||||
|
hide-selected
|
||||||
|
map-options
|
||||||
|
:required="true"
|
||||||
|
/>
|
||||||
|
<VnSelect
|
||||||
|
:label="t('basicData.agency')"
|
||||||
|
v-model="agencyModeId"
|
||||||
|
option-value="id"
|
||||||
|
option-label="name"
|
||||||
|
:options="agenciesOptions"
|
||||||
|
hide-selected
|
||||||
|
map-options
|
||||||
|
@focus="agencyFetchRef.fetch()"
|
||||||
|
/>
|
||||||
|
<VnSelect
|
||||||
|
:label="t('basicData.zone')"
|
||||||
|
v-model="zoneId"
|
||||||
|
option-value="id"
|
||||||
|
option-label="name"
|
||||||
|
:options="zonesOptions"
|
||||||
|
hide-selected
|
||||||
|
map-options
|
||||||
|
:required="true"
|
||||||
|
@focus="zonesFetchRef.fetch()"
|
||||||
|
>
|
||||||
|
<template #option="scope">
|
||||||
|
<QItem v-bind="scope.itemProps">
|
||||||
|
<QItemSection>
|
||||||
|
<QItemLabel
|
||||||
|
>{{ scope.opt?.name }} - Max.
|
||||||
|
{{ toTimeFormat(scope.opt?.hour) }}
|
||||||
|
h.</QItemLabel
|
||||||
|
>
|
||||||
|
</QItemSection>
|
||||||
|
</QItem>
|
||||||
|
</template>
|
||||||
|
</VnSelect>
|
||||||
|
</VnRow>
|
||||||
|
<VnRow class="row q-gutter-md q-mb-md">
|
||||||
|
<VnInputDate
|
||||||
|
:label="t('basicData.shipped')"
|
||||||
|
v-model="formData.shipped"
|
||||||
|
:required="true"
|
||||||
|
/>
|
||||||
|
<VnInputTime
|
||||||
|
:label="t('basicData.shippedHour')"
|
||||||
|
v-model="formData.shipped"
|
||||||
|
:required="true"
|
||||||
|
/>
|
||||||
|
<VnInputDate
|
||||||
|
:label="t('basicData.landed')"
|
||||||
|
v-model="formData.landed"
|
||||||
|
:required="true"
|
||||||
|
/>
|
||||||
|
</VnRow>
|
||||||
|
</QForm>
|
||||||
|
</template>
|
|
@ -0,0 +1,195 @@
|
||||||
|
<script setup>
|
||||||
|
import { ref, onBeforeMount } from 'vue';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import { useRoute, useRouter } from 'vue-router';
|
||||||
|
|
||||||
|
import BasicDataTable from './BasicDataTable.vue';
|
||||||
|
import TicketBasicDataForm from './TicketBasicDataForm.vue';
|
||||||
|
import { useVnConfirm } from 'composables/useVnConfirm';
|
||||||
|
|
||||||
|
import axios from 'axios';
|
||||||
|
import useNotify from 'src/composables/useNotify.js';
|
||||||
|
|
||||||
|
const { notify } = useNotify();
|
||||||
|
const route = useRoute();
|
||||||
|
const router = useRouter();
|
||||||
|
const { t } = useI18n();
|
||||||
|
const stepperRef = ref(null);
|
||||||
|
const { openConfirmationModal } = useVnConfirm();
|
||||||
|
|
||||||
|
const step = ref(1);
|
||||||
|
const formData = ref({});
|
||||||
|
const initialDataLoaded = ref(false);
|
||||||
|
const haveNegatives = ref(false);
|
||||||
|
|
||||||
|
const ticketFilter = {
|
||||||
|
include: [
|
||||||
|
{ relation: 'address' },
|
||||||
|
{
|
||||||
|
relation: 'client',
|
||||||
|
scope: {
|
||||||
|
fields: [
|
||||||
|
'salesPersonFk',
|
||||||
|
'name',
|
||||||
|
'isActive',
|
||||||
|
'isFreezed',
|
||||||
|
'isTaxDataChecked',
|
||||||
|
'credit',
|
||||||
|
'email',
|
||||||
|
'phone',
|
||||||
|
'mobile',
|
||||||
|
'hasElectronicInvoice',
|
||||||
|
],
|
||||||
|
include: {
|
||||||
|
relation: 'salesPersonUser',
|
||||||
|
scope: { fields: ['id', 'name'] },
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{ relation: 'invoiceOut' },
|
||||||
|
],
|
||||||
|
};
|
||||||
|
|
||||||
|
const getTicketData = async () => {
|
||||||
|
const params = { filter: JSON.stringify(ticketFilter) };
|
||||||
|
const { data } = await axios.get(`tickets/${route.params.id}`, { params });
|
||||||
|
formData.value = data;
|
||||||
|
initialDataLoaded.value = true;
|
||||||
|
};
|
||||||
|
|
||||||
|
const isFormInvalid = () => {
|
||||||
|
return (
|
||||||
|
!formData.value.clientFk ||
|
||||||
|
!formData.value.addressFk ||
|
||||||
|
!formData.value.agencyModeFk ||
|
||||||
|
!formData.value.companyFk ||
|
||||||
|
!formData.value.shipped ||
|
||||||
|
!formData.value.landed ||
|
||||||
|
!formData.value.zoneFk
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const getPriceDifference = async () => {
|
||||||
|
try {
|
||||||
|
const params = {
|
||||||
|
landed: formData.value.landed,
|
||||||
|
addressId: formData.value.addressFk,
|
||||||
|
agencyModeId: formData.value.agencyModeFk,
|
||||||
|
zoneId: formData.value.zoneFk,
|
||||||
|
warehouseId: formData.value.warehouseFk,
|
||||||
|
shipped: formData.value.shipped,
|
||||||
|
};
|
||||||
|
const { data } = await axios.post(
|
||||||
|
`tickets/${formData.value.id}/priceDifference`,
|
||||||
|
params
|
||||||
|
);
|
||||||
|
formData.value.sale = data;
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const submit = async () => {
|
||||||
|
try {
|
||||||
|
if (!formData.value.option)
|
||||||
|
return notify(t('basicData.chooseAnOption'), 'negative');
|
||||||
|
|
||||||
|
const params = {
|
||||||
|
clientFk: formData.value.clientFk,
|
||||||
|
nickname: formData.value.nickname,
|
||||||
|
agencyModeFk: formData.value.agencyModeFk,
|
||||||
|
addressFk: formData.value.addressFk,
|
||||||
|
zoneFk: formData.value.zoneFk,
|
||||||
|
warehouseFk: formData.value.warehouseFk,
|
||||||
|
companyFk: formData.value.companyFk,
|
||||||
|
shipped: formData.value.shipped,
|
||||||
|
landed: formData.value.landed,
|
||||||
|
isDeleted: formData.value.isDeleted,
|
||||||
|
option: formData.value.option,
|
||||||
|
isWithoutNegatives: formData.value.withoutNegatives,
|
||||||
|
withWarningAccept: formData.value.withWarningAccept,
|
||||||
|
keepPrice: false,
|
||||||
|
};
|
||||||
|
|
||||||
|
const { data } = await axios.post(
|
||||||
|
`tickets/${formData.value.id}/componentUpdate`,
|
||||||
|
params
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!data) return;
|
||||||
|
|
||||||
|
const ticketToMove = data.id;
|
||||||
|
notify(t('basicData.unroutedTicket'), 'positive');
|
||||||
|
router.push({ name: 'TicketSummary', params: { id: ticketToMove } });
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const submitWithNegatives = async () => {
|
||||||
|
formData.value.withWarningAccept = true;
|
||||||
|
submit();
|
||||||
|
};
|
||||||
|
|
||||||
|
const onNextStep = async () => {
|
||||||
|
if (step.value === 1) {
|
||||||
|
if (isFormInvalid())
|
||||||
|
return notify(t('basicData.someFieldsAreInvalid'), 'negative');
|
||||||
|
|
||||||
|
await getPriceDifference();
|
||||||
|
stepperRef.value.next();
|
||||||
|
} else if (step.value === 2) {
|
||||||
|
if (haveNegatives.value && !formData.value.withoutNegatives)
|
||||||
|
openConfirmationModal(
|
||||||
|
t('basicData.negativesConfirmTitle'),
|
||||||
|
t('basicData.negativesConfirmMessage'),
|
||||||
|
submitWithNegatives
|
||||||
|
);
|
||||||
|
else submit();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
onBeforeMount(async () => await getTicketData());
|
||||||
|
</script>
|
||||||
|
<template>
|
||||||
|
<QStepper
|
||||||
|
v-model="step"
|
||||||
|
ref="stepperRef"
|
||||||
|
color="primary"
|
||||||
|
animated
|
||||||
|
keep-alive
|
||||||
|
style="max-width: 800px; margin: auto"
|
||||||
|
>
|
||||||
|
<QStep :name="1" :title="t('globals.pageTitles.basicData')" :done="step > 1">
|
||||||
|
<TicketBasicDataForm
|
||||||
|
v-if="initialDataLoaded"
|
||||||
|
@update-form="($event) => (formData = $event)"
|
||||||
|
:form-data="formData"
|
||||||
|
/>
|
||||||
|
</QStep>
|
||||||
|
<QStep :name="2" :title="t('basicData.priceDifference')">
|
||||||
|
<BasicDataTable
|
||||||
|
:form-data="formData"
|
||||||
|
v-model:haveNegatives="haveNegatives"
|
||||||
|
@update-form="($event) => (formData = $event)"
|
||||||
|
/>
|
||||||
|
</QStep>
|
||||||
|
<template #navigation>
|
||||||
|
<QStepperNavigation class="flex justify-between">
|
||||||
|
<QBtn
|
||||||
|
flat
|
||||||
|
color="primary"
|
||||||
|
@click="stepperRef.previous()"
|
||||||
|
:label="t('basicData.back')"
|
||||||
|
class="q-ml-sm"
|
||||||
|
:class="{ invisible: step === 1 }"
|
||||||
|
/>
|
||||||
|
<QBtn
|
||||||
|
@click="onNextStep()"
|
||||||
|
color="primary"
|
||||||
|
:label="step === 2 ? t('basicData.finalize') : t('basicData.next')"
|
||||||
|
/>
|
||||||
|
</QStepperNavigation>
|
||||||
|
</template>
|
||||||
|
</QStepper>
|
||||||
|
</template>
|
|
@ -1,3 +0,0 @@
|
||||||
<template>
|
|
||||||
<QCard>Basic Data</QCard>
|
|
||||||
</template>
|
|
|
@ -14,6 +14,7 @@ const routeName = computed(() => route.name);
|
||||||
const searchBarDataKeys = {
|
const searchBarDataKeys = {
|
||||||
TicketSummary: 'TicketSummary',
|
TicketSummary: 'TicketSummary',
|
||||||
TicketSale: 'TicketSale',
|
TicketSale: 'TicketSale',
|
||||||
|
TicketPurchaseRequest: 'TicketPurchaseRequest',
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
<template>
|
<template>
|
||||||
|
|
|
@ -0,0 +1,69 @@
|
||||||
|
<script setup>
|
||||||
|
import { ref } from 'vue';
|
||||||
|
import { useRoute } from 'vue-router';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
|
||||||
|
import FormModelPopup from 'components/FormModelPopup.vue';
|
||||||
|
import VnRow from 'components/ui/VnRow.vue';
|
||||||
|
import VnInput from 'src/components/common/VnInput.vue';
|
||||||
|
import VnSelect from 'src/components/common/VnSelect.vue';
|
||||||
|
import FetchData from 'components/FetchData.vue';
|
||||||
|
|
||||||
|
const emit = defineEmits(['onRequestCreated']);
|
||||||
|
|
||||||
|
const route = useRoute();
|
||||||
|
const { t } = useI18n();
|
||||||
|
|
||||||
|
const attendersOptions = ref([]);
|
||||||
|
</script>
|
||||||
|
<template>
|
||||||
|
<FetchData
|
||||||
|
url="TicketRequests/getItemTypeWorker"
|
||||||
|
:filter="{ fields: ['id', 'nickname'], order: 'nickname ASC' }"
|
||||||
|
auto-load
|
||||||
|
@on-fetch="(data) => (attendersOptions = data)"
|
||||||
|
/>
|
||||||
|
<FormModelPopup
|
||||||
|
:title="t('Create request')"
|
||||||
|
url-create="TicketRequests"
|
||||||
|
model="CreateTicketRequest"
|
||||||
|
:form-initial-data="{ ticketFk: route.params.id }"
|
||||||
|
@on-data-saved="() => emit('onRequestCreated')"
|
||||||
|
>
|
||||||
|
<template #form-inputs="{ data }">
|
||||||
|
<VnRow class="row q-gutter-md q-mb-md">
|
||||||
|
<VnInput
|
||||||
|
v-model="data.description"
|
||||||
|
:label="t('purchaseRequest.description')"
|
||||||
|
/>
|
||||||
|
<VnSelect
|
||||||
|
:label="t('purchaseRequest.atender')"
|
||||||
|
v-model="data.attenderFk"
|
||||||
|
:options="attendersOptions"
|
||||||
|
hide-selected
|
||||||
|
option-label="nickname"
|
||||||
|
option-value="id"
|
||||||
|
/>
|
||||||
|
</VnRow>
|
||||||
|
<VnRow class="row q-gutter-md q-mb-md">
|
||||||
|
<VnInput
|
||||||
|
v-model="data.quantity"
|
||||||
|
:label="t('purchaseRequest.quantity')"
|
||||||
|
type="number"
|
||||||
|
min="1"
|
||||||
|
/>
|
||||||
|
<VnInput
|
||||||
|
v-model="data.price"
|
||||||
|
:label="t('purchaseRequest.price')"
|
||||||
|
type="number"
|
||||||
|
min="0"
|
||||||
|
/>
|
||||||
|
</VnRow>
|
||||||
|
</template>
|
||||||
|
</FormModelPopup>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<i18n>
|
||||||
|
es:
|
||||||
|
Create request: Crear petición de compra
|
||||||
|
</i18n>
|
|
@ -1,7 +1,10 @@
|
||||||
<script setup>
|
<script setup>
|
||||||
import VnLog from 'src/components/common/VnLog.vue';
|
import VnLog from 'src/components/common/VnLog.vue';
|
||||||
|
import { useRoute } from 'vue-router';
|
||||||
|
|
||||||
|
const route = useRoute();
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<VnLog model="Ticket" url="/TicketLogs"></VnLog>
|
<VnLog model="Ticket" url="/TicketLogs" :key="route.params.id"></VnLog>
|
||||||
</template>
|
</template>
|
||||||
|
|
|
@ -0,0 +1,267 @@
|
||||||
|
<script setup>
|
||||||
|
import { ref, computed, watch, reactive } from 'vue';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import { useRoute } from 'vue-router';
|
||||||
|
|
||||||
|
import VnInput from 'src/components/common/VnInput.vue';
|
||||||
|
import WorkerDescriptorProxy from 'src/pages/Worker/Card/WorkerDescriptorProxy.vue';
|
||||||
|
import ItemDescriptorProxy from 'src/pages/Item/Card/ItemDescriptorProxy.vue';
|
||||||
|
import CrudModel from 'src/components/CrudModel.vue';
|
||||||
|
import TicketCreateRequest from './TicketCreateRequest.vue';
|
||||||
|
|
||||||
|
import { dashIfEmpty } from 'src/filters';
|
||||||
|
import { toDateFormat } from 'src/filters/date.js';
|
||||||
|
|
||||||
|
const route = useRoute();
|
||||||
|
const { t } = useI18n();
|
||||||
|
const createTicketRequestDialogRef = ref(null);
|
||||||
|
const crudModelRef = ref(null);
|
||||||
|
|
||||||
|
watch(
|
||||||
|
() => route.params.id,
|
||||||
|
async (val) => {
|
||||||
|
crudModelFilter.where.ticketFk = val;
|
||||||
|
crudModelRef.value.reload();
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
const crudModelFilter = reactive({
|
||||||
|
include: [
|
||||||
|
{
|
||||||
|
relation: 'atender',
|
||||||
|
scope: {
|
||||||
|
include: {
|
||||||
|
relation: 'user',
|
||||||
|
scope: {
|
||||||
|
fields: ['nickname'],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
relation: 'requester',
|
||||||
|
scope: {
|
||||||
|
include: {
|
||||||
|
relation: 'user',
|
||||||
|
scope: {
|
||||||
|
fields: ['nickname'],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
relation: 'sale',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
fields: [
|
||||||
|
'id',
|
||||||
|
'description',
|
||||||
|
'created',
|
||||||
|
'requesterFk',
|
||||||
|
'attenderFk',
|
||||||
|
'quantity',
|
||||||
|
'price',
|
||||||
|
'saleFk',
|
||||||
|
'isOk',
|
||||||
|
],
|
||||||
|
order: ['created ASC'],
|
||||||
|
where: {
|
||||||
|
ticketFk: route.params.id,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const columns = computed(() => [
|
||||||
|
{
|
||||||
|
label: t('purchaseRequest.id'),
|
||||||
|
name: 'id',
|
||||||
|
field: 'id',
|
||||||
|
align: 'left',
|
||||||
|
columnFilter: null,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: t('purchaseRequest.description'),
|
||||||
|
name: 'description',
|
||||||
|
field: 'description',
|
||||||
|
align: 'left',
|
||||||
|
format: (val) => dashIfEmpty(val),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: t('purchaseRequest.created'),
|
||||||
|
name: 'created',
|
||||||
|
field: 'created',
|
||||||
|
align: 'left',
|
||||||
|
format: (val) => toDateFormat(val),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: t('purchaseRequest.requester'),
|
||||||
|
name: 'requester',
|
||||||
|
align: 'left',
|
||||||
|
sortable: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: t('purchaseRequest.atender'),
|
||||||
|
name: 'atender',
|
||||||
|
align: 'left',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: t('purchaseRequest.quantity'),
|
||||||
|
name: 'quantity',
|
||||||
|
align: 'left',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: t('purchaseRequest.price'),
|
||||||
|
name: 'price',
|
||||||
|
align: 'left',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: t('purchaseRequest.saleFk'),
|
||||||
|
name: 'saleFk',
|
||||||
|
align: 'left',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: t('purchaseRequest.state'),
|
||||||
|
name: 'state',
|
||||||
|
field: 'isOk',
|
||||||
|
align: 'left',
|
||||||
|
format: (val) => t(getRequestState(val)),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: '',
|
||||||
|
name: 'actions',
|
||||||
|
align: 'left',
|
||||||
|
columnFilter: null,
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
const getRequestState = (state) => {
|
||||||
|
switch (state) {
|
||||||
|
case null:
|
||||||
|
return 'New';
|
||||||
|
case false:
|
||||||
|
return 'Denied';
|
||||||
|
case true:
|
||||||
|
return 'Acepted';
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const isEditable = (isOk) => isOk !== null;
|
||||||
|
|
||||||
|
const removeLine = async (row) => crudModelRef.value.remove([row]);
|
||||||
|
|
||||||
|
const openCreateModal = () => createTicketRequestDialogRef.value.show();
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<QPage class="column items-center q-pa-md">
|
||||||
|
<CrudModel
|
||||||
|
data-key="PurchaseRequests"
|
||||||
|
url="TicketRequests"
|
||||||
|
ref="crudModelRef"
|
||||||
|
:filter="crudModelFilter"
|
||||||
|
:order="['created ASC']"
|
||||||
|
:default-remove="false"
|
||||||
|
:default-save="false"
|
||||||
|
:default-reset="false"
|
||||||
|
:limit="0"
|
||||||
|
auto-load
|
||||||
|
>
|
||||||
|
<template #body="{ rows }">
|
||||||
|
<QTable
|
||||||
|
:rows="rows"
|
||||||
|
:columns="columns"
|
||||||
|
row-key="id"
|
||||||
|
:pagination="{ rowsPerPage: 0 }"
|
||||||
|
class="full-width q-mt-md"
|
||||||
|
:no-data-label="t('globals.noResults')"
|
||||||
|
@row-click="(_, row) => redirectToTicketSummary(row.ticketFk)"
|
||||||
|
>
|
||||||
|
<template #body-cell-description="{ row }">
|
||||||
|
<QTd @click.stop>
|
||||||
|
<VnInput
|
||||||
|
v-model="row.description"
|
||||||
|
@blur="crudModelRef.saveChanges()"
|
||||||
|
:disable="isEditable(row.isOk)"
|
||||||
|
/>
|
||||||
|
</QTd>
|
||||||
|
</template>
|
||||||
|
<template #body-cell-requester="{ row }">
|
||||||
|
<QTd @click.stop>
|
||||||
|
<QBtn flat color="primary">
|
||||||
|
{{ row.requester?.user?.nickname }}
|
||||||
|
<WorkerDescriptorProxy :id="row.requesterFk" />
|
||||||
|
</QBtn>
|
||||||
|
</QTd>
|
||||||
|
</template>
|
||||||
|
<template #body-cell-atender="{ row }">
|
||||||
|
<QTd @click.stop>
|
||||||
|
<QBtn flat color="primary">
|
||||||
|
{{ row.atender?.user?.nickname }}
|
||||||
|
<WorkerDescriptorProxy :id="row.attenderFk" />
|
||||||
|
</QBtn>
|
||||||
|
</QTd>
|
||||||
|
</template>
|
||||||
|
<template #body-cell-quantity="{ row }">
|
||||||
|
<QTd @click.stop>
|
||||||
|
<VnInput
|
||||||
|
v-model="row.quantity"
|
||||||
|
@blur="crudModelRef.saveChanges()"
|
||||||
|
:disable="isEditable(row.isOk)"
|
||||||
|
/>
|
||||||
|
</QTd>
|
||||||
|
</template>
|
||||||
|
<template #body-cell-price="{ row }">
|
||||||
|
<QTd @click.stop>
|
||||||
|
<VnInput
|
||||||
|
v-model="row.price"
|
||||||
|
@blur="crudModelRef.saveChanges()"
|
||||||
|
:disable="isEditable(row.isOk)"
|
||||||
|
/>
|
||||||
|
</QTd>
|
||||||
|
</template>
|
||||||
|
<template #body-cell-saleFk="{ row }">
|
||||||
|
<QTd @click.stop>
|
||||||
|
<QBtn v-if="row.saleFk" flat color="primary">
|
||||||
|
{{ row.sale.itemFk }}
|
||||||
|
<ItemDescriptorProxy :id="row.sale.itemFk" />
|
||||||
|
</QBtn>
|
||||||
|
</QTd>
|
||||||
|
</template>
|
||||||
|
<template #body-cell-actions="{ row }">
|
||||||
|
<QTd>
|
||||||
|
<QIcon
|
||||||
|
@click.stop="removeLine(row)"
|
||||||
|
class="q-ml-sm cursor-pointer"
|
||||||
|
color="primary"
|
||||||
|
name="delete"
|
||||||
|
size="sm"
|
||||||
|
>
|
||||||
|
<QTooltip>
|
||||||
|
{{ t('globals.delete') }}
|
||||||
|
</QTooltip>
|
||||||
|
</QIcon>
|
||||||
|
</QTd>
|
||||||
|
</template>
|
||||||
|
</QTable>
|
||||||
|
</template>
|
||||||
|
</CrudModel>
|
||||||
|
<QDialog
|
||||||
|
ref="createTicketRequestDialogRef"
|
||||||
|
transition-show="scale"
|
||||||
|
transition-hide="scale"
|
||||||
|
>
|
||||||
|
<TicketCreateRequest @on-request-created="crudModelRef.reload()" />
|
||||||
|
</QDialog>
|
||||||
|
<QPageSticky :offset="[20, 20]">
|
||||||
|
<QBtn @click="openCreateModal()" color="primary" fab icon="add" />
|
||||||
|
<QTooltip class="text-no-wrap">
|
||||||
|
{{ t('purchaseRequest.newRequest') }}
|
||||||
|
</QTooltip>
|
||||||
|
</QPageSticky>
|
||||||
|
</QPage>
|
||||||
|
</template>
|
||||||
|
<i18n>
|
||||||
|
es:
|
||||||
|
New: Nueva
|
||||||
|
Denied: Denegada
|
||||||
|
Accepted: Aceptada
|
||||||
|
</i18n>
|
|
@ -353,15 +353,14 @@ const newOrderFromTicket = async () => {
|
||||||
};
|
};
|
||||||
|
|
||||||
const goToLog = (saleId) => {
|
const goToLog = (saleId) => {
|
||||||
//TODO: Redireccionar cuando exista la vista TicketLog
|
router.push({
|
||||||
// router.push({
|
name: 'TicketLog',
|
||||||
// name: 'TicketLog',
|
params: {
|
||||||
// params: {
|
originId: route.params.id,
|
||||||
// originId: route.params.id,
|
changedModel: 'Sale',
|
||||||
// changedModel: 'Sale',
|
changedModelId: saleId,
|
||||||
// changedModelId: saleId,
|
},
|
||||||
// },
|
});
|
||||||
// });
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const changeTicketState = async (val) => {
|
const changeTicketState = async (val) => {
|
||||||
|
@ -725,8 +724,9 @@ onUnmounted(() => (stateStore.rightDrawer = false));
|
||||||
</QTd>
|
</QTd>
|
||||||
</template>
|
</template>
|
||||||
<template #body-cell-history="{ row }">
|
<template #body-cell-history="{ row }">
|
||||||
<QTd v-if="row.hasLogs">
|
<QTd>
|
||||||
<QBtn
|
<QBtn
|
||||||
|
v-if="row.$hasLogs"
|
||||||
@click.stop="goToLog(row.id)"
|
@click.stop="goToLog(row.id)"
|
||||||
color="primary"
|
color="primary"
|
||||||
icon="history"
|
icon="history"
|
||||||
|
|
|
@ -0,0 +1,326 @@
|
||||||
|
<script setup>
|
||||||
|
import { onMounted, ref, computed, reactive, onUnmounted } from 'vue';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import { useRouter } from 'vue-router';
|
||||||
|
|
||||||
|
import FetchData from 'components/FetchData.vue';
|
||||||
|
import TableVisibleColumns from 'src/components/common/TableVisibleColumns.vue';
|
||||||
|
import VnInput from 'src/components/common/VnInput.vue';
|
||||||
|
import VnSelect from 'src/components/common/VnSelect.vue';
|
||||||
|
import CustomerDescriptorProxy from 'src/pages/Customer/Card/CustomerDescriptorProxy.vue';
|
||||||
|
import TicketDescriptorProxy from 'src/pages/Ticket/Card/TicketDescriptorProxy.vue';
|
||||||
|
import WorkerDescriptorProxy from 'src/pages/Worker/Card/WorkerDescriptorProxy.vue';
|
||||||
|
import VnPaginate from 'components/ui/VnPaginate.vue';
|
||||||
|
import VnSubToolbar from 'src/components/ui/VnSubToolbar.vue';
|
||||||
|
import VnSearchbar from 'src/components/ui/VnSearchbar.vue';
|
||||||
|
|
||||||
|
import { useStateStore } from 'stores/useStateStore';
|
||||||
|
import { dashIfEmpty } from 'src/filters';
|
||||||
|
import { useVnConfirm } from 'composables/useVnConfirm';
|
||||||
|
import { useArrayData } from 'composables/useArrayData';
|
||||||
|
import useNotify from 'src/composables/useNotify.js';
|
||||||
|
import axios from 'axios';
|
||||||
|
|
||||||
|
const router = useRouter();
|
||||||
|
const stateStore = useStateStore();
|
||||||
|
const { t } = useI18n();
|
||||||
|
const { openConfirmationModal } = useVnConfirm();
|
||||||
|
const { notify } = useNotify();
|
||||||
|
|
||||||
|
const paginateRef = ref(null);
|
||||||
|
const agencyModesOptions = ref([]);
|
||||||
|
const visibleColumns = ref([]);
|
||||||
|
const allColumnNames = ref([]);
|
||||||
|
|
||||||
|
const arrayData = useArrayData('WeeklyTickets');
|
||||||
|
const { store } = arrayData;
|
||||||
|
|
||||||
|
const weekdays = [
|
||||||
|
{ id: 0, name: t('weekdays.mon') },
|
||||||
|
{ id: 1, name: t('weekdays.tue') },
|
||||||
|
{ id: 2, name: t('weekdays.wed') },
|
||||||
|
{ id: 3, name: t('weekdays.thu') },
|
||||||
|
{ id: 4, name: t('weekdays.fri') },
|
||||||
|
{ id: 5, name: t('weekdays.sat') },
|
||||||
|
{ id: 6, name: t('weekdays.sun') },
|
||||||
|
];
|
||||||
|
|
||||||
|
const exprBuilder = (param, value) => {
|
||||||
|
switch (param) {
|
||||||
|
case 'clientName':
|
||||||
|
return { 'c.name': value };
|
||||||
|
case 'nickName':
|
||||||
|
return { 'u.name': value };
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const params = reactive({});
|
||||||
|
|
||||||
|
const applyColumnFilter = async (col) => {
|
||||||
|
try {
|
||||||
|
const paramKey = col.columnFilter?.filterParamKey || col.field;
|
||||||
|
params[paramKey] = col.columnFilter.filterValue;
|
||||||
|
await paginateRef.value.addFilter(null, params);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Error applying column filter', err);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const getInputEvents = (col) => ({ 'keyup.enter': () => applyColumnFilter(col) });
|
||||||
|
|
||||||
|
const columns = computed(() => [
|
||||||
|
{
|
||||||
|
label: t('weeklyTickets.id'),
|
||||||
|
name: 'id',
|
||||||
|
field: 'ticketFk',
|
||||||
|
align: 'left',
|
||||||
|
sortable: true,
|
||||||
|
columnFilter: null,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: t('weeklyTickets.client'),
|
||||||
|
name: 'client',
|
||||||
|
field: 'clientName',
|
||||||
|
align: 'left',
|
||||||
|
sortable: true,
|
||||||
|
columnFilter: {
|
||||||
|
component: VnInput,
|
||||||
|
type: 'text',
|
||||||
|
filterValue: null,
|
||||||
|
event: getInputEvents,
|
||||||
|
attrs: {
|
||||||
|
dense: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
format: (val) => dashIfEmpty(val),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: t('weeklyTickets.shipment'),
|
||||||
|
name: 'shipment',
|
||||||
|
field: 'weekDay',
|
||||||
|
align: 'left',
|
||||||
|
sortable: true,
|
||||||
|
columnFilter: null,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: t('weeklyTickets.agency'),
|
||||||
|
name: 'agency',
|
||||||
|
field: 'agencyModeFk',
|
||||||
|
align: 'left',
|
||||||
|
sortable: true,
|
||||||
|
columnFilter: null,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: t('weeklyTickets.warehouse'),
|
||||||
|
name: 'warehouse',
|
||||||
|
field: 'warehouseName',
|
||||||
|
align: 'left',
|
||||||
|
sortable: true,
|
||||||
|
columnFilter: null,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: t('weeklyTickets.salesperson'),
|
||||||
|
field: 'salesperson',
|
||||||
|
name: 'salesperson',
|
||||||
|
align: 'left',
|
||||||
|
sortable: true,
|
||||||
|
columnFilter: {
|
||||||
|
component: VnInput,
|
||||||
|
type: 'text',
|
||||||
|
filterValue: null,
|
||||||
|
event: getInputEvents,
|
||||||
|
filterParamKey: 'nickName',
|
||||||
|
attrs: {
|
||||||
|
dense: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: '',
|
||||||
|
name: 'actions',
|
||||||
|
align: 'left',
|
||||||
|
columnFilter: null,
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
const redirectToTicketSummary = (ticketFk) =>
|
||||||
|
router.push({ name: 'TicketSummary', params: { id: ticketFk } });
|
||||||
|
|
||||||
|
const deleteWeekly = async (ticketFk) => {
|
||||||
|
try {
|
||||||
|
await axios.delete(`TicketWeeklies/${ticketFk}`);
|
||||||
|
notify(t('globals.dataSaved'), 'positive');
|
||||||
|
const ticketIndex = store.data.findIndex((e) => e.ticketFk == ticketFk);
|
||||||
|
store.data.splice(ticketIndex, 1);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Error deleting weekly', err);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const onUpdate = async (ticketFk, field, value) => {
|
||||||
|
try {
|
||||||
|
const params = { ticketFk, [field]: value };
|
||||||
|
await axios.patch('TicketWeeklies', params);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Error updating weekly', err);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
onMounted(async () => {
|
||||||
|
stateStore.rightDrawer = true;
|
||||||
|
const filteredColumns = columns.value.filter((col) => col.name !== 'actions');
|
||||||
|
allColumnNames.value = filteredColumns.map((col) => col.name);
|
||||||
|
});
|
||||||
|
|
||||||
|
onUnmounted(() => (stateStore.rightDrawer = false));
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<FetchData
|
||||||
|
url="AgencyModes/isActive"
|
||||||
|
:filter="{ fields: ['id', 'name'], order: 'name' }"
|
||||||
|
auto-load
|
||||||
|
@on-fetch="(data) => (agencyModesOptions = data)"
|
||||||
|
/>
|
||||||
|
<VnSearchbar
|
||||||
|
data-key="WeeklyTickets"
|
||||||
|
:label="t('weeklyTickets.search')"
|
||||||
|
:info="t('weeklyTickets.searchInfo')"
|
||||||
|
/>
|
||||||
|
<VnSubToolbar>
|
||||||
|
<template #st-data>
|
||||||
|
<TableVisibleColumns
|
||||||
|
:all-columns="allColumnNames"
|
||||||
|
table-code="itemsIndex"
|
||||||
|
labels-traductions-path="weeklyTickets"
|
||||||
|
@on-config-saved="visibleColumns = [...$event, 'actions']"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
</VnSubToolbar>
|
||||||
|
<QPage class="column items-center q-pa-md">
|
||||||
|
<VnPaginate
|
||||||
|
ref="paginateRef"
|
||||||
|
data-key="WeeklyTickets"
|
||||||
|
url="TicketWeeklies/filter"
|
||||||
|
:order="['weekDay', 'ticketFk']"
|
||||||
|
:limit="20"
|
||||||
|
:expr-builder="exprBuilder"
|
||||||
|
:user-params="params"
|
||||||
|
:offset="50"
|
||||||
|
auto-load
|
||||||
|
>
|
||||||
|
<template #body="{ rows }">
|
||||||
|
<QTable
|
||||||
|
:rows="rows"
|
||||||
|
:columns="columns"
|
||||||
|
row-key="id"
|
||||||
|
:pagination="{ rowsPerPage: 0 }"
|
||||||
|
class="full-width q-mt-md"
|
||||||
|
:visible-columns="visibleColumns"
|
||||||
|
:no-data-label="t('globals.noResults')"
|
||||||
|
@row-click="(_, row) => redirectToTicketSummary(row.ticketFk)"
|
||||||
|
>
|
||||||
|
<template #top-row="{ cols }">
|
||||||
|
<QTr>
|
||||||
|
<QTd
|
||||||
|
v-for="(col, index) in cols"
|
||||||
|
:key="index"
|
||||||
|
style="max-width: 100px"
|
||||||
|
>
|
||||||
|
<component
|
||||||
|
:is="col.columnFilter.component"
|
||||||
|
v-if="col.columnFilter"
|
||||||
|
v-model="col.columnFilter.filterValue"
|
||||||
|
v-bind="col.columnFilter.attrs"
|
||||||
|
v-on="col.columnFilter.event(col)"
|
||||||
|
dense
|
||||||
|
/>
|
||||||
|
</QTd>
|
||||||
|
</QTr>
|
||||||
|
</template>
|
||||||
|
<template #body-cell-id="{ row }">
|
||||||
|
<QTd @click.stop>
|
||||||
|
<QBtn flat color="primary">
|
||||||
|
{{ row.ticketFk }}
|
||||||
|
<TicketDescriptorProxy :id="row.ticketFk" />
|
||||||
|
</QBtn>
|
||||||
|
</QTd>
|
||||||
|
</template>
|
||||||
|
<template #body-cell-salesperson="{ row }">
|
||||||
|
<QTd @click.stop>
|
||||||
|
<QBtn flat color="primary">
|
||||||
|
{{ row.userName }}
|
||||||
|
<WorkerDescriptorProxy :id="row.workerFk" />
|
||||||
|
</QBtn>
|
||||||
|
</QTd>
|
||||||
|
</template>
|
||||||
|
<template #body-cell-client="{ row }">
|
||||||
|
<QTd @click.stop>
|
||||||
|
<QBtn flat color="primary" dense>
|
||||||
|
{{ row.clientName }}
|
||||||
|
<CustomerDescriptorProxy :id="row.clientFk" />
|
||||||
|
</QBtn>
|
||||||
|
</QTd>
|
||||||
|
</template>
|
||||||
|
<template #body-cell-shipment="{ row }">
|
||||||
|
<QTd @click.stop>
|
||||||
|
<VnSelect
|
||||||
|
:options="weekdays"
|
||||||
|
hide-selected
|
||||||
|
option-label="name"
|
||||||
|
option-value="id"
|
||||||
|
v-model="row.weekDay"
|
||||||
|
@update:model-value="
|
||||||
|
onUpdate(row.ticketFk, 'weekDay', $event)
|
||||||
|
"
|
||||||
|
/>
|
||||||
|
</QTd>
|
||||||
|
</template>
|
||||||
|
<template #body-cell-agency="{ row }">
|
||||||
|
<QTd @click.stop>
|
||||||
|
<VnSelect
|
||||||
|
:options="agencyModesOptions"
|
||||||
|
hide-selected
|
||||||
|
option-label="name"
|
||||||
|
option-value="id"
|
||||||
|
v-model="row.agencyModeFk"
|
||||||
|
@update:model-value="
|
||||||
|
onUpdate(row.ticketFk, 'agencyModeFk', $event)
|
||||||
|
"
|
||||||
|
/>
|
||||||
|
</QTd>
|
||||||
|
</template>
|
||||||
|
<template #body-cell-actions="{ row }">
|
||||||
|
<QTd>
|
||||||
|
<QIcon
|
||||||
|
@click.stop="
|
||||||
|
openConfirmationModal(
|
||||||
|
t('You are going to delete this weekly ticket'),
|
||||||
|
t(
|
||||||
|
'This ticket will be removed from weekly tickets! Continue anyway?'
|
||||||
|
),
|
||||||
|
() => deleteWeekly(row.ticketFk)
|
||||||
|
)
|
||||||
|
"
|
||||||
|
class="q-ml-sm cursor-pointer"
|
||||||
|
color="primary"
|
||||||
|
name="delete"
|
||||||
|
size="sm"
|
||||||
|
>
|
||||||
|
<QTooltip>
|
||||||
|
{{ t('globals.delete') }}
|
||||||
|
</QTooltip>
|
||||||
|
</QIcon>
|
||||||
|
</QTd>
|
||||||
|
</template>
|
||||||
|
</QTable>
|
||||||
|
</template>
|
||||||
|
</VnPaginate>
|
||||||
|
</QPage>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<i18n>
|
||||||
|
es:
|
||||||
|
You are going to delete this weekly ticket: Vas a eliminar este ticket programado
|
||||||
|
This ticket will be removed from weekly tickets! Continue anyway?: Este ticket se eliminará de tickets programados! ¿Continuar de todas formas?
|
||||||
|
</i18n>
|
|
@ -22,9 +22,6 @@ ticketSale:
|
||||||
shipped: Shipped
|
shipped: Shipped
|
||||||
agency: Agency
|
agency: Agency
|
||||||
address: Address
|
address: Address
|
||||||
card:
|
|
||||||
search: Search tickets
|
|
||||||
searchInfo: You can search by ticket id or alias
|
|
||||||
advanceTickets:
|
advanceTickets:
|
||||||
origin: Origin
|
origin: Origin
|
||||||
destination: Destination
|
destination: Destination
|
||||||
|
@ -80,3 +77,62 @@ futureTickets:
|
||||||
moveTicketSuccess: Tickets moved successfully!
|
moveTicketSuccess: Tickets moved successfully!
|
||||||
searchInfo: Search future tickets by date
|
searchInfo: Search future tickets by date
|
||||||
futureTicket: Future tickets
|
futureTicket: Future tickets
|
||||||
|
basicData:
|
||||||
|
next: Next
|
||||||
|
back: Back
|
||||||
|
finalize: Finalize
|
||||||
|
client: Client
|
||||||
|
warehouse: Warehouse
|
||||||
|
address: Address
|
||||||
|
inactive: (Inactive)
|
||||||
|
noDeliveryZoneAvailable: No delivery zone available for this landing date
|
||||||
|
editAddress: Edit address
|
||||||
|
alias: Alias
|
||||||
|
company: Company
|
||||||
|
agency: Agency
|
||||||
|
zone: Zone
|
||||||
|
shipped: Shipped
|
||||||
|
landed: Landed
|
||||||
|
shippedHour: Shipped hour
|
||||||
|
priceDifference: Price difference
|
||||||
|
someFieldsAreInvalid: Some fields are invalid
|
||||||
|
item: Item
|
||||||
|
description: Description
|
||||||
|
movable: Movable
|
||||||
|
quantity: Quantity
|
||||||
|
pricePPU: Price (PPU)
|
||||||
|
newPricePPU: New (PPU)
|
||||||
|
difference: Difference
|
||||||
|
total: Total
|
||||||
|
price: Price
|
||||||
|
newPrice: New price
|
||||||
|
chargeDifference: Charge difference to
|
||||||
|
withoutNegatives: Create without negatives
|
||||||
|
withoutNegativesInfo: Clone this ticket with the changes and only sales availables
|
||||||
|
negativesConfirmTitle: Edit basic data
|
||||||
|
negativesConfirmMessage: Negatives are going to be generated, are you sure you want to advance all the lines?
|
||||||
|
chooseAnOption: Choose an option
|
||||||
|
unroutedTicket: The ticket has been unrouted
|
||||||
|
card:
|
||||||
|
search: Search tickets
|
||||||
|
searchInfo: You can search by ticket id or alias
|
||||||
|
purchaseRequest:
|
||||||
|
id: Id
|
||||||
|
description: Description
|
||||||
|
created: Created
|
||||||
|
requester: Requester
|
||||||
|
atender: Atender
|
||||||
|
quantity: Quantity
|
||||||
|
price: Price
|
||||||
|
saleFk: Item id
|
||||||
|
state: State
|
||||||
|
newRequest: New request
|
||||||
|
weeklyTickets:
|
||||||
|
id: Ticket ID
|
||||||
|
client: Client
|
||||||
|
shipment: Shipment
|
||||||
|
agency: Agency
|
||||||
|
warehouse: Warehouse
|
||||||
|
salesperson: Salesperson
|
||||||
|
search: Search weekly tickets
|
||||||
|
searchInfo: Search weekly tickets by id or client id
|
||||||
|
|
|
@ -1,3 +1,62 @@
|
||||||
|
card:
|
||||||
|
search: Buscar tickets
|
||||||
|
searchInfo: Buscar tickets por identificador o alias
|
||||||
|
purchaseRequest:
|
||||||
|
Id: Id
|
||||||
|
description: Descripción
|
||||||
|
created: Fecha creación
|
||||||
|
requester: Solicitante
|
||||||
|
atender: Comprador
|
||||||
|
quantity: Cantidad
|
||||||
|
price: Precio
|
||||||
|
saleFk: Id artículo
|
||||||
|
state: Estado
|
||||||
|
newRequest: Crear petición
|
||||||
|
basicData:
|
||||||
|
next: Siguiente
|
||||||
|
back: Anterior
|
||||||
|
finalize: Finalizar
|
||||||
|
client: Cliente
|
||||||
|
warehouse: Almacén
|
||||||
|
address: Consignatario
|
||||||
|
inactive: (Inactivo)
|
||||||
|
noDeliveryZoneAvailable: No hay una zona de reparto disponible para la fecha de envío seleccionada
|
||||||
|
editAddress: Editar dirección
|
||||||
|
alias: Alias
|
||||||
|
company: Empresa
|
||||||
|
agency: Agencia
|
||||||
|
zone: Zona
|
||||||
|
shipped: F. Envío
|
||||||
|
landed: F. Entrega
|
||||||
|
shippedHour: Hora de envío
|
||||||
|
priceDifference: Diferencia de precio
|
||||||
|
someFieldsAreInvalid: Algunos campos no son válidos
|
||||||
|
item: Artículo
|
||||||
|
description: Descripción
|
||||||
|
movable: Movible
|
||||||
|
quantity: Cantidad
|
||||||
|
pricePPU: Precio (Ud.)
|
||||||
|
newPricePPU: Nuevo (Ud.)
|
||||||
|
difference: Diferencia
|
||||||
|
total: Total
|
||||||
|
price: Precio
|
||||||
|
newPrice: Nuevo precio
|
||||||
|
chargeDifference: Cargar diferencia a
|
||||||
|
withoutNegatives: Crear sin negativos
|
||||||
|
withoutNegativesInfo: Clonar este ticket con los cambios y solo ventas disponibles
|
||||||
|
negativesConfirmTitle: Editar datos básicos
|
||||||
|
negativesConfirmMessage: Se van a generar negativos, ¿seguro que quieres adelantar todas las líneas?
|
||||||
|
chooseAnOption: Elige una opción
|
||||||
|
unroutedTicket: El ticket ha sido desenrutado
|
||||||
|
weeklyTickets:
|
||||||
|
id: ID Ticket
|
||||||
|
client: Cliente
|
||||||
|
shipment: Salida
|
||||||
|
agency: Agencia
|
||||||
|
warehouse: Almacén
|
||||||
|
salesperson: Comercial
|
||||||
|
search: Buscar por tickets programados
|
||||||
|
searchInfo: Buscar tickets programados por el identificador o el identificador del cliente
|
||||||
advanceTickets:
|
advanceTickets:
|
||||||
origin: Origen
|
origin: Origen
|
||||||
destination: Destinatario
|
destination: Destinatario
|
||||||
|
@ -79,6 +138,3 @@ ticketSale:
|
||||||
shipped: F. Envío
|
shipped: F. Envío
|
||||||
agency: Agencia
|
agency: Agencia
|
||||||
address: Consignatario
|
address: Consignatario
|
||||||
card:
|
|
||||||
search: Buscar tickets
|
|
||||||
searchInfo: Buscar tickets por identificador o alias
|
|
||||||
|
|
|
@ -0,0 +1,91 @@
|
||||||
|
<script setup>
|
||||||
|
import { ref, computed } from 'vue';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import { useRoute } from 'vue-router';
|
||||||
|
import VnTable from 'components/VnTable/VnTable.vue';
|
||||||
|
const tableRef = ref();
|
||||||
|
const { t } = useI18n();
|
||||||
|
const route = useRoute();
|
||||||
|
const entityId = computed(() => route.params.id);
|
||||||
|
|
||||||
|
const columns = computed(() => [
|
||||||
|
{
|
||||||
|
align: 'left',
|
||||||
|
name: 'paymentDate',
|
||||||
|
label: t('worker.balance.tableVisibleColumns.paymentDate'),
|
||||||
|
create: true,
|
||||||
|
component: 'date',
|
||||||
|
field: 'paymentDate',
|
||||||
|
cardVisible: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
align: 'left',
|
||||||
|
name: 'incomeTypeFk',
|
||||||
|
label: t('worker.balance.tableVisibleColumns.incomeType'),
|
||||||
|
create: true,
|
||||||
|
component: 'select',
|
||||||
|
attrs: {
|
||||||
|
url: 'payrollComponents',
|
||||||
|
fields: ['id', 'name'],
|
||||||
|
},
|
||||||
|
cardVisible: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
align: 'left',
|
||||||
|
name: 'debit',
|
||||||
|
label: t('worker.balance.tableVisibleColumns.debit'),
|
||||||
|
create: true,
|
||||||
|
component: 'input',
|
||||||
|
field: 'debit',
|
||||||
|
cardVisible: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
align: 'left',
|
||||||
|
name: 'credit',
|
||||||
|
label: t('worker.balance.tableVisibleColumns.credit'),
|
||||||
|
create: true,
|
||||||
|
component: 'input',
|
||||||
|
field: 'credit',
|
||||||
|
cardVisible: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
align: 'left',
|
||||||
|
name: 'concept',
|
||||||
|
label: t('worker.balance.tableVisibleColumns.concept'),
|
||||||
|
create: true,
|
||||||
|
component: 'input',
|
||||||
|
field: 'concept',
|
||||||
|
cardVisible: true,
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<VnTable
|
||||||
|
ref="tableRef"
|
||||||
|
data-key="WorkerBalance"
|
||||||
|
:url="`Workers/${entityId}/incomes`"
|
||||||
|
:url-create="`Workers/${entityId}/incomes`"
|
||||||
|
save-url="WorkerIncomes/crud"
|
||||||
|
:create="{
|
||||||
|
urlCreate: 'workerIncomes',
|
||||||
|
title: t('Create workerBalance'),
|
||||||
|
onDataSaved: () => tableRef.reload(),
|
||||||
|
formInitialData: {
|
||||||
|
workerFk: entityId,
|
||||||
|
},
|
||||||
|
}"
|
||||||
|
order="paymentDate DESC"
|
||||||
|
:columns="columns"
|
||||||
|
default-mode="table"
|
||||||
|
auto-load
|
||||||
|
:right-search="false"
|
||||||
|
:is-editable="true"
|
||||||
|
:use-model="true"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<i18n>
|
||||||
|
es:
|
||||||
|
Create workerBalance: Crear balance
|
||||||
|
</i18n>
|
|
@ -35,6 +35,7 @@ const columns = computed(() => [
|
||||||
url: 'TrainingCourseTypes',
|
url: 'TrainingCourseTypes',
|
||||||
fields: ['id', 'name'],
|
fields: ['id', 'name'],
|
||||||
},
|
},
|
||||||
|
cardVisible: true,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
align: 'left',
|
align: 'left',
|
||||||
|
@ -52,6 +53,7 @@ const columns = computed(() => [
|
||||||
component: 'date',
|
component: 'date',
|
||||||
field: 'ended',
|
field: 'ended',
|
||||||
create: true,
|
create: true,
|
||||||
|
cardVisible: true,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
align: 'left',
|
align: 'left',
|
||||||
|
@ -63,6 +65,7 @@ const columns = computed(() => [
|
||||||
url: 'TrainingCenters',
|
url: 'TrainingCenters',
|
||||||
fields: ['id', 'name'],
|
fields: ['id', 'name'],
|
||||||
},
|
},
|
||||||
|
cardVisible: true,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
align: 'left',
|
align: 'left',
|
||||||
|
@ -78,6 +81,7 @@ const columns = computed(() => [
|
||||||
component: 'input',
|
component: 'input',
|
||||||
field: 'amount',
|
field: 'amount',
|
||||||
create: true,
|
create: true,
|
||||||
|
cardVisible: true,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
align: 'left',
|
align: 'left',
|
||||||
|
|
|
@ -274,7 +274,7 @@ export default {
|
||||||
name: 'CustomerBalance',
|
name: 'CustomerBalance',
|
||||||
meta: {
|
meta: {
|
||||||
title: 'balance',
|
title: 'balance',
|
||||||
icon: 'vn:invoice',
|
icon: 'balance',
|
||||||
},
|
},
|
||||||
component: () =>
|
component: () =>
|
||||||
import('src/pages/Customer/Card/CustomerBalance.vue'),
|
import('src/pages/Customer/Card/CustomerBalance.vue'),
|
||||||
|
|
|
@ -11,8 +11,15 @@ export default {
|
||||||
component: RouterView,
|
component: RouterView,
|
||||||
redirect: { name: 'TicketMain' },
|
redirect: { name: 'TicketMain' },
|
||||||
menus: {
|
menus: {
|
||||||
main: ['TicketList', 'TicketAdvance', 'TicketFuture'],
|
main: ['TicketList', 'TicketAdvance', 'TicketWeekly', 'TicketFuture'],
|
||||||
card: ['TicketBoxing', 'TicketSms', 'TicketSale', 'TicketLog'],
|
card: [
|
||||||
|
'TicketBasicData',
|
||||||
|
'TicketBoxing',
|
||||||
|
'TicketSms',
|
||||||
|
'TicketSale',
|
||||||
|
'TicketLog',
|
||||||
|
'TicketPurchaseRequest',
|
||||||
|
],
|
||||||
},
|
},
|
||||||
children: [
|
children: [
|
||||||
{
|
{
|
||||||
|
@ -41,13 +48,13 @@ export default {
|
||||||
component: () => import('src/pages/Ticket/TicketCreate.vue'),
|
component: () => import('src/pages/Ticket/TicketCreate.vue'),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'TicketAdvance',
|
name: 'TicketWeekly',
|
||||||
path: 'advance',
|
path: 'weekly',
|
||||||
meta: {
|
meta: {
|
||||||
title: 'ticketAdvance',
|
title: 'weeklyTickets',
|
||||||
icon: 'keyboard_double_arrow_left',
|
icon: 'access_time',
|
||||||
},
|
},
|
||||||
component: () => import('src/pages/Ticket/TicketAdvance.vue'),
|
component: () => import('src/pages/Ticket/TicketWeekly.vue'),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'TicketFuture',
|
name: 'TicketFuture',
|
||||||
|
@ -58,6 +65,15 @@ export default {
|
||||||
},
|
},
|
||||||
component: () => import('src/pages/Ticket/TicketFuture.vue'),
|
component: () => import('src/pages/Ticket/TicketFuture.vue'),
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
name: 'TicketAdvance',
|
||||||
|
path: 'advance',
|
||||||
|
meta: {
|
||||||
|
title: 'ticketAdvance',
|
||||||
|
icon: 'keyboard_double_arrow_left',
|
||||||
|
},
|
||||||
|
component: () => import('src/pages/Ticket/TicketAdvance.vue'),
|
||||||
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -82,7 +98,8 @@ export default {
|
||||||
title: 'basicData',
|
title: 'basicData',
|
||||||
icon: 'vn:settings',
|
icon: 'vn:settings',
|
||||||
},
|
},
|
||||||
component: () => import('src/pages/Ticket/Card/TicketBasicData.vue'),
|
component: () =>
|
||||||
|
import('src/pages/Ticket/Card/BasicData/TicketBasicDataView.vue'),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'TicketSale',
|
name: 'TicketSale',
|
||||||
|
@ -93,6 +110,25 @@ export default {
|
||||||
},
|
},
|
||||||
component: () => import('src/pages/Ticket/Card/TicketSale.vue'),
|
component: () => import('src/pages/Ticket/Card/TicketSale.vue'),
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
path: 'request',
|
||||||
|
name: 'TicketPurchaseRequest',
|
||||||
|
meta: {
|
||||||
|
title: 'purchaseRequest',
|
||||||
|
icon: 'vn:buyrequest',
|
||||||
|
},
|
||||||
|
component: () =>
|
||||||
|
import('src/pages/Ticket/Card/TicketPurchaseRequest.vue'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: 'log',
|
||||||
|
name: 'TicketLog',
|
||||||
|
meta: {
|
||||||
|
title: 'log',
|
||||||
|
icon: 'history',
|
||||||
|
},
|
||||||
|
component: () => import('src/pages/Ticket/Card/TicketLog.vue'),
|
||||||
|
},
|
||||||
{
|
{
|
||||||
path: 'boxing',
|
path: 'boxing',
|
||||||
name: 'TicketBoxing',
|
name: 'TicketBoxing',
|
||||||
|
@ -111,15 +147,6 @@ export default {
|
||||||
},
|
},
|
||||||
component: () => import('src/pages/Ticket/Card/TicketSms.vue'),
|
component: () => import('src/pages/Ticket/Card/TicketSms.vue'),
|
||||||
},
|
},
|
||||||
{
|
|
||||||
path: 'log',
|
|
||||||
name: 'TicketLog',
|
|
||||||
meta: {
|
|
||||||
title: 'log',
|
|
||||||
icon: 'history',
|
|
||||||
},
|
|
||||||
component: () => import('src/pages/Ticket/Card/TicketLog.vue'),
|
|
||||||
},
|
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
|
|
|
@ -23,6 +23,7 @@ export default {
|
||||||
'WorkerDms',
|
'WorkerDms',
|
||||||
'WorkerTimeControl',
|
'WorkerTimeControl',
|
||||||
'WorkerLocker',
|
'WorkerLocker',
|
||||||
|
'WorkerBalance',
|
||||||
'WorkerFormation',
|
'WorkerFormation',
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
@ -177,6 +178,15 @@ export default {
|
||||||
},
|
},
|
||||||
component: () => import('src/pages/Worker/Card/WorkerLocker.vue'),
|
component: () => import('src/pages/Worker/Card/WorkerLocker.vue'),
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
name: 'WorkerBalance',
|
||||||
|
path: 'balance',
|
||||||
|
meta: {
|
||||||
|
title: 'balance',
|
||||||
|
icon: 'balance',
|
||||||
|
},
|
||||||
|
component: () => import('src/pages/Worker/Card/WorkerBalance.vue'),
|
||||||
|
},
|
||||||
{
|
{
|
||||||
name: 'WorkerFormation',
|
name: 'WorkerFormation',
|
||||||
path: 'formation',
|
path: 'formation',
|
||||||
|
|
|
@ -6,15 +6,14 @@ describe('InvoiceInCorrective', () => {
|
||||||
const saveDialog = '.q-card > .q-card__actions > .q-btn--standard ';
|
const saveDialog = '.q-card > .q-card__actions > .q-btn--standard ';
|
||||||
|
|
||||||
it('should create a correcting invoice', () => {
|
it('should create a correcting invoice', () => {
|
||||||
|
cy.viewport(1280, 720);
|
||||||
cy.login('developer');
|
cy.login('developer');
|
||||||
cy.visit(`/#/invoice-in/1/summary?limit=10`);
|
cy.visit(`/#/invoice-in/1/summary?limit=10`);
|
||||||
|
|
||||||
cy.openLeftMenu();
|
|
||||||
cy.openActionsDescriptor();
|
cy.openActionsDescriptor();
|
||||||
|
|
||||||
cy.get(createRectificative).click();
|
cy.get(createRectificative).click();
|
||||||
cy.get(saveDialog).click();
|
cy.get(saveDialog).click();
|
||||||
cy.openLeftMenu();
|
|
||||||
cy.get(rectificativeSection).click();
|
cy.get(rectificativeSection).click();
|
||||||
cy.get('tbody > tr:visible').should('have.length', 1);
|
cy.get('tbody > tr:visible').should('have.length', 1);
|
||||||
});
|
});
|
||||||
|
|
|
@ -5,18 +5,15 @@ describe('InvoiceInDescriptor', () => {
|
||||||
'.q-card:nth-child(3) .vn-label-value:nth-child(5) > .value > span';
|
'.q-card:nth-child(3) .vn-label-value:nth-child(5) > .value > span';
|
||||||
|
|
||||||
it('should booking and unbooking the invoice properly', () => {
|
it('should booking and unbooking the invoice properly', () => {
|
||||||
|
cy.viewport(1280, 720);
|
||||||
cy.login('developer');
|
cy.login('developer');
|
||||||
cy.visit(`/#/invoice-in/1/summary?limit=10`);
|
cy.visit('/#/invoice-in/1/summary');
|
||||||
|
|
||||||
cy.openLeftMenu();
|
|
||||||
cy.openActionsDescriptor();
|
cy.openActionsDescriptor();
|
||||||
cy.get(firstDescritorOpt).click();
|
cy.get(firstDescritorOpt).click();
|
||||||
cy.get(dialogBtns).eq(1).click();
|
cy.get(dialogBtns).eq(1).click();
|
||||||
cy.get('.fullscreen').first().click();
|
|
||||||
cy.get(isBookedField).should('have.attr', 'title', 'true');
|
cy.get(isBookedField).should('have.attr', 'title', 'true');
|
||||||
|
|
||||||
cy.openLeftMenu();
|
|
||||||
cy.openActionsDescriptor();
|
|
||||||
cy.get(firstDescritorOpt).click();
|
cy.get(firstDescritorOpt).click();
|
||||||
cy.get(dialogBtns).eq(1).click();
|
cy.get(dialogBtns).eq(1).click();
|
||||||
cy.get(isBookedField).should('have.attr', 'title', 'false');
|
cy.get(isBookedField).should('have.attr', 'title', 'false');
|
||||||
|
|
|
@ -5,14 +5,11 @@ import jsconfigPaths from 'vite-jsconfig-paths';
|
||||||
import VueI18nPlugin from '@intlify/unplugin-vue-i18n/vite';
|
import VueI18nPlugin from '@intlify/unplugin-vue-i18n/vite';
|
||||||
import path from 'path';
|
import path from 'path';
|
||||||
|
|
||||||
// https://vitejs.dev/config/
|
|
||||||
export default defineConfig({
|
export default defineConfig({
|
||||||
test: {
|
test: {
|
||||||
environment: 'happy-dom',
|
environment: 'happy-dom',
|
||||||
setupFiles: 'test/vitest/setup-file.js',
|
setupFiles: 'test/vitest/setup-file.js',
|
||||||
include: [
|
include: [
|
||||||
// Matches vitest tests in any subfolder of 'src' or into 'test/vitest/__tests__'
|
|
||||||
// Matches all files with extension 'js', 'jsx', 'ts' and 'tsx'
|
|
||||||
'test/vitest/__tests__/**/*.{test,spec}.{js,mjs,cjs,ts,mts,cts,jsx,tsx}',
|
'test/vitest/__tests__/**/*.{test,spec}.{js,mjs,cjs,ts,mts,cts,jsx,tsx}',
|
||||||
],
|
],
|
||||||
},
|
},
|
Loading…
Reference in New Issue