- Per generare la sinossi è possibile estrarre con 1 click la descrizione sia da Amazon che da GruppoMacro.

- corretto piccolo bug sul catalogo.
This commit is contained in:
Surya Paolo
2025-05-20 12:21:31 +02:00
parent 8ff204657e
commit fea40922cb
26 changed files with 406 additions and 125 deletions

View File

@@ -779,6 +779,7 @@ export default defineComponent({
const options = {
update,
aggiornasoloSeVuoti,
forzaricarica: true,
};
const myparam = { product_id: myproduct.value._id, options };
@@ -787,14 +788,22 @@ export default defineComponent({
const dataupdated = await globalStore
.scrapingBook(myparam)
.then((dataupdated) => {
if (dataupdated) {
if (dataupdated && update) {
tools.showPositiveNotif($q, t('dbgm.scrapingOkUpdated'));
}
if (!dataupdated) {
tools.showNegativeNotif($q, t('dbgm.dati_non_estratti'));
visudataextracted.value = false
}
return dataupdated;
})
.catch((e) => {
console.error('Errore aggiornamento singolo libro:', e);
tools.showNegativeNotif($q, t('dbgm.updateLocalDb_ERROR'));
if (update) {
console.error('Errore aggiornamento singolo libro:', e);
tools.showNegativeNotif($q, t('dbgm.updateLocalDb_ERROR'));
} else {
tools.showNegativeNotif($q, t('dbgm.err_extraction'));
}
loading.value = false;
});
loading.value = false;
@@ -803,7 +812,6 @@ export default defineComponent({
}
async function getDataExtracted() {
visudataextracted.value = true;
loading.value = true;
@@ -814,8 +822,9 @@ export default defineComponent({
const html = tools.generateHtmlTableFromObject(data);
dataextractedWeb.value = html;
} catch (error) {
console.error('Errore durante l\'estrazione dei dati:', error);
console.error("Errore durante l'estrazione dei dati:", error);
dataextractedWeb.value = 'Errore nel caricamento dei dati.';
visudataextracted.value = false;
} finally {
loading.value = false;
}

View File

@@ -885,7 +885,7 @@
>
<q-card class="dialog_card">
<q-toolbar class="bg-primary text-white">
<q-toolbar-title> Visu </q-toolbar-title>
<q-toolbar-title> Visu dati Estratti </q-toolbar-title>
<q-btn
flat
round
@@ -895,7 +895,8 @@
></q-btn>
</q-toolbar>
<q-card-section class="q-pa-xs inset-shadow">
<div class="row justify-center">Dati</div><br>
<div class="row justify-center"><h3>Dati Estratti</h3></div>
<br />
<q-inner-loading
id="spinner"
:showing="loading"
@@ -909,6 +910,28 @@
<div v-html="dataextractedWeb"></div>
</q-card-section>
<q-card-actions class="justify-center">
<q-btn
v-if="tools.isAdmin()"
rounded
:disabled="loading"
class="q-ma-sm"
color="accent"
icon="fas fa-save"
label="Aggiorna (solo Vuoti)"
@click="scrapingBook(true, true)"
></q-btn>
<q-btn
v-if="tools.isAdmin()"
rounded
:disabled="loading"
class="q-ma-sm"
color="negative"
icon="fas fa-database"
label="Sovrascrivi Tutti i Dati"
@click="scrapingBook(true, false)"
></q-btn>
</q-card-actions>
</q-card>
</q-dialog>
<q-dialog

View File

@@ -1,31 +1,34 @@
import { PropType, computed, defineComponent, onMounted, ref, watch } from "vue";
import draggable from 'vuedraggable'
import {
PropType,
computed,
defineComponent,
onMounted,
ref,
watch,
} from 'vue';
import draggable from 'vuedraggable';
import { tools } from '@tools'
import { tools } from '@tools';
import { useGlobalStore } from '@src/store/globalStore'
import { useGlobalStore } from '@src/store/globalStore';
import { CMyEditorAI } from '@src/components/CMyEditorAI'
import { CAITools } from '@src/components/CAITools'
import { CMyEditorAI } from '@src/components/CMyEditorAI';
import { CAITools } from '@src/components/CAITools';
import { costanti } from '@costanti'
import type {
IMyScheda,
IProduct,
IRecFields
} from '@src/model';
import { shared_consts } from "app/src/common/shared_vuejs";
import { useProducts } from "app/src/store/Products";
import { useI18n } from "vue-i18n";
import { useQuasar } from "quasar";
import { costanti } from '@costanti';
import type { IMyScheda, IProduct, IRecFields } from '@src/model';
import { shared_consts } from 'app/src/common/shared_vuejs';
import { useProducts } from 'app/src/store/Products';
import { useI18n } from 'vue-i18n';
import { useQuasar } from 'quasar';
export default defineComponent({
name: "CModifTrafiletto",
name: 'CModifTrafiletto',
emits: ['updateproductmodif', 'close', 'Savedb'],
components: {
CMyEditorAI, CAITools
CMyEditorAI,
CAITools,
},
props: {
modelValue: {
@@ -64,38 +67,57 @@ export default defineComponent({
type: Number,
required: false,
default: 0,
}
},
},
setup(props, { emit }) {
// Copia locale della lista_prodotti per manipolazione interna
const $q = useQuasar()
const { t } = useI18n()
const $q = useQuasar();
const { t } = useI18n();
const globalStore = useGlobalStore()
const products = useProducts()
const globalStore = useGlobalStore();
const products = useProducts();
const mytab = ref('descr')
const mytab = ref('descr');
const loading = ref(false)
const loading = ref(false);
const updatetogm = ref(false)
const field_updated_toGM = ref('')
const updatetogm = ref(false);
const field_updated_toGM = ref('');
const myproduct = ref<IProduct>({ ...props.modelValue })
const myproduct = ref<IProduct>({ ...props.modelValue });
const id = computed(() => myproduct.value.productInfo._id)
const id = computed(() => myproduct.value.productInfo._id);
const myvalue = computed<string>(() => {
return myproduct.value.productInfo[props.mykey]
})
return myproduct.value.productInfo[props.mykey];
});
watch(() => props.modelValue, (newVal) => {
myproduct.value = { ...newVal };
}, { deep: false });
watch(
() => props.modelValue,
(newVal) => {
myproduct.value = { ...newVal };
},
{ deep: false }
);
const loadMyScrapingBook = async () => {
loading.value = true;
try {
const myscrapingbook = await products.loadMyScrapingBook(
myproduct.value.isbn, false
);
myproduct.value.myscrapingbook = myscrapingbook;
} catch (error) {
console.error('Errore caricamento MyScrapingBook:', error);
} finally {
loading.value = false;
}
};
async function mounted() {
// carico anche altri dati del product
await loadMyScrapingBook();
}
/*
@@ -110,45 +132,94 @@ export default defineComponent({
function updateproductmodif(element: any) {
// console.log('CModifTrafiletto updateproductmodif ', element)
emit('updateproductmodif', element)
emit('updateproductmodif', element);
}
function Savedb(element: any) {
tools.saveInDBForTypes($q, props.mykey, element, props.type, true, props.table, props.mysubkey, id.value, null, '');
tools.saveInDBForTypes(
$q,
props.mykey,
element,
props.type,
true,
props.table,
props.mysubkey,
id.value,
null,
''
);
}
async function updateproduct(load?: boolean) {
myproduct.value = await products.getProductById(myproduct.value._id, load)
myproduct.value = await products.getProductById(
myproduct.value._id,
load
);
}
const copyToClipboard = (text) => {
navigator.clipboard.writeText(text)
navigator.clipboard
.writeText(text)
.then(() => {
$q.notify({
message: 'Testo copiato negli appunti!',
color: 'positive',
icon: 'check',
position: 'top'
position: 'top',
});
})
.catch(err => {
.catch((err) => {
console.error('Errore durante la copia:', err);
$q.notify({
message: 'Errore nella copia',
color: 'negative',
icon: 'error',
position: 'top'
position: 'top',
});
});
}
};
function handleShowAndSave(payload: any) {
Savedb(payload);
updateproductmodif(payload);
}
onMounted(mounted)
function getPrompt() {
// Prompt:
let mydescr = 'Scrivimi la sinossi del seguente libro, che andrà nel catalogo libri, di massimo 680 battute, senza spiegazione, senza titolo iniziale, solo la sinossi. Togli eventuali riferimenti a chi ha fatto la prefazione. Rendilo un po\' accattivante, ma non troppo. \n\n';
return mydescr
}
function copyDescrizioneFromScrapingData() {
let mydescr = '';
const data = myproduct?.value.myscrapingbook;
if (!data) return false;
mydescr = getPrompt()
mydescr += 'Titolo Libro: ' + data.titolo + '\n';
mydescr += 'Autore Libro: ' + data.autore + '\n';
mydescr += 'DESCRIZIONE LIBRO: \n';
mydescr += data.descrizione_lunga;
tools.copyStringToClipboard($q, mydescr, false);
}
function copyDescrizioneFromGruppoMacro() {
let mydescr = '';
mydescr = getPrompt()
mydescr += 'Titolo Libro: ' + myproduct?.value.productInfo?.name + '\n';
mydescr += 'Autore Libro: ' + products.getAutoriByArrayAuthors(myproduct?.productInfo?.authors) + '\n';
mydescr += 'DESCRIZIONE LIBRO: \n';
mydescr += myproduct?.value.productInfo?.descrizione_completa_macro;
tools.copyStringToClipboard($q, mydescr, false);
}
onMounted(mounted);
return {
tools,
@@ -167,6 +238,9 @@ export default defineComponent({
copyToClipboard,
myvalue,
handleShowAndSave,
}
}
})
copyDescrizioneFromScrapingData,
copyDescrizioneFromGruppoMacro,
loading,
};
},
});

View File

@@ -1,7 +1,9 @@
<template>
<q-card class="dialog_card">
<q-toolbar class="bg-primary text-white">
<q-toolbar-title> Modifica a {{ myproduct.productInfo.name }}</q-toolbar-title>
<q-toolbar-title>
Modifica a '{{ myproduct.productInfo.name }}''</q-toolbar-title
>
<q-btn
flat
round
@@ -64,6 +66,60 @@
:maxlength="maxlength"
>
</CMyEditorAI>
<p
v-if="copied"
class="text-green q-ma-md"
>
Descrizione Copiata negli appunti!
</p>
<q-spinner
v-if="loading"
color="primary"
size="3em"
:thickness="2"
/>
<div v-if="!loading">
<div
v-if="myproduct?.myscrapingbook?.descrizione_lunga"
class="q-ma-xs q-pa-xs"
>
<div class="text-h6 q-mb-sm">
Descrizione Estratta da Amazon:
</div>
{{ myproduct?.myscrapingbook?.descrizione_lunga }}
<br />
<q-btn
rounded
color="primary"
label="Copia per ChatGPT"
icon="content_copy"
@click="copyDescrizioneFromScrapingData"
class="q-ma-md justify-center"
>
</q-btn>
</div>
<div
v-if="myproduct?.productInfo?.descrizione_completa_macro"
class="q-ma-xs q-pa-xs"
>
<div class="text-h6 q-mb-sm">
Descrizione Estratta dal sito GruppoMacro.com:
</div>
{{ myproduct?.productInfo?.descrizione_completa_macro }}
<br />
<q-btn
rounded
color="primary"
label="Copia per ChatGPT"
icon="content_copy"
@click="copyDescrizioneFromGruppoMacro"
class="q-ma-md justify-center"
>
</q-btn>
</div>
</div>
</q-tab-panel>
</q-tab-panels>
</div>

View File

@@ -20,6 +20,7 @@ import { CMyValueDb } from '@src/components/CMyValueDb';
import { CSchedaProdotto } from '@src/components/CSchedaProdotto';
import { CSearchProduct } from '@src/components/CSearchProduct';
import { CMyDialog } from '@src/components/CMyDialog';
import { CModifTrafiletto } from '@src/components/CModifTrafiletto';
import { costanti } from '@costanti';
import { IAuthor, ICatProd } from 'app/src/model';
@@ -42,6 +43,7 @@ export default defineComponent({
CViewTable,
CLabel,
CSchedaProdotto,
CModifTrafiletto,
},
props: {
lista_prodotti: {
@@ -90,6 +92,7 @@ export default defineComponent({
const field_updated_fromGM = ref('');
const modifOn = ref(false);
const modifTrafiletto = ref(false);
const sortAttribute = ref('');
const sortDirection = ref(1);
@@ -100,6 +103,10 @@ export default defineComponent({
function handleUpdate(newList) {
internalProducts.value = newList;
riaggiornaListaProdAlGenitore()
}
function riaggiornaListaProdAlGenitore() {
emit('update:lista_prodotti', internalProducts.value);
}
@@ -545,8 +552,8 @@ export default defineComponent({
case 'trafiletto':
return element.productInfo?.descr_trafiletto_catalogo?.length >
100
? 'SI'
: 'NO';
? 'SI ✏️'
: 'NO ✏️';
case 'catprods':
return tools.formatCatProds(element.productInfo?.catprods);
@@ -699,6 +706,11 @@ export default defineComponent({
if (!element) return {};
switch (field.field) {
case 'trafiletto':
return {
cursor: 'pointer',
textAlign: 'center',
};
case 'prezzo':
case 'prezzo_sconto':
return { width: '55px', textAlign: 'right' };
@@ -725,11 +737,6 @@ export default defineComponent({
textAlign: 'right',
};
case 'trafiletto':
return {
textAlign: 'center',
};
default:
return {};
}
@@ -910,6 +917,14 @@ export default defineComponent({
modifOn.value = true;
}
}
function modifyTrafiletto(element: any) {
if (element) {
selProd.value = element;
cmd.value = shared_consts.SCHEDA_PRODOTTO.CMD_MODIFICA;
modifTrafiletto.value = true;
}
}
function updateProduct(element: any) {
selProd.value = element;
@@ -941,6 +956,7 @@ export default defineComponent({
}
return prod;
});
riaggiornaListaProdAlGenitore()
} catch (e) {
console.error('err', e);
}
@@ -1014,6 +1030,8 @@ export default defineComponent({
function getFieldClick(element: any, field: any): (() => void) | null {
switch (field.field) {
case 'trafiletto':
return () => modifyTrafiletto(element);
case 'validato':
return () => modifyProduct(element);
case 'image':
@@ -1124,6 +1142,7 @@ export default defineComponent({
loading,
showQtaDisponibile,
modifOn,
modifTrafiletto,
updateproductmodif,
optionscatalogo,
t,

View File

@@ -228,6 +228,25 @@
>
</CSearchProduct>
</CMyDialog>
<q-dialog
v-if="selProd && modifTrafiletto"
v-model="modifTrafiletto"
maximized
>
<CModifTrafiletto
v-model="selProd"
titolo="Sinossi"
table="productinfos"
mykey="descr_trafiletto_catalogo"
:canModify="true"
:type="costanti.FieldType.editor_nohtml"
@updateproductmodif="updateproductmodif"
@close="modifTrafiletto = false"
:maxlength="680"
>
</CModifTrafiletto>
</q-dialog>
<q-dialog
v-model="modifOn"
maximized

View File

@@ -20,4 +20,16 @@
vertical-align: middle;
}
.fullscreen-container{
position: fixed;
top: 0;
right: 0;
bottom: 0;
left: 0;
background-color: rgba(0,0,0,0.5);
z-index: 1000;
display: flex;
align-items: center;
justify-content: center;
}

View File

@@ -613,6 +613,7 @@ export default defineComponent({
const options = {
update,
aggiornasoloSeVuoti,
forzaricarica: true,
}
const myparam = { product_id: myproduct.value._id, options };

View File

@@ -1,5 +1,5 @@
<template>
<q-card class="dialog_card">
<q-card class="dialog_card" :maximized="true">
<q-toolbar class="bg-primary text-white">
<q-toolbar-title> Modifica a {{ myproduct.productInfo.name }}</q-toolbar-title>
<q-btn
@@ -72,8 +72,8 @@
</q-tab-panels>
</div>
</q-card-section>
<q-card-actions>
<div class="">
<q-card-actions class="fixed-bottom">
<div>
<q-btn
rounded
class="q-ma-sm"