feat(claim_photo): add upload button
gitea/salix-front/pipeline/head This commit looks good Details

This commit is contained in:
Alex Moreno 2023-02-16 14:10:17 +01:00
parent 0b0e73d65b
commit 38a6b93a5d
5 changed files with 125 additions and 71 deletions

View File

@ -241,6 +241,7 @@ export default {
summary: 'Summary', summary: 'Summary',
basicData: 'Basic Data', basicData: 'Basic Data',
rma: 'RMA', rma: 'RMA',
photos: 'Photos',
}, },
list: { list: {
customer: 'Customer', customer: 'Customer',
@ -296,7 +297,7 @@ export default {
}, },
photo: { photo: {
fileDescription: fileDescription:
'Reclamacion ID {{claimId}} del cliente {{clientName}} id {{clientId}}', 'Reclamacion ID {claimId} del cliente {clientName} id {clientId}',
}, },
}, },
invoiceOut: { invoiceOut: {

View File

@ -240,6 +240,7 @@ export default {
summary: 'Resumen', summary: 'Resumen',
basicData: 'Datos básicos', basicData: 'Datos básicos',
rma: 'RMA', rma: 'RMA',
photos: 'Fotos',
}, },
list: { list: {
customer: 'Cliente', customer: 'Cliente',
@ -294,8 +295,7 @@ export default {
returnOfMaterial: 'Autorización de retorno de materiales (RMA)', returnOfMaterial: 'Autorización de retorno de materiales (RMA)',
}, },
photo: { photo: {
fileDescription: fileDescription: 'Claim id {claimId} from client {clientName} id {clientId}',
'Claim id {{claimId}} from client {{clientName}} id {{clientId}}',
}, },
}, },
invoiceOut: { invoiceOut: {

View File

@ -3,6 +3,7 @@ import { ref, computed } from 'vue';
import axios from 'axios'; import axios from 'axios';
import { useQuasar } from 'quasar'; import { useQuasar } from 'quasar';
import VnConfirm from 'src/components/ui/VnConfirm.vue'; import VnConfirm from 'src/components/ui/VnConfirm.vue';
import TeleportSlot from 'src/components/ui/TeleportSlot.vue';
import { useRouter } from 'vue-router'; import { useRouter } from 'vue-router';
import { useI18n } from 'vue-i18n'; import { useI18n } from 'vue-i18n';
@ -22,6 +23,7 @@ const claimId = computed(function () {
}); });
const claimDms = ref([]); const claimDms = ref([]);
const claimDmsRef = ref();
const dmsType = ref({}); const dmsType = ref({});
const config = ref({}); const config = ref({});
const dms = ref({}); const dms = ref({});
@ -75,22 +77,28 @@ async function deleteDms(index) {
function setClaimDms(data) { function setClaimDms(data) {
claimDms.value = data.map((media) => { claimDms.value = data.map((media) => {
media.isVideo = media.dms.contentType == 'video/mp4'; media.isVideo = media.dms.contentType == 'video/mp4';
media.url = getImagePath(media);
return media; return media;
}); });
} }
function uploadFile() { async function uploadFile() {
claimDmsRef.value.fetch();
const element = document.createElement('input'); const element = document.createElement('input');
element.setAttribute('type', 'file'); element.setAttribute('type', 'file');
element.setAttribute('multiple', true); element.setAttribute('multiple', true);
element.click(); element.click();
element.addEventListener('change', () => setDefaultParams()); element.addEventListener('change', () => {
setDefaultParams();
create(element.files).then(() => claimDmsRef.value.fetch());
});
} }
function setDefaultParams() { function setDefaultParams() {
console.log(config.value); console.log(config.value);
console.log(dmsType.value); console.log(dmsType.value);
console.log(claimDms.value);
dms.value = { dms.value = {
hasFile: false, hasFile: false,
hasFileAttached: false, hasFileAttached: false,
@ -100,12 +108,32 @@ function setDefaultParams() {
dmsTypeId: dmsType.value.id, dmsTypeId: dmsType.value.id,
description: t('claim.photo.fileDescription', { description: t('claim.photo.fileDescription', {
claimId: claimId.value, claimId: claimId.value,
clientId: claimDms.value.clientFk,
clientName: claimDms.value.clientName, clientName: claimDms.value.clientName,
clientId: claimDms.value.clientFk,
}).toUpperCase(), }).toUpperCase(),
}; };
console.log(dms.value); console.log(dms.value);
} }
async function create(files) {
const formData = new FormData();
for (let i = 0; i < files.length; i++) formData.append(files[i].name, files[i]);
const query = `claims/${claimId.value}/uploadFile`;
await axios({
method: 'POST',
url: query,
data: formData,
params: dms.value,
});
quasar.notify({
message: t('globals.dataSaved'),
type: 'positive',
icon: 'check',
});
}
</script> </script>
<template> <template>
<fetch-data <fetch-data
@ -114,6 +142,7 @@ function setDefaultParams() {
@on-fetch="(data) => setClaimDms(data)" @on-fetch="(data) => setClaimDms(data)"
limit="20" limit="20"
auto-load auto-load
ref="claimDmsRef"
/> />
<fetch-data <fetch-data
url="DmsTypes/findOne" url="DmsTypes/findOne"
@ -133,8 +162,7 @@ function setDefaultParams() {
color="primary" color="primary"
text-color="white" text-color="white"
size="md" size="md"
class="all-pointer-events absolute" class="all-pointer-events absolute delete-button zindex"
style="top: 10px; left: 10px; z-index: 1"
@click.stop="viewDeleteDms(index)" @click.stop="viewDeleteDms(index)"
round round
/> />
@ -142,8 +170,7 @@ function setDefaultParams() {
name="play_arrow" name="play_arrow"
color="primary" color="primary"
size="xl" size="xl"
class="absolute-center" class="absolute-center zindex"
style="z-index: 1"
v-if="media.isVideo" v-if="media.isVideo"
@click.stop="openDialog(media.dmsFk)" @click.stop="openDialog(media.dmsFk)"
flat flat
@ -152,77 +179,76 @@ function setDefaultParams() {
</q-icon> </q-icon>
<q-card class="multimedia relative-position"> <q-card class="multimedia relative-position">
<q-img <q-img
:src="getImagePath(media)" :src="media.url"
class="rounded-borders cursor-pointer" class="rounded-borders cursor-pointer fit"
@click="openDialog(media.dmsFk)" @click="openDialog(media.dmsFk)"
v-if="!media.isVideo" v-if="!media.isVideo"
> >
</q-img> </q-img>
<video <video
:src="getImagePath(media)" :src="media.url"
class="rounded-borders cursor-pointer" class="rounded-borders cursor-pointer fit"
muted="muted" muted="muted"
v-if="media.isVideo" v-if="media.isVideo"
@click="openDialog(media.dmsFk)" @click="openDialog(media.dmsFk)"
></video> />
</q-card> </q-card>
</div> </div>
</div> </div>
<q-page-sticky position="bottom-right" :offset="[18, 18]"> <teleport-slot v-if="!quasar.platform.is.mobile" to="#header-actions">
<q-btn fab icon="add" color="primary" @click="uploadFile()" /> <div class="row q-gutter-x-sm">
</q-page-sticky> <q-btn @click="uploadFile()" icon="add" color="primary" dense rounded>
<q-tooltip bottom> {{ t('globals.add') }} </q-tooltip>
</q-btn>
<q-separator vertical />
</div>
</teleport-slot>
<teleport-slot to=".q-footer">
<q-tabs align="justify" inline-label narrow-indicator>
<q-tab @click="uploadFile()" icon="add_circle" :label="t('globals.add')" />
</q-tabs>
</teleport-slot>
<!-- MULTIMEDIA DIALOG START--> <!-- MULTIMEDIA DIALOG START-->
<q-dialog <q-dialog
v-model="multimediaDialog" v-model="multimediaDialog"
:maximized="multimediaMaximizedDialog"
transition-show="slide-up" transition-show="slide-up"
transition-hide="slide-down" transition-hide="slide-down"
> >
<q-card style="height: 80%; min-width: 80%"> <q-toolbar class="absolute zindex close-button">
<q-card-section class="row items-center q-pb-none q-gutter-md"> <q-space />
<q-avatar <q-btn icon="close" color="primary" round dense v-close-popup />
:icon="icon" </q-toolbar>
color="primary" <q-carousel swipeable animated v-model="multimediaSlide" arrows class="fit">
text-color="white" <q-carousel-slide
size="xl" v-for="media of claimDms"
v-if="icon" :key="media.dmsFk"
:name="media.dmsFk"
>
<q-img
:src="media.url"
class="fit"
fit="scale-down"
v-if="!media.isVideo"
/> />
<span class="text-h6 text-grey">{{ message }}</span> <q-video
<q-space /> class="q-ma-none fit"
<q-btn icon="close" flat round dense v-close-popup /> :src="media.url"
</q-card-section> v-if="media.isVideo"
muted
<q-card-section class="q-pt-none"> />
<q-carousel swipeable animated v-model="multimediaSlide" arrows> </q-carousel-slide>
<q-carousel-slide </q-carousel>
v-for="media of claimDms"
:key="media.dmsFk"
:name="media.dmsFk"
class="fit"
>
<q-img
height="100%"
fit="scale-down"
:src="getImagePath(media)"
v-if="!media.isVideo"
/>
<q-video
class="q-ma-none"
:src="getImagePath(media)"
v-if="media.isVideo"
muted
/>
</q-carousel-slide>
</q-carousel>
</q-card-section>
</q-card>
</q-dialog> </q-dialog>
<!-- MULTIMEDIA DIALOG END--> <!-- MULTIMEDIA DIALOG END-->
</template> </template>
<style lang="scss" scoped> <style lang="scss" scoped>
.q-dialog__inner--minimized > div {
max-width: 80%;
}
.multimediaParent { .multimediaParent {
display: grid; display: grid;
grid-template-columns: repeat(auto-fill, minmax(250px, 1fr)); grid-template-columns: repeat(auto-fill, minmax(250px, 1fr));
@ -238,14 +264,10 @@ function setDefaultParams() {
height: 250px; height: 250px;
.q-img { .q-img {
height: 100%;
width: 100%;
object-fit: cover; object-fit: cover;
background-color: black; background-color: black;
} }
video { video {
height: 100%;
width: 100%;
object-fit: cover; object-fit: cover;
background-color: black; background-color: black;
} }
@ -255,9 +277,18 @@ function setDefaultParams() {
opacity: 0.5; opacity: 0.5;
} }
.q-video { .delete-button {
height: 100%; top: 10px;
width: 100%; left: 10px;
}
.close-button {
top: 1%;
right: 10%;
}
.zindex {
z-index: 1;
} }
</style> </style>

View File

@ -6,7 +6,7 @@ import { useRoute } from 'vue-router';
import axios from 'axios'; import axios from 'axios';
import Paginate from 'src/components/PaginateData.vue'; import Paginate from 'src/components/PaginateData.vue';
import FetchData from 'components/FetchData.vue'; import FetchData from 'components/FetchData.vue';
import TeleportSlot from 'components/ui/TeleportSlot'; import TeleportSlot from 'components/ui/TeleportSlot.vue';
import { toDate } from 'src/filters'; import { toDate } from 'src/filters';
const quasar = useQuasar(); const quasar = useQuasar();
@ -90,22 +90,38 @@ function hide() {
<q-list> <q-list>
<q-item class="q-pa-none"> <q-item class="q-pa-none">
<q-item-section> <q-item-section>
<q-item-label caption>{{ t('claim.rma.user') }}</q-item-label> <q-item-label caption>{{
<q-item-label>{{ row.worker.user.name }}</q-item-label> t('claim.rma.user')
}}</q-item-label>
<q-item-label>{{
row.worker.user.name
}}</q-item-label>
</q-item-section> </q-item-section>
</q-item> </q-item>
<q-item class="q-pa-none"> <q-item class="q-pa-none">
<q-item-section> <q-item-section>
<q-item-label caption>{{ t('claim.rma.created') }}</q-item-label> <q-item-label caption>{{
t('claim.rma.created')
}}</q-item-label>
<q-item-label> <q-item-label>
{{ toDate(row.created, { timeStyle: 'medium' }) }} {{
toDate(row.created, {
timeStyle: 'medium',
})
}}
</q-item-label> </q-item-label>
</q-item-section> </q-item-section>
</q-item> </q-item>
</q-list> </q-list>
</q-item-section> </q-item-section>
<q-card-actions vertical class="justify-between"> <q-card-actions vertical class="justify-between">
<q-btn flat round color="orange" icon="vn:bin" @click="confirmRemove(row.id)"> <q-btn
flat
round
color="orange"
icon="vn:bin"
@click="confirmRemove(row.id)"
>
<q-tooltip>{{ t('globals.remove') }}</q-tooltip> <q-tooltip>{{ t('globals.remove') }}</q-tooltip>
</q-btn> </q-btn>
</q-card-actions> </q-card-actions>
@ -124,7 +140,13 @@ function hide() {
</q-card-section> </q-card-section>
<q-card-actions align="right"> <q-card-actions align="right">
<q-btn flat :label="t('globals.no')" color="primary" v-close-popup autofocus /> <q-btn
flat
:label="t('globals.no')"
color="primary"
v-close-popup
autofocus
/>
<q-btn flat :label="t('globals.yes')" color="primary" @click="remove()" /> <q-btn flat :label="t('globals.yes')" color="primary" @click="remove()" />
</q-card-actions> </q-card-actions>
</q-card> </q-card>

View File

@ -81,7 +81,7 @@ export default {
path: 'photos', path: 'photos',
meta: { meta: {
title: 'photos', title: 'photos',
icon: 'vn:image', icon: 'image',
}, },
component: () => import('src/pages/Claim/Card/ClaimPhoto.vue'), component: () => import('src/pages/Claim/Card/ClaimPhoto.vue'),
}, },