diff --git a/assets/src/css/style.css b/assets/src/css/style.css index 6cb3e9fc3..582a9a2a1 100755 --- a/assets/src/css/style.css +++ b/assets/src/css/style.css @@ -1658,6 +1658,9 @@ body:not(.sidebar-collapse) .sidebar .nav-treeview .nav-treeview .nav-link { list-style: none; } +.readmore{ + overflow: hidden; +} @media screen and (max-width: 767px) { .navbar-left { diff --git a/assets/src/js/base/custom.js b/assets/src/js/base/custom.js index 986db2896..1a8d0fd89 100644 --- a/assets/src/js/base/custom.js +++ b/assets/src/js/base/custom.js @@ -98,6 +98,18 @@ $(document).ready(function () { }); }, 1000); + // Plugin readmore.js + $('.readmore').each( function(){ + height = $(this).data('height') ? parseInt($(this).data('height')) : 50; + + $(this).readmore({ + collapsedHeight: height, + moreLink: '' + globals.translations.readmore + '', + lessLink: '' + globals.translations.readless + '', + beforeToggle: function(){ setTimeout( 'alignMaxHeight(".module-header .card");', 300 ); }, + }); + }); + alignMaxHeight(".module-header .card"); $("#main_loading").fadeOut() @@ -130,6 +142,9 @@ function sendWhatsAppMessage(phoneNumber, message) { } function alignMaxHeight(element){ + // Azzera l'altezza fissa + $(element).css('height', 'auto'); + max_height = 0; $(element).each( function(){ if($(this).height() > max_height){ diff --git a/gulpfile.js b/gulpfile.js index 4d74e7058..e3020a593 100755 --- a/gulpfile.js +++ b/gulpfile.js @@ -127,6 +127,7 @@ const JS = gulp.parallel(() => { 'leaflet.fullscreen/Control.FullScreen.js', 'ismobilejs/dist/isMobile.min.js', 'ua-parser-js/dist/ua-parser.min.js', + 'readmore.js/readmore.js', ]; for (const i in vendor) { diff --git a/include/top.php b/include/top.php index be6aacb9d..21e04534a 100755 --- a/include/top.php +++ b/include/top.php @@ -124,6 +124,8 @@ if (Auth::check()) { 'singleCalendar' => tr("E' presente un solo periodo."), 'noResults' => tr('Nessun elemento trovato'), 'signatureMissing' => tr('Firma mancante.'), + 'readmore' => tr('Mostra tutto'), + 'readless' => tr('Mostra meno'), ]; foreach ($translations as $key => $value) { echo ' diff --git a/mail.php b/mail.php index 317acc898..7dec7e070 100755 --- a/mail.php +++ b/mail.php @@ -135,7 +135,7 @@ echo '
- {[ "type": "text", "label": "'.tr('Oggetto').'", "name": "subject", "value": "'.$subject.'", "required": 1 ]} + {[ "type": "text", "label": "'.tr('Oggetto').'", "name": "subject", "value": "'.prepare($subject).'", "required": 1 ]}
diff --git a/modules/articoli/plugins/articoli.giacenze.php b/modules/articoli/plugins/articoli.giacenze.php index 66bb7a2ba..3abcf3984 100755 --- a/modules/articoli/plugins/articoli.giacenze.php +++ b/modules/articoli/plugins/articoli.giacenze.php @@ -46,24 +46,6 @@ $query = 'SELECT HAVING `qta_ordinata` > 0"; -echo ' -
-
-

'.tr('Articolo').'

-
- -
-
-
- '.tr('Codice: ').''.$articolo->codice.' -
- -
- '.tr('Descrizione: ').''.$articolo->getTranslation('title').' -
-
-
-
'; /* ** Impegnato diff --git a/modules/articoli/plugins/articoli.movimenti.php b/modules/articoli/plugins/articoli.movimenti.php index ea1a91ccd..a71fc0392 100755 --- a/modules/articoli/plugins/articoli.movimenti.php +++ b/modules/articoli/plugins/articoli.movimenti.php @@ -22,25 +22,6 @@ include_once __DIR__.'/../../../core.php'; use Models\Module; use Modules\Anagrafiche\Anagrafica; -echo ' -
-
-

'.tr('Articolo').'

-
- -
-
-
- '.tr('Codice: ').''.$articolo->codice.' -
- -
- '.tr('Descrizione: ').''.$articolo->getTranslation('title').' -
-
-
-
'; - // Movimentazione degli articoli echo '
diff --git a/modules/articoli/src/Articolo.php b/modules/articoli/src/Articolo.php index 485a40b46..53b0f1f0f 100755 --- a/modules/articoli/src/Articolo.php +++ b/modules/articoli/src/Articolo.php @@ -29,6 +29,9 @@ use Modules\Interventi\Components\Articolo as ArticoloIntervento; use Modules\Iva\Aliquota; use Plugins\ListinoFornitori\DettaglioFornitore; use Traits\RecordTrait; +use Modules\Preventivi\Components\Articolo as ArticoloPreventivo; +use Models\Plugin; + class Articolo extends Model { @@ -82,6 +85,45 @@ class Articolo extends Model $this->save(); } + if (Plugin::where('name', 'Distinta base')->first()->id) { + $descrizione_ricorsivo = ''; + if (!empty($array['reference_type']) && !empty($array['reference_id'])) { + $object = new $array['reference_type'](); + $document = $object::find($array['reference_id']); + $dir = $document->direzione; + } + + if (!$this->componenti->isEmpty() && setting('Produci articoli della distinta base in fase di vendita') && $manuale == false && $dir == 'entrata' && $descrizone != "tr('Produzione articolo')" && $descrizone != "tr('Scomposizione articolo')") { + if ($qta < 0) { + $descrizione_ricorsivo = tr('Produzione articolo'); + } else { + $descrizione_ricorsivo = tr('Scomposizione articolo'); + } + $this->movimenta(-$qta, $descrizione_ricorsivo, $data, false); + $this->movimentaRicorsivo($qta, $descrizione_ricorsivo); + } + + if (!$this->componenti->isEmpty() && setting('Scomponi articolo padre in fase di acquisto') && $manuale == false && $dir == 'uscita' && $descrizone != "tr('Produzione articolo')" && $descrizone != "tr('Scomposizione articolo')") { + if ($qta < 0) { + $descrizione_ricorsivo = tr('Produzione articolo'); + } else { + $descrizione_ricorsivo = tr('Scomposizione articolo'); + } + $this->movimenta(-$qta, $descrizione_ricorsivo, $data, false); + $this->movimentaRicorsivo($qta, $descrizione_ricorsivo); + } + + if (!$this->componenti->isEmpty() && setting('Movimenta gli articoli figlio tramite i movimenti manuali') && $manuale) { + if ($qta < 0) { + $descrizione_ricorsivo = tr('Produzione articolo'); + } else { + $descrizione_ricorsivo = tr('Scomposizione articolo'); + } + $this->movimenta(-$qta, $descrizione_ricorsivo, $data, false); + $this->movimentaRicorsivo($qta, $descrizione_ricorsivo); + } + } + return $id; } @@ -351,4 +393,133 @@ class Articolo extends Model { return self::$translated_fields; } + /** + * @version distinta_base + */ + public function componenti() + { + return $this->belongsToMany(Articolo::class, 'mg_articoli_distinte', 'id_articolo', 'id_figlio')->withPivot('qta'); + } + + public function parti() + { + return $this->belongsToMany(Articolo::class, 'mg_articoli_distinte', 'id_figlio', 'id_articolo')->withPivot('qta'); + } + + /** + * @version distinta_base + */ + public function triggerChange(Articolo $trigger) + { + if ($this->sincronizza_prezzo_vendita) { + $this->prezzo_vendita = $this->totale_vendita; + } + + if ($this->sincronizza_prezzo_acquisto) { + $this->prezzo_acquisto = $this->totale_acquisto; + } + + $this->save(); + } + + /** + * @version distinta_base + */ + public function save(array $attributes = []) + { + if (Plugin::where('name', 'Distinta base')->first()->id) { + // Supporto al plugin Fornitori (prodotto per il coefficiente relativo) + if (!$this->componenti->isEmpty()) { + if ($this->sincronizza_prezzo_vendita) { + $this->prezzo_vendita = $this->totale_vendita; + } + + if ($this->sincronizza_prezzo_acquisto) { + $this->prezzo_acquisto = $this->totale_acquisto; + } + } + + $result = parent::save($attributes); + + $parti = $this->parti; + foreach ($parti as $parte) { + $parte->pivot->qta = 1; // Fix per le quantità inverse + $parte->triggerChange($this); + } + + return $result; + } else { + return parent::save($attributes); + } + } + + /** + * Funzione per inserire i movimenti di magazzino. + * + * @version distinta_base + * + * @param null $descrizone + * @param null $data + * @param bool $manuale + * @param array $array + * + * @return bool + */ + public function movimentaRicorsivo($qta, $descrizone = null, $data = null, $manuale = false, $array = []) + { + $componenti = $this->componenti; + + $suffix = ' (di.ba.)'; + $descrizone = str_contains((string) $descrizone, $suffix) ? $descrizone : $descrizone.$suffix; + + foreach ($componenti as $componente) { + $qta_componente = $qta * $componente->pivot->qta; + $componente->movimenta($qta_componente, $descrizone, $data, $manuale, $array); + } + } + + public function getTotaleAcquistoAttribute() + { + $componenti = $this->componenti; + $prezzo_acquisto = 0; + + foreach ($componenti as $componente) { + $prezzo_acquisto += $componente->prezzo_acquisto * $componente->pivot->qta; + } + + return $prezzo_acquisto; + } + + public function getTotaleVenditaAttribute() + { + $componenti = $this->componenti; + $prezzo_vendita = 0; + + foreach ($componenti as $componente) { + $prezzo_vendita += $componente->prezzo_vendita * $componente->pivot->qta; + } + + return $prezzo_vendita; + } + + /** + * @version distinta_base + */ + public function componenti_preventivo($idrigapreventivo = null) + { + return database()->select('mg_articoli_distinte_preventivi', ['id_figlio', 'qta', 'prezzo_acquisto', 'prezzo_vendita', 'sconto', 'tipo_sconto'], [ + 'idrigapreventivo' => $idrigapreventivo, + 'id_articolo' => $this->id, + ]); + } + + public function triggerChangePreventivo($idrigapreventivo) + { + $riga = ArticoloPreventivo::find($idrigapreventivo); + + $riga->costo_unitario = $riga->totale_acquisto ?: 0; + $riga->setPrezzoUnitario($riga->totale_vendita, $riga->idiva); + + $riga->save(); + } } diff --git a/modules/banche/src/IBAN.php b/modules/banche/src/IBAN.php index 601812ab9..2fe44a6e5 100644 --- a/modules/banche/src/IBAN.php +++ b/modules/banche/src/IBAN.php @@ -392,8 +392,7 @@ class IBAN $this->nation = $nation = substr($iban, 0, 2); $info = self::$countries[$nation]; - $structure = $info['structure']; - $structure = str_replace(' ', '', $structure); + $structure = $info['structure'] ? str_replace(' ', '', $info['structure']) : ''; $regex = $nation; $keys = array_keys(self::$parsers); diff --git a/modules/ddt/actions.php b/modules/ddt/actions.php index 661efc028..8daf27729 100755 --- a/modules/ddt/actions.php +++ b/modules/ddt/actions.php @@ -304,6 +304,12 @@ switch (filter('op')) { $ddt->idreferente = $documento->idreferente; $ddt->idagente = $documento->idagente; + if (filter('op') == 'add_ordine') { + $ddt->idspedizione = $documento->idspedizione; + $ddt->idporto = $documento->idporto; + $ddt->idvettore = $documento->idvettore; + } + $ddt->save(); $id_record = $ddt->id; diff --git a/modules/ddt/edit.php b/modules/ddt/edit.php index 0288e835f..ea0b4efcf 100755 --- a/modules/ddt/edit.php +++ b/modules/ddt/edit.php @@ -274,7 +274,7 @@ $esterno = $dbo->selectOne('dt_spedizione', 'esterno', [ 'id' => $record['idspedizione'], ])['esterno']; ?> - {[ "type": "select", "label": "", "name": "idvettore", "ajax-source": "vettori", "value": "$idvettore$", "disabled": , "required": , "icon-": "add|first()->id; ?>|tipoanagrafica=Vettore&readonly_tipo=1|btn_idvettore|", "class": "" ]} + {[ "type": "select", "label": "", "name": "idvettore", "ajax-source": "vettori", "value": "$idvettore$", "disabled": , "required": , "icon-after": "add|first()->id; ?>|tipoanagrafica=Vettore&readonly_tipo=1|btn_idvettore|", "class": "" ]}
diff --git a/modules/fatture/bulk.php b/modules/fatture/bulk.php index dc19f7225..5f8fcbd99 100755 --- a/modules/fatture/bulk.php +++ b/modules/fatture/bulk.php @@ -212,7 +212,7 @@ switch (post('op')) { $dst = $fe->getFilename(); $src = $dbo->selectOne('zz_files', 'filename', ['original' => $dst])['filename']; } else { - $src = basename($fattura->uploads()->where('name', 'Fattura Elettronica')->first()->filepath); + $src = basename($fattura->uploads()->where('name', 'Fattura Elettronica')->first()->filename); $dst = basename($fattura->uploads()->where('name', 'Fattura Elettronica')->first()->original_name); } @@ -271,7 +271,7 @@ switch (post('op')) { foreach ($fatture as $r) { $fattura = Fattura::find($r['id']); $zz_file = $dbo->table('zz_files')->where('id_module', '=', $id_module)->where('id_record', '=', $fattura->id)->where('name', 'like', 'Ricevuta%')->first(); - $src = basename($fattura->uploads()->where('id', $zz_file->id)->first()->filepath); + $src = basename($fattura->uploads()->where('id', $zz_file->id)->first()->filename); $dst = basename($fattura->uploads()->where('id', $zz_file->id)->first()->original_name); $file = slashes($module->upload_directory.'/'.$src); @@ -710,29 +710,31 @@ $operations['exportFE-bulk'] = [ ], ]; + +$operations['export-ricevute-bulk'] = [ + 'text' => ' '.tr('Esporta ricevute').'', + 'data' => [ + 'title' => '', + 'msg' => tr('Vuoi davvero esportare le ricevute selezionate in un archivio ZIP?'), + 'button' => tr('Procedi'), + 'class' => 'btn btn-lg btn-warning', + 'blank' => true, + ], +]; + + +$operations['export-xml-bulk'] = [ + 'text' => ' '.tr('Esporta XML').'', + 'data' => [ + 'title' => '', + 'msg' => tr('Vuoi davvero esportare le fatture elettroniche selezionate in un archivio ZIP?'), + 'button' => tr('Procedi'), + 'class' => 'btn btn-lg btn-warning', + 'blank' => true, + ], +]; + if ($module->getTranslation('title') == 'Fatture di vendita') { - $operations['export-ricevute-bulk'] = [ - 'text' => ' '.tr('Esporta ricevute').'', - 'data' => [ - 'title' => '', - 'msg' => tr('Vuoi davvero esportare le ricevute selezionate in un archivio ZIP?'), - 'button' => tr('Procedi'), - 'class' => 'btn btn-lg btn-warning', - 'blank' => true, - ], - ]; - - $operations['export-xml-bulk'] = [ - 'text' => ' '.tr('Esporta XML').'', - 'data' => [ - 'title' => '', - 'msg' => tr('Vuoi davvero esportare le fatture elettroniche selezionate in un archivio ZIP?'), - 'button' => tr('Procedi'), - 'class' => 'btn btn-lg btn-warning', - 'blank' => true, - ], - ]; - $operations['genera-xml'] = [ 'text' => ' '.tr('Genera fatture elettroniche').'', 'data' => [ diff --git a/modules/interventi/header.php b/modules/interventi/header.php index 394e256b9..0d3e85c3a 100644 --- a/modules/interventi/header.php +++ b/modules/interventi/header.php @@ -172,9 +172,11 @@ echo '

'.(count($interventi_programmati) == 0 ? tr('Non sono presenti altre attività programmate') : 'Attività aperte:'); if (count($interventi_programmati) != 0) { + echo '

'; foreach ($interventi_programmati as $intervento_programmato) { echo ' '.$intervento_programmato->codice.' ('.(new Carbon($intervento_programmato->data_richiesta))->diffForHumans().')'; } + echo '
'; } echo '

'; diff --git a/modules/ordini/actions.php b/modules/ordini/actions.php index 34ce5550d..aef304f5d 100755 --- a/modules/ordini/actions.php +++ b/modules/ordini/actions.php @@ -94,6 +94,9 @@ switch (post('op')) { $ordine->idagente = post('idagente'); $ordine->idstatoordine = $idstatoordine; $ordine->idpagamento = $idpagamento; + $ordine->idspedizione = post('idspedizione'); + $ordine->idporto = post('idporto'); + $ordine->idvettore = post('idvettore'); $ordine->idsede = $idsede; $ordine->idconto = post('idconto'); $ordine->idrivalsainps = $idrivalsainps; diff --git a/modules/ordini/edit.php b/modules/ordini/edit.php index c83eeacf6..3a198c45c 100755 --- a/modules/ordini/edit.php +++ b/modules/ordini/edit.php @@ -144,26 +144,77 @@ echo '
-
> +
> {[ "type": "text", "label": "", "name": "numero", "required": 1, "class": "text-center", "value": "$numero$" ]}
-
+
{[ "type": "text", "label": "", "name": "numero_esterno", "class": "text-center", "value": "$numero_esterno$" ]}
-
+
{[ "type": "date", "label": "", "name": "data", "required": 1, "value": "$data$" ]}
-
+
{[ "type": "select", "label": "", "name": "idpagamento", "required": 0, "ajax-source": "pagamenti", "value": "$idpagamento$" ]}
+
-
+
+
+ {[ "type": "select", "label": "", "name": "idspedizione", "placeholder": "-", "values": "query=SELECT `dt_spedizione`.`id`, `dt_spedizione_lang`.`title` as `descrizione`, `esterno` FROM `dt_spedizione` LEFT JOIN `dt_spedizione_lang` ON (`dt_spedizione_lang`.`id_record` = `dt_spedizione`.`id` AND `dt_spedizione_lang`.`id_lang` = id); ?>) ORDER BY `title` ASC", "value": "$idspedizione$" ]} +
+ +
+ {[ "type": "select", "label": "", "name": "idporto", "placeholder": "-", "help": "
  • Franco: pagamento del trasporto a carico del mittente
  • Assegnato: pagamento del trasporto a carico del destinatario
  • '); ?>", "values": "query=SELECT `dt_porto`.`id`, `dt_porto_lang`.`title` as descrizione FROM `dt_porto` LEFT JOIN `dt_porto_lang` ON (`dt_porto`.`id` = `dt_porto_lang`.`id_record` AND `dt_porto_lang`.`id_lang` = id); ?>) ORDER BY `title` ASC", "value": "$idporto$" ]} +
    + +
    + selectOne('dt_spedizione', 'esterno', [ + 'id' => $record['idspedizione'], + ])['esterno']; + ?> + {[ "type": "select", "label": "", "name": "idvettore", "ajax-source": "vettori", "value": "$idvettore$", "disabled": , "required": , "icon-after": "add|first()->id; ?>|tipoanagrafica=Vettore&readonly_tipo=1|btn_idvettore|", "class": "" ]} +
    + +
    {[ "type": "number", "label": "", "name": "sconto_finale", "value": "sconto_finale_percentuale ?: $ordine->sconto_finale; ?>", "icon-after": "choice|untprc|sconto_finale) ? 'PRC' : 'UNT'; ?>", "help": "." ]}
    + + . + */ + +namespace Modules\Pagamenti\API\v1; + +use API\Interfaces\RetrieveInterface; +use API\Resource; +use Modules\Pagamenti\Pagamento; + +class Pagamenti extends Resource implements RetrieveInterface +{ + public function retrieve($request) + { + // Periodo per selezionare interventi + $user = \Auth::user(); + + $table = 'co_pagamenti'; + + $select = [ + 'co_pagamenti.*' + ]; + + $where = []; + + $group = 'co_pagamenti.id'; + + return [ + 'table' => $table, + 'select' => $select, + 'where' => $where, + 'group' => $group, + ]; + } +} \ No newline at end of file diff --git a/package.json b/package.json index 7f448a9bd..5d25102bf 100755 --- a/package.json +++ b/package.json @@ -42,6 +42,7 @@ "pdfjs-viewer-element": "2.5.4", "popper.js": "^1.16.1", "pwstrength-bootstrap": "^3.0.4", + "readmore.js": "^2.0.4", "select2": "^4.0.3", "signature_pad": "^2.3.0", "smartwizard": "^4.2.2", diff --git a/plugins/exportFE/src/FatturaElettronica.php b/plugins/exportFE/src/FatturaElettronica.php index 41436f39e..8f2c9785d 100755 --- a/plugins/exportFE/src/FatturaElettronica.php +++ b/plugins/exportFE/src/FatturaElettronica.php @@ -1666,7 +1666,7 @@ class FatturaElettronica implements \Stringable $co_scadenziario = $database->fetchArray('SELECT * FROM `co_scadenziario` WHERE `iddocumento` = '.prepare($documento['id'])); foreach ($co_scadenziario as $scadenza) { $co_pagamenti = Pagamento::find($scadenza['id_pagamento']); - $banca = ($co_pagamenti && $co_pagamenti->isRiBa()) ? Banca::find($scadenza['id_banca_controparte']) : ($scadenza['id_banca_azienda'] ?? Banca::where('id_anagrafica', $scadenza['idanagrafica'])->where('predefined', 1)->first()); + $banca = Banca::find(($co_pagamenti && $co_pagamenti->isRiBa()) ? $scadenza['id_banca_controparte'] : ($scadenza['id_banca_azienda'])) ?? Banca::where('id_anagrafica', $scadenza['idanagrafica'])->where('predefined', 1)->first(); $pagamento = [ 'ModalitaPagamento' => $co_pagamenti['codice_modalita_pagamento_fe'], 'DataScadenzaPagamento' => $scadenza['scadenza'], diff --git a/plugins/importFE/src/FatturaElettronica.php b/plugins/importFE/src/FatturaElettronica.php index 0dacbbf1d..175e68cc4 100755 --- a/plugins/importFE/src/FatturaElettronica.php +++ b/plugins/importFE/src/FatturaElettronica.php @@ -47,9 +47,9 @@ class FatturaElettronica /** @var Fattura Fattura collegata */ protected $fattura; - public function __construct($name) + public function __construct($name, $directory = null, $plugin = null) { - $this->file = static::getImportDirectory().'/'.$name; + $this->file = static::getImportDirectory($directory ?: 'Fatture di acquisto', $plugin).'/'.$name; if (string_ends_with($name, '.p7m')) { $file = XML::decodeP7M($this->file); @@ -80,18 +80,17 @@ class FatturaElettronica } } - public static function getImportDirectory() + public static function getImportDirectory($name = null, $plugin = null) { - $module = Module::where('name', 'Fatture di acquisto')->first(); + $module = Module::where('name', $name ?: 'Fatture di acquisto')->first(); $plugins = $module->plugins; if (!empty($plugins)) { - $plugin = $plugins->first(fn ($value, $key) => $value->getTranslation('title') == 'Fatturazione Elettronica'); + $plugin = $plugins->first(fn ($value, $key) => $value->getTranslation('title') == ($plugin ?: 'Fatturazione Elettronica')); self::$directory = base_dir().'/'.$plugin->upload_directory; } - return self::$directory; } @@ -106,24 +105,24 @@ class FatturaElettronica return $filename; } - public static function isValid($name) + public static function isValid($name, $directory = null, $plugin = null) { try { - new static($name); + new static($name, $directory, $plugin); return true; } catch (\UnexpectedValueException) { - $file = static::getImportDirectory().'/'.$name; + $file = static::getImportDirectory($directory ?: 'Fatture di acquisto').'/'.$name; delete($file); return false; } } - public static function manage($name) + public static function manage($name, $directory = null, $plugin = null) { try { - $manager = new FatturaOrdinaria($name); + $manager = new FatturaOrdinaria($name, $directory, $plugin); $tipo = $manager->getBody()['DatiGenerali']['DatiGeneraliDocumento']['TipoDocumento']; if ($tipo == 'TD06') { @@ -160,11 +159,11 @@ class FatturaElettronica return array_clean($result); } - public function saveAllegati() + public function saveAllegati($name = null) { $allegati = $this->getAllegati(); - $id_module = Module::where('name', 'Fatture di acquisto')->first()->id; + $id_module = Module::where('name', $name ?: 'Fatture di acquisto')->first()->id; $info = [ 'category' => tr('Fattura Elettronica'), @@ -201,9 +200,9 @@ class FatturaElettronica ])); } - public function findAnagrafica() + public function findAnagrafica($tipo = null) { - $info = $this->getAnagrafe(); + $info = $this->getAnagrafe($tipo); if (!empty($info['partita_iva']) && !empty($info['codice_fiscale'])) { $anagrafica = Anagrafica::where('piva', $info['partita_iva']) @@ -233,9 +232,9 @@ class FatturaElettronica * * @return Anagrafica */ - public function saveAnagrafica($type = 'Fornitore') + public function saveAnagrafica($type = null) { - $anagrafica = $this->findAnagrafica(); + $anagrafica = $this->findAnagrafica($type); if (!empty($anagrafica)) { return $anagrafica; @@ -310,12 +309,12 @@ class FatturaElettronica * * @return Fattura */ - public function saveFattura($id_pagamento, $id_sezionale, $id_tipo, $data_registrazione, $ref_fattura, $is_ritenuta_pagata = false) + public function saveFattura($id_pagamento, $id_sezionale, $id_tipo, $data_registrazione, $ref_fattura, $is_ritenuta_pagata = false, $tipo = null) { $dati_generali = $this->getBody()['DatiGenerali']['DatiGeneraliDocumento']; $data = self::parseDate($dati_generali['Data']); - $fattura = $this->prepareFattura($id_tipo, $data, $data_registrazione, $id_sezionale, $ref_fattura); + $fattura = $this->prepareFattura($id_tipo, $data, $data_registrazione, $id_sezionale, $ref_fattura, $tipo); $this->fattura = $fattura; $numero_esterno = $dati_generali['Numero']; @@ -385,9 +384,9 @@ class FatturaElettronica return $this->fattura; } - public function save($info = []) + public function save($info = [], $tipo = null) { - $this->saveFattura($info['id_pagamento'], $info['id_segment'], $info['id_tipo'], $info['data_registrazione'], $info['ref_fattura'], $info['is_ritenuta_pagata']); + $this->saveFattura($info['id_pagamento'], $info['id_segment'], $info['id_tipo'], $info['data_registrazione'], $info['ref_fattura'], $info['is_ritenuta_pagata'], $tipo); $this->saveRighe($info['articoli'], $info['iva'], $info['conto'], $info['movimentazione'], $info['crea_articoli'], $info['tipo_riga_riferimento'], $info['id_riga_riferimento'], $info['tipo_riga_riferimento_vendita'], $info['id_riga_riferimento_vendita'], $info['update_info'], $info['serial']); @@ -403,9 +402,9 @@ class FatturaElettronica return date('Y-m-d', strtotime((string) $data)); } - protected function prepareFattura($id_tipo, $data, $data_registrazione, $id_sezionale, $ref_fattura) + protected function prepareFattura($id_tipo, $data, $data_registrazione, $id_sezionale, $ref_fattura, $tipo = null) { - $anagrafica = $this->saveAnagrafica(); + $anagrafica = $this->saveAnagrafica($tipo); $tipo = TipoFattura::where('id', $id_tipo)->first(); diff --git a/plugins/importFE/src/FatturaOrdinaria.php b/plugins/importFE/src/FatturaOrdinaria.php index bf298e3fb..d1359535d 100755 --- a/plugins/importFE/src/FatturaOrdinaria.php +++ b/plugins/importFE/src/FatturaOrdinaria.php @@ -37,18 +37,22 @@ use Util\XML; */ class FatturaOrdinaria extends FatturaElettronica { - public function __construct($name) + public function __construct($name, $directory = null, $plugin = null) { - parent::__construct($name); + parent::__construct($name, $directory, $plugin); if ($this->getHeader()['DatiTrasmissione']['FormatoTrasmissione'] == 'FSM10') { throw new \UnexpectedValueException(); } } - public function getAnagrafe() + public function getAnagrafe($tipo = null) { - $dati = $this->getHeader()['CedentePrestatore']; + if ($tipo == 'Cliente') { + $dati = $this->getHeader()['CessionarioCommittente']; + } else { + $dati = $this->getHeader()['CedentePrestatore']; + } $anagrafe = $dati['DatiAnagrafici']; $rea = $dati['IscrizioneREA']; @@ -510,9 +514,9 @@ class FatturaOrdinaria extends FatturaElettronica } } - protected function prepareFattura($id_tipo, $data, $data_registrazione, $id_sezionale, $ref_fattura) + protected function prepareFattura($id_tipo, $data, $data_registrazione, $id_sezionale, $ref_fattura, $tipo = null) { - $fattura = parent::prepareFattura($id_tipo, $data, $data_registrazione, $id_sezionale, $ref_fattura); + $fattura = parent::prepareFattura($id_tipo, $data, $data_registrazione, $id_sezionale, $ref_fattura, $tipo); $database = database(); $righe = $this->getRighe(); diff --git a/plugins/importFE/src/FatturaSemplificata.php b/plugins/importFE/src/FatturaSemplificata.php index e9a1773fe..bb17f5d97 100755 --- a/plugins/importFE/src/FatturaSemplificata.php +++ b/plugins/importFE/src/FatturaSemplificata.php @@ -40,9 +40,13 @@ class FatturaSemplificata extends FatturaElettronica } } - public function getAnagrafe() + public function getAnagrafe($tipo = null) { - $anagrafe = $this->getHeader()['CedentePrestatore']; + if ($tipo == 'Cliente') { + $anagrafe = $this->getHeader()['CessionarioCommittente']; + } else { + $anagrafe = $this->getHeader()['CedentePrestatore']; + } $rea = $anagrafe['IscrizioneREA']; $sede = $anagrafe['Sede']; diff --git a/plugins/importFE/src/Interaction.php b/plugins/importFE/src/Interaction.php index 71eacd3d1..d60982971 100755 --- a/plugins/importFE/src/Interaction.php +++ b/plugins/importFE/src/Interaction.php @@ -34,12 +34,12 @@ class Interaction extends Services return parent::isEnabled() && self::verificaRisorsaAttiva('Fatturazione Elettronica'); } - public static function getInvoiceList() + public static function getInvoiceList($directory = null, $plugin = null) { $list = self::getRemoteList(); // Ricerca fisica - $result = self::getFileList($list); + $result = self::getFileList($list, $directory, $plugin); // Aggiornamento cache hook Cache::where('name', 'Fatture Elettroniche')->first()->set($result); @@ -64,12 +64,12 @@ class Interaction extends Services return $list; } - public static function getFileList($list = []) + public static function getFileList($list = [], $directory = null, $plugin = null) { $names = array_column($list, 'name'); // Ricerca fisica - $directory = FatturaElettronica::getImportDirectory(); + $directory = FatturaElettronica::getImportDirectory($directory, $plugin); $files = glob($directory.'/*.xml*'); foreach ($files as $id => $file) { diff --git a/plugins/importFE/src/Parcella.php b/plugins/importFE/src/Parcella.php index 1ea473b1b..35dbc523c 100755 --- a/plugins/importFE/src/Parcella.php +++ b/plugins/importFE/src/Parcella.php @@ -40,7 +40,7 @@ class Parcella extends FatturaOrdinaria * * @return Fattura */ - public function prepareFattura($id_tipo, $data, $data_registrazione, $id_sezionale, $ref_fattura) + public function prepareFattura($id_tipo, $data, $data_registrazione, $id_sezionale, $ref_fattura, $tipo = null) { if (empty($ref_fattura)) { return parent::prepareFattura($id_tipo, $data, $data_registrazione, $id_sezionale, $ref_fattura); diff --git a/plugins/importFE_ZIP/actions.php b/plugins/importFE_ZIP/actions.php new file mode 100644 index 000000000..a1088087a --- /dev/null +++ b/plugins/importFE_ZIP/actions.php @@ -0,0 +1,769 @@ +. + */ + +include_once __DIR__.'/../../core.php'; +include __DIR__.'/settings.php'; + +use Carbon\Carbon; +use Modules\Anagrafiche\Tipo; +use Modules\DDT\DDT; +use Modules\Fatture\Fattura; +use Modules\Fatture\Gestori\Movimenti as GestoreMovimenti; +use Modules\Fatture\Stato; +use Modules\Ordini\Ordine; +use Modules\Pagamenti\Pagamento; +use Modules\PrimaNota\Mastrino; +use Modules\PrimaNota\Movimento; +use Plugins\ImportFE\FatturaElettronica; +use Plugins\ImportFE\Interaction; +use Util\XML; + +$file = null; +switch (filter('op')) { + case 'list': + $list = Interaction::getRemoteList(); + + echo json_encode($list); + + break; + + case 'prepare': + if (!isset($file)) { + $name = filter('name'); + $file = Interaction::getInvoiceFile($name); + } + + try { + if (!FatturaElettronica::isValid($file, 'Fatture di vendita', 'Importazione FE')) { + echo json_encode([ + 'already' => 1, + ]); + + return; + } + } catch (Exception) { + } + + // Individuazione ID fisico + $files = Interaction::getFileList([], 'Fatture di vendita', 'Importazione FE'); + foreach ($files as $key => $value) { + if ($value['name'] == $file) { + $index = $key; + + break; + } + } + + echo json_encode([ + 'id' => $index + 1, + ]); + + case 'save': + $directory = FatturaElettronica::getImportDirectory('Fatture di vendita', 'Importazione FE'); + + if (!$file) { + $temp_name = $_FILES['blob1']['tmp_name']; + $name = $_FILES['blob1']['name']; + } + + if (setting('Metodo di importazione XML fatture di vendita') == 'Automatico') { + if (string_ends_with($name, '.zip')) { + Util\Zip::extract($temp_name, $directory); + + // Redirect forzato per l'importazione + echo json_encode([ + 'id' => 1, + ]); + + $files_xml = scandir($directory); + } else { + $files_xml[] = $file ?: $files; + } + + $i = 1; + foreach ($files_xml as $xml) { + if ($xml != '.' && $xml != '..' && is_file($directory.'/'.$xml)) { + if (FatturaElettronica::isValid($xml, 'Fatture di vendita', 'Importazione FE')) { + if (string_ends_with($xml, '.p7m')) { + $file = XML::decodeP7M($directory.'/'.$xml); + } else { + $file = XML::readFile($directory.'/'.$xml); + } + + $data = $file['FatturaElettronicaBody']['DatiGenerali']['DatiGeneraliDocumento']['Data']; + + $fattura_pa = FatturaElettronica::manage($xml, 'Fattura di vendita', 'Importazione FE'); + + // Tipo documento + $fattura_body = $fattura_pa->getBody(); + $dati_generali = $fattura_body['DatiGenerali']['DatiGeneraliDocumento']; + $id_tipo = $database->fetchOne('SELECT id FROM co_tipidocumento WHERE dir = "entrata" AND codice_tipo_documento_fe = '.prepare($dati_generali['TipoDocumento']))['id']; + + // Tipo pagamento + $pagamenti = []; + if (isset($fattura_body['DatiPagamento'])) { + $pagamenti = $fattura_body['DatiPagamento']; + $pagamenti = isset($pagamenti[0]) ? $pagamenti : [$pagamenti]; + } + $metodi = isset($pagamenti[0]['DettaglioPagamento']) ? $pagamenti[0]['DettaglioPagamento'] : []; + $metodi = isset($metodi[0]) ? $metodi : [$metodi]; + $codice_modalita_pagamento = $metodi[0]['ModalitaPagamento']; + $pagamento = Pagamento::where('codice_modalita_pagamento_fe', $codice_modalita_pagamento)->where('predefined', '1')->first(); + + if (!empty($pagamento)) { + $id_pagamento = $pagamento->id; + } else { + $id_pagamento = $settings['id_pagamento']; + } + + $fattura_pa->saveFattura($id_pagamento, $settings['id_segment'], $id_tipo, $data, false, false, 'Cliente'); + + $righe = $fattura_pa->getRighe(); + + $articoli = []; + $iva = []; + $conti = []; + + foreach ($righe as $key => $riga) { + $articoli[$key] = 0; + + $aliquota_iva = $riga['AliquotaIVA']; + + $iva[$key] = $settings['iva'][$aliquota_iva]; + + $conti[$key] = $settings['id_conto']; + } + + $fattura_pa->saveRighe($articoli, $iva, $conti); + + $fattura_pa->saveAllegati('Fatture di vendita'); + + $id_record = $fattura_pa->getFattura()->id; + $fattura = Fattura::find($id_record); + $fattura->gestoreMovimenti = new GestoreMovimenti($fattura); + + // Registrazione scadenze + $fattura->registraScadenze(false); + + // Registrazione movimenti + + $fattura->gestoreMovimenti->registra(); + + // Imposto lo stato in GEN + $dbo->query("UPDATE co_documenti SET codice_stato_fe='GEN', data_stato_fe=NOW() WHERE id=".prepare($id_record)); + delete($directory.'/'.$xml); + } + } + } + } else { + if (string_ends_with($name, '.zip')) { + $directory = FatturaElettronica::getImportDirectory('Fatture di vendita', 'Importazione FE'); + + Util\Zip::extract($temp_name, $directory); + + // Redirect forzato per l'importazione + echo json_encode([ + 'id' => 1, + ]); + + exit; + } else { + $files_xml[] = $file ?: $files; + + $i = 1; + foreach ($files_xml as $xml) { + if ($xml != '.' && $xml != '..' && is_file($directory.'/'.$xml)) { + if (FatturaElettronica::isValid($xml, 'Fatture di vendita', 'Importazione FE')) { + if (string_ends_with($xml, '.p7m')) { + $file = XML::decodeP7M($directory.'/'.$xml); + } else { + $file = XML::readFile($directory.'/'.$xml); + } + + $file = FatturaElettronica::store($xml, $content); + } + } + } + } + } + + break; + + case 'delete': + $file_id = get('file_id'); + + $directory = FatturaElettronica::getImportDirectory('Fatture di vendita', 'Importazione FE'); + $files = Interaction::getFileList([], 'Fatture di vendita', 'Importazione FE'); + $file = $files[$file_id]; + + if (!empty($file)) { + delete($directory.'/'.$file['name']); + } + + break; + + case 'download': + $file_id = get('file_id'); + + $directory = FatturaElettronica::getImportDirectory('Fatture di vendita'); + $files = Interaction::getFileList([], 'Fatture di vendita', 'Importazione FE'); + $file = $files[$file_id]; + + if (!empty($file)) { + download($directory.'/'.$file['name']); + } + + break; + + case 'generate': + $filename = post('filename'); + + $info = [ + 'id_pagamento' => post('pagamento'), + 'id_segment' => post('id_segment'), + 'id_tipo' => post('id_tipo'), + 'ref_fattura' => post('ref_fattura'), + 'data_registrazione' => post('data_registrazione'), + 'articoli' => post('articoli'), + 'iva' => post('iva'), + 'conto' => post('conto'), + 'tipo_riga_riferimento' => post('tipo_riga_riferimento'), + 'id_riga_riferimento' => post('id_riga_riferimento'), + 'tipo_riga_riferimento_vendita' => post('tipo_riga_riferimento_vendita'), + 'id_riga_riferimento_vendita' => post('id_riga_riferimento_vendita'), + 'movimentazione' => post('movimentazione'), + 'crea_articoli' => post('crea_articoli'), + 'is_ritenuta_pagata' => post('is_ritenuta_pagata'), + 'update_info' => post('update_info'), + 'serial' => post('flag_crea_seriali') ? post('serial') : [], + ]; + + $fattura_pa = FatturaElettronica::manage($filename, 'Fatture di vendita', 'Importazione FE'); + $id_fattura = $fattura_pa->save($info, 'Cliente'); + $fattura_pa->delete(); + $fattura = Fattura::find($id_fattura); + $id_autofattura = post('autofattura'); + $new_stato = Stato::where('name', 'Pagato')->first()->id; + + if ($fattura->isAutofattura() && !empty($id_autofattura)) { + $autofattura_collegata = Fattura::find($id_autofattura); + $fattura->registraScadenze(true); + $autofattura_collegata->registraScadenze(true); + + $fattura->stato()->associate($new_stato); + $autofattura_collegata->stato()->associate($new_stato); + + $mastrino = Mastrino::build('Compensazione autofattura', $fattura->data, false, true); + + $movimento1 = Movimento::build($mastrino, $fattura->anagrafica->idconto_cliente); + $movimento1->setTotale($fattura->totale, 0); + $movimento1->save(); + + $movimento2 = Movimento::build($mastrino, $fattura->anagrafica->idconto_fornitore); + $movimento2->setTotale(0, $fattura->totale); + $movimento2->save(); + + $fattura->id_autofattura = $id_autofattura; + $fattura->save(); + $autofattura_collegata->save(); + } + + // Aggiorno la tipologia di anagrafica cliente + $anagrafica = $database->fetchOne('SELECT `idanagrafica` FROM `co_documenti` WHERE `co_documenti`.`id`='.prepare($id_fattura)); + $id_tipo = Tipo::where('name', 'Cliente')->first()->id; + $rs_t = $database->fetchOne('SELECT * FROM `an_tipianagrafiche_anagrafiche` WHERE `idtipoanagrafica`='.prepare($id_tipo).' AND `idanagrafica`='.prepare($anagrafica['idanagrafica'])); + + // Se non trovo corrispondenza aggiungo all'anagrafica la tipologia cliente + if (empty($rs_t)) { + $database->query("INSERT INTO `an_tipianagrafiche_anagrafiche` (`idtipoanagrafica`, `idanagrafica`) VALUES ($id_tipo, ".prepare($anagrafica['idanagrafica']).')'); + } + + // Processo il file ricevuto + if (Interaction::isEnabled()) { + $process_result = Interaction::processInvoice($filename); + if ($process_result != '') { + flash()->error($process_result); + redirect(base_path().'/controller.php?id_module='.$id_module); + + return; + } + } + + $files = Interaction::getFileList([], 'Fatture di vendita', 'Importazione FE'); + $file = $files[$id_record - 1]; + + if (get('sequence') == null) { + redirect(base_path().'/editor.php?id_module='.$id_module.'&id_record='.$id_fattura); + } elseif (!empty($file)) { + redirect(base_path().'/editor.php?id_module='.$id_module.'&id_plugin='.$id_plugin.'&id_record='.$id_record.'&sequence=1'); + flash()->info(tr('La fattura numero _NUM_ del _DATA_ (_ANAGRAFICA_) è stata importata correttamente', [ + '_NUM_' => $fattura->numero, + '_DATA_' => dateFormat($fattura->data), + '_ANAGRAFICA_' => $fattura->anagrafica->ragione_sociale, + ])); + } else { + flash()->info(tr('Tutte le fatture salvate sono state importate!')); + redirect(base_path().'/controller.php?id_module='.$id_module); + } + + $record = null; + break; + + case 'process': + $name = get('name'); + + // Processo il file ricevuto + if (Interaction::isEnabled()) { + $process_result = Interaction::processInvoice($name); + if (!empty($process_result)) { + flash()->error($process_result); + } + } + + break; + + case 'compile': + // Gestione del caso di anagrafica inesistente + if (empty($anagrafica)) { + echo json_encode([]); + + return; + } + + $fatture = $anagrafica->fattureAcquisto() + ->contabile() + ->orderBy('created_at', 'DESC') + ->take(10) + ->get(); + + $righe = collect(); + foreach ($fatture as $fattura) { + $righe->push($fattura->righe); + $righe->push($fattura->articoli); + } + $righe = $righe->flatten(); + + // Gestione del caso di anagrafica senza fatture o con fatture senza righe + if ($fatture->isEmpty() || $righe->isEmpty()) { + echo json_encode([]); + + return; + } + + // Ricerca del tipo di documento più utilizzato + $tipi = $fatture->groupBy(fn ($item, $key) => $item->tipo->id)->transform(fn ($item, $key) => $item->count()); + $id_tipo = $tipi->sort()->keys()->last(); + + // Ricerca del conto più utilizzato + $conti = $righe->groupBy(fn ($item, $key) => $item->idconto)->transform(fn ($item, $key) => $item->count()); + $id_conto = $conti->sort()->keys()->last(); + $conto = $database->fetchOne('SELECT * FROM co_pianodeiconti3 WHERE id = '.prepare($id_conto)); + + // Ricerca dell'IVA più utilizzata secondo percentuali + $iva = []; + $percentuali_iva = $righe->groupBy(fn ($item, $key) => $item->aliquota->percentuale); + foreach ($percentuali_iva as $key => $values) { + $aliquote = $values->mapToGroups(fn ($item, $key) => [$item->aliquota->id => $item->aliquota]); + $id_aliquota = $aliquote->map(fn ($item, $key) => $item->count())->sort()->keys()->last(); + $aliquota = $aliquote[$id_aliquota]->first(); + + $iva[$key] = [ + 'id' => $aliquota->id, + 'descrizione' => $aliquota->getTranslation('title'), + ]; + } + + echo json_encode([ + 'id_tipo' => $id_tipo, + 'conto' => [ + 'id' => $conto['id'], + 'descrizione' => $conto['descrizione'], + ], + 'iva' => $iva, + ]); + break; + + case 'riferimenti-automatici': + if (empty($anagrafica)) { + echo json_encode([]); + + return; + } + + $results = []; + + // Dati ordini + $DatiOrdini = XML::forceArray($fattura_pa->getBody()['DatiGenerali']['DatiOrdineAcquisto']); + $DatiDDT = XML::forceArray($fattura_pa->getBody()['DatiGenerali']['DatiDDT']); + + $replaces = ['n ', 'N ', 'n. ', 'N. ', 'nr ', 'NR ', 'nr. ', 'NR. ', 'num ', 'NUM ', 'num. ', 'NUM. ', 'numero ', 'NUMERO ']; + + // Riorganizzazione dati ordini per numero di riga + $dati_ordini = []; + foreach ($DatiOrdini as $dato) { + if (is_array($dato['RiferimentoNumeroLinea'])) { + foreach ($dato['RiferimentoNumeroLinea'] as $dati => $linea) { + foreach ($replaces as $replace) { + if (string_starts_with($dato['IdDocumento'], $replace)) { + $dato['IdDocumento'] = str_replace($replace, '', $dato['IdDocumento']); + break; + } + } + + try { + $dati_ordini[(int) $linea] = [ + 'numero' => $dato['IdDocumento'], + 'anno' => (new Carbon($dato['Data']))->format('Y'), + ]; + } catch (Exception) { + $dati_ordini[(int) $linea] = [ + 'numero' => $dato['IdDocumento'], + ]; + } + } + } else { + foreach ($replaces as $replace) { + if (string_starts_with($dato['IdDocumento'], $replace)) { + $dato['IdDocumento'] = str_replace($replace, '', $dato['IdDocumento']); + break; + } + } + + try { + $dati_ordini[(int) $dato['RiferimentoNumeroLinea']] = [ + 'numero' => $dato['IdDocumento'], + 'anno' => (new Carbon($dato['Data']))->format('Y'), + ]; + } catch (Exception) { + $dati_ordini[(int) $dato['RiferimentoNumeroLinea']] = [ + 'numero' => $dato['IdDocumento'], + ]; + } + } + } + + // Riorganizzazione dati ddt per numero di riga + $dati_ddt = []; + foreach ($DatiDDT as $dato) { + if (is_array($dato['RiferimentoNumeroLinea'])) { + foreach ($dato['RiferimentoNumeroLinea'] as $dati => $linea) { + foreach ($replaces as $replace) { + if (string_starts_with($dato['NumeroDDT'], $replace)) { + $dato['NumeroDDT'] = str_replace($replace, '', $dato['NumeroDDT']); + break; + } + } + + try { + $dati_ddt[(int) $linea] = [ + 'numero' => $dato['NumeroDDT'], + 'anno' => (new Carbon($dato['DataDDT']))->format('Y'), + ]; + } catch (Exception) { + $dati_ddt[(int) $linea] = [ + 'numero' => $dato['NumeroDDT'], + ]; + } + } + } else { + foreach ($replaces as $replace) { + if (string_starts_with($dato['NumeroDDT'], $replace)) { + $dato['NumeroDDT'] = str_replace($replace, '', $dato['NumeroDDT']); + break; + } + } + try { + $dati_ddt[(int) $dato['RiferimentoNumeroLinea']] = [ + 'numero' => $dato['NumeroDDT'], + 'anno' => (new Carbon($dato['DataDDT']))->format('Y'), + ]; + } catch (Exception) { + $dati_ddt[(int) $dato['RiferimentoNumeroLinea']] = [ + 'numero' => $dato['NumeroDDT'], + ]; + } + } + } + + // Iterazione sulle singole righe + $righe = $fattura_pa->getRighe(); + foreach ($righe as $key => $riga) { + // Se la riga è descrittiva non la collego a documenti + if ($riga['PrezzoTotale'] == 0) { + continue; + } + + $collegamento = null; + $match_documento_da_fe = true; + + $numero_linea = (int) $riga['NumeroLinea']; + + // Visualizzazione codici articoli + $codici = $riga['CodiceArticolo'] ?: []; + $codici = !empty($codici) && !isset($codici[0]) ? [$codici] : $codici; + + // Ricerca dell'articolo collegato a ogni codice associato alla riga + $id_articolo = null; + foreach ($codici as $codice) { + if (!empty($anagrafica) && empty($id_articolo)) { + $id_articolo = $database->fetchOne('SELECT id_articolo AS id FROM mg_fornitore_articolo WHERE codice_fornitore = '.prepare($codice['CodiceValore']).' AND id_fornitore = '.prepare($anagrafica->id))['id']; + + if (empty($id_articolo)) { + $id_articolo = $database->fetchOne('SELECT id_articolo AS id FROM mg_fornitore_articolo WHERE REPLACE(codice_fornitore, " ", "") = '.prepare($codice['CodiceValore']).' AND id_fornitore = '.prepare($anagrafica->id))['id']; + } + } + + if (empty($id_articolo)) { + $id_articolo = $database->fetchOne('SELECT `id` FROM `mg_articoli` WHERE `codice` = '.prepare($codice['CodiceValore']).' AND `deleted_at` IS NULL')['id']; + + if (empty($id_articolo)) { + $id_articolo = $database->fetchOne('SELECT `id` FROM `mg_articoli` WHERE REPLACE(`codice`, " ", "") = '.prepare($codice['CodiceValore']).' AND `deleted_at` IS NULL')['id']; + } + + // Controllo se esistono articoli con barcode corrispondente al codice + if (empty($id_articolo)) { + $id_articolo = $database->fetchOne('SELECT `id` FROM `mg_articoli` WHERE `barcode` = '.prepare($codice['CodiceValore']).' AND `deleted_at` IS NULL')['id']; + } + + if (empty($id_articolo)) { + $id_articolo = $database->fetchOne('SELECT `id` FROM `mg_articoli` WHERE REPLACE(`barcode`, " ", "") = '.prepare($codice['CodiceValore']).' AND `deleted_at` IS NULL')['id']; + } + } + + if (!empty($id_articolo)) { + break; + } + } + + // Se nella fattura elettronica è indicato un DDT cerco quel documento specifico + $ddt = $dati_ddt[$numero_linea]; + $query = "SELECT + `dt_righe_ddt`.`id`, + `dt_righe_ddt`.`idddt` AS id_documento, + `dt_righe_ddt`.`is_descrizione`, + `dt_righe_ddt`.`idarticolo`, + `dt_righe_ddt`.`is_sconto`, 'ddt' AS ref, + CONCAT('DDT num. ', IF(`numero_esterno` != '', `numero_esterno`, `numero`), ' del ', DATE_FORMAT(`data`, '%d/%m/%Y'), ' [', `dt_statiddt_lang`.`title`, ']') AS opzione + FROM + `dt_righe_ddt` + INNER JOIN `dt_ddt` ON `dt_ddt`.`id` = `dt_righe_ddt`.`idddt` + INNER JOIN `dt_statiddt` ON `dt_statiddt`.`id` = `dt_ddt`.`idstatoddt` + LEFT JOIN `dt_statiddt_lang` ON `dt_statiddt_lang`.`id_record` = `dt_statiddt`.`id` AND `dt_statiddt_lang`.`id_lang` = ".prepare(Models\Locale::getDefault()->id).' + WHERE + `dt_ddt`.`numero_esterno` = '.prepare($ddt['numero']).' AND + YEAR(`dt_ddt`.`data`) = '.prepare($ddt['anno']).' AND + `dt_ddt`.`idanagrafica` = '.prepare($anagrafica->id).' AND + `dt_righe_ddt`.`qta` > `dt_righe_ddt`.`qta_evasa` AND + |where|'; + + // Ricerca di righe DDT con stesso Articolo + if (!empty($id_articolo)) { + $query_articolo = replace($query, [ + '|where|' => '`dt_righe_ddt`.`idarticolo` = '.prepare($id_articolo), + ]); + + $collegamento = $database->fetchOne($query_articolo); + } + + // Ricerca di righe DDT per stessa descrizione + if (empty($collegamento)) { + $query_descrizione = replace($query, [ + '|where|' => '`dt_righe_ddt`.`descrizione` = '.prepare($riga['Descrizione']), + ]); + + $collegamento = $database->fetchOne($query_descrizione); + } + + // Se nella fattura elettronica NON è indicato un DDT ed è indicato anche un ordine + // cerco per quell'ordine + if (empty($collegamento)) { + $ordine = $dati_ordini[$numero_linea]; + $query = "SELECT + `or_righe_ordini`.`id`, + `or_righe_ordini`.`idordine` AS id_documento, + `or_righe_ordini`.`is_descrizione`, + `or_righe_ordini`.`idarticolo`, + `or_righe_ordini`.`is_sconto`, + 'ordine' AS ref, + CONCAT('Ordine num. ', IF(`numero_esterno` != '', `numero_esterno`, `numero`), ' del ', DATE_FORMAT(`data`, '%d/%m/%Y'), ' [', `or_statiordine_lang`.`title` , ']') AS opzione + FROM `or_righe_ordini` + INNER JOIN `or_ordini` ON `or_ordini`.`id` = `or_righe_ordini`.`idordine` + INNER JOIN `or_statiordine` ON `or_statiordine`.`id` = `or_ordini`.`idstatoordine` + LEFT JOIN `or_statiordine_lang` ON `or_statiordine_lang`.`id_record` = `or_statiordine`.`id` AND `or_statiordine_lang`.`id_lang` = ".prepare(Models\Locale::getDefault()->id).' + WHERE + `or_ordini`.`numero_esterno` = '.prepare($ordine['numero']).' + AND YEAR(`or_ordini`.`data`) = '.prepare($ordine['anno']).' + AND `or_ordini`.`idanagrafica` = '.prepare($anagrafica->id).' + AND `or_righe_ordini`.`qta` > `or_righe_ordini`.`qta_evasa` + AND |where|'; + + // Ricerca di righe Ordine con stesso Articolo + if (!empty($id_articolo)) { + $query_articolo = replace($query, [ + '|where|' => '`or_righe_ordini`.`idarticolo` = '.prepare($id_articolo), + ]); + + $collegamento = $database->fetchOne($query_articolo); + } + + // Ricerca di righe Ordine per stessa descrizione + if (empty($collegamento)) { + $query_descrizione = replace($query, [ + '|where|' => '`or_righe_ordini`.`descrizione` = '.prepare($riga['Descrizione']), + ]); + + $collegamento = $database->fetchOne($query_descrizione); + } + } + + /* + * TENTATIVO 2: ricerca solo per articolo o descrizione su documenti + * non referenziati nella fattura elettronica + */ + // Se non ci sono Ordini o DDT cerco per contenuto + if (empty($collegamento)) { + $match_documento_da_fe = false; + $query = "SELECT + `dt_righe_ddt`.`id`, + `dt_righe_ddt`.`idddt` AS id_documento, + `dt_righe_ddt`.`is_descrizione`, + `dt_righe_ddt`.`idarticolo`, + `dt_righe_ddt`.`is_sconto`, + 'ddt' AS ref, + CONCAT('DDT num. ', IF(`numero_esterno` != '', `numero_esterno`, `numero`), ' del ', DATE_FORMAT(`data`, '%d/%m/%Y'), ' [', `dt_statiddt_lang`.`title`, ']') AS opzione + FROM + `dt_righe_ddt` + INNER JOIN `dt_ddt` ON `dt_ddt`.`id` = `dt_righe_ddt`.`idddt` + INNER JOIN `dt_statiddt` ON `dt_statiddt`.`id` = `dt_ddt`.`idstatoddt` + LEFT JOIN `dt_statiddt_lang` ON (`dt_statiddt_lang`.`id_record` = `dt_statiddt`.`id` AND `dt_statiddt_lang`.`id_lang` = ".prepare(Models\Locale::getDefault()->id).') + INNER JOIN `dt_tipiddt` ON `dt_ddt`.`idtipoddt` = `dt_tipiddt`.`id` + WHERE + `dt_ddt`.`idanagrafica` = '.prepare($anagrafica->id)." AND + |where_ddt| AND + `dt_righe_ddt`.`qta` > `dt_righe_ddt`.`qta_evasa` AND + `dt_statiddt_lang`.`title` != 'Fatturato' AND + `dt_tipiddt`.`dir` = 'entrata' + UNION + SELECT + `or_righe_ordini`.`id`, + `or_righe_ordini`.`idordine` AS id_documento, + `or_righe_ordini`.`is_descrizione`, + `or_righe_ordini`.`idarticolo`, + `or_righe_ordini`.`is_sconto`, + 'ordine' AS ref, + CONCAT('Ordine num. ', IF(`numero_esterno` != '', `numero_esterno`, `numero`), ' del ', DATE_FORMAT(`data`, '%d/%m/%Y'), ' [', (SELECT `descrizione` FROM or_stati`ordine WHERE `id` = `idstatoordine`) , ']') AS opzione + FROM + `or_righe_ordini` + INNER JOIN `or_ordini` ON `or_ordini`.`id` = `or_righe_ordini`.`idordine` + INNER JOIN `or_statiordine` ON `or_statiordine`.`id` = `or_ordini`.`idstatoordine` + LEFT JOIN `or_statiordine_lang` ON (`or_statiordine_lang`.`id_record` = `or_statiordine`.`id` AND `or_statiordine_lang`.`id_lang` = ".prepare(Models\Locale::getDefault()->id).') + INNER JOIN `or_tipiordine` ON `or_ordini`.`idtipoordine` = `or_tipiordine`.`id` + WHERE + `or_ordini`.`idanagrafica` = '.prepare($anagrafica->id)." AND + |where_ordini| AND + `or_righe_ordini`.`qta` > `or_righe_ordini`.`qta_evasa` AND + `or_statiordine_lang` WHERE `title` != 'Fatturato' AND + `or_tipiordine`.`dir` ='entrata'"; + + // Ricerca di righe DDT/Ordine con stesso Articolo + if (!empty($id_articolo)) { + $query_articolo = replace($query, [ + '|where_ddt|' => '`dt_righe_ddt`.`idarticolo` = '.prepare($id_articolo), + '|where_ordini|' => '`or_righe_ordini`.`idarticolo` = '.prepare($id_articolo), + ]); + + $collegamento = $database->fetchOne($query_articolo); + } + + // Ricerca di righe DDT/Ordine per stessa descrizione + if (empty($collegamento)) { + $query_descrizione = replace($query, [ + '|where_ddt|' => '`dt_righe_ddt`.`descrizione` = '.prepare($riga['Descrizione']), + '|where_ordini|' => '`or_righe_ordini`.`descrizione` = '.prepare($riga['Descrizione']), + ]); + + $collegamento = $database->fetchOne($query_descrizione); + } + + // Ricerca di righe DDT/Ordine per stesso importo + if (empty($collegamento)) { + $query_descrizione = replace($query, [ + '|where_ddt|' => '`dt_righe_ddt`.`prezzo_unitario` = '.prepare($riga['PrezzoUnitario']), + '|where_ordini|' => '`or_righe_ordini`.`prezzo_unitario` = '.prepare($riga['PrezzoUnitario']), + ]); + + $collegamento = $database->fetchOne($query_descrizione); + } + } + + if (!empty($collegamento)) { + // Individuazione del documento + $documento = $collegamento['ref'] == 'ddt' ? DDT::find($collegamento['id_documento']) : Ordine::find($collegamento['id_documento']); + + // Individuazione della classe di gestione per la riga + $namespace = $collegamento['ref'] == 'ddt' ? 'Modules\\DDT\\Components\\' : 'Modules\\Ordini\\Components\\'; + if (!empty($collegamento['idarticolo'])) { + $type = 'Articolo'; + } elseif (!empty($collegamento['is_sconto'])) { + $type = 'Sconto'; + } elseif (!empty($collegamento['is_descrizione'])) { + $type = 'Descrizione'; + } else { + $type = 'Riga'; + } + + // Ricerca della riga + $riga = $documento->getRiga($namespace.$type, $collegamento['id']); + $riga_origine = $riga->getOriginalComponent(); + + if (!empty($riga->idarticolo)) { + $desc_conto = $dbo->fetchOne('SELECT CONCAT( co_pianodeiconti2.numero, ".", co_pianodeiconti3.numero, " ", co_pianodeiconti3.descrizione ) AS descrizione FROM co_pianodeiconti3 INNER JOIN co_pianodeiconti2 ON co_pianodeiconti3.idpianodeiconti2=co_pianodeiconti2.id WHERE co_pianodeiconti3.id = '.prepare($riga->articolo->idconto_acquisto))['descrizione']; + } + + // Compilazione dei dati + $results[$key] = [ + 'documento' => [ + 'tipo' => $collegamento['ref'], + 'id' => $collegamento['id_documento'], + 'descrizione' => reference($documento, tr('Origine')), + 'opzione' => $collegamento['opzione'], + 'match_documento_da_fe' => $match_documento_da_fe, + ], + 'riga' => [ + 'tipo' => $riga::class, + 'id' => $riga->id, + 'descrizione' => $riga->descrizione, + 'qta' => $riga->qta, + 'um' => $riga->um, + 'prezzo_unitario' => $riga->prezzo_unitario ?: $riga_origine->prezzo_unitario, + 'id_iva' => $riga->id_iva, + 'iva_percentuale' => $riga->aliquota->percentuale, + 'id_articolo' => $riga->idarticolo, + 'desc_articolo' => str_replace(' ', '_', $riga->articolo->codice.' - '.$riga->articolo->getTranslation('title')), + 'id_conto' => $riga->articolo->idconto_acquisto, + 'desc_conto' => str_replace(' ', '_', $desc_conto), + ], + ]; + } + } + + echo json_encode($results); + + break; +} diff --git a/plugins/importFE_ZIP/buttons.php b/plugins/importFE_ZIP/buttons.php new file mode 100644 index 000000000..7ddb9b5c4 --- /dev/null +++ b/plugins/importFE_ZIP/buttons.php @@ -0,0 +1,124 @@ +. + */ + +echo ' +
    + +
    + +
    + +
    + +'; diff --git a/plugins/importFE_ZIP/edit.php b/plugins/importFE_ZIP/edit.php new file mode 100644 index 000000000..8ac2ecd7b --- /dev/null +++ b/plugins/importFE_ZIP/edit.php @@ -0,0 +1,266 @@ + + function upload1(btn) { + if ($("#blob1").val()) { + swal({ + title: "'.tr('Avviare la procedura?').'", + type: "warning", + showCancelButton: true, + confirmButtonText: "'.tr('Sì').'" + }).then(function (result) { + var restore = buttonLoading(btn); + + $("#upload1").ajaxSubmit({ + url: globals.rootdir + "/actions.php", + data: { + op: "save", + id_module: "'.$id_module.'", + id_plugin: "'.$id_plugin.'", + }, + type: "post", + success: function(data){ + + swal("Caricamento completato!", "", "success"); + + $("#blob1").val(""); + buttonRestore(btn, restore); + }, + error: function(xhr) { + alert("'.tr('Errore').': " + xhr.responseJSON.error.message); + + buttonRestore(btn, restore); + } + }); + }) + } else { + swal({ + title: "'.tr('Selezionare un file!').'", + type: "error", + }) + } + } + '; +} else { + if (!empty($record)) { + include $structure->filepath('generate.php'); + + return; + } + echo ' + '; +} + +echo ' +
    +
    +

    + '.tr('Carica un file ZIP contenente i file XML').' + + + + + +

    +
    +
    +
    +
    + +
    + +
    + +
    +
    +
    +
    '; + +echo ' +
    +
    +

    + '.tr('Fatture da importare').' +

    + +
    + '; + +// Ricerca automatica +if (Interaction::isEnabled()) { + echo ' + '; +} + +echo ' +
    +
    +
    '; + +if (Interaction::isEnabled()) { + echo ' +

    '.tr('Per vedere le fatture da importare utilizza il pulsante _BUTTON_', [ + '_BUTTON_' => '"'.tr('Ricerca fatture di acquisto').'"', + ]).'.

    '; +} else { + include $structure->filepath('list.php'); +} + +echo ' + +
    +
    + +'; diff --git a/plugins/importFE_ZIP/generate.php b/plugins/importFE_ZIP/generate.php new file mode 100644 index 000000000..b15ba1d53 --- /dev/null +++ b/plugins/importFE_ZIP/generate.php @@ -0,0 +1,980 @@ +. + */ + +use Carbon\Carbon; +use Models\Module; +use Modules\Fatture\Fattura; +use Modules\Pagamenti\Pagamento; +use Plugins\ImportFE\FatturaElettronica; +use Util\XML; + +include_once __DIR__.'/../../core.php'; + +echo ' +'; + +$skip_link = $has_next && post('sequence') ? base_path().'/editor.php?id_module='.$id_module.'&id_plugin='.$id_plugin.'&id_record='.($id_record + 1).'&sequence='.get('sequence') : base_path().'/editor.php?id_module='.$id_module; + +if (empty($fattura)) { + if (!empty($error)) { + echo ' +

    '.tr("Errore durante l'apertura della fattura elettronica _NAME_", [ + '_NAME_' => $record['name'], + ]).'.

    '; + } elseif (!empty($imported)) { + echo ' +

    '.tr('La fattura elettronica _NAME_ è già stata importata in passato', [ + '_NAME_' => $record['name'], + ]).'.

    '; + } + + echo ' +
    +
    '; + + if (!empty($imported)) { + echo ' + '; + } + + echo ' + +
    +
    + +'; + + return; +} + +// Cliente +$cliente = $fattura->getAnagrafe('Cliente'); + +$ragione_sociale = $cliente['ragione_sociale'] ?: $cliente['cognome'].' '.$cliente['nome']; +$codice_fiscale = $cliente['codice_fiscale']; +$partita_iva = $cliente['partita_iva']; + +$sede = $cliente['sede']; + +$cap = $sede['cap']; +$citta = $sede['comune']; +$provincia = $sede['provincia']; + +// Dati generali +$fattura_body = $fattura->getBody(); +$dati_generali = $fattura_body['DatiGenerali']['DatiGeneraliDocumento']; + +$tipo_documento = $database->fetchOne('SELECT CONCAT("(", `codice`, ") ", `title`) AS descrizione FROM `fe_tipi_documento` LEFT JOIN `fe_tipi_documento_lang` ON (`fe_tipi_documento_lang`.`id_record` = `fe_tipi_documento`.`codice` AND `fe_tipi_documento_lang`.`id_lang` = '.prepare(Models\Locale::getDefault()->id).') WHERE codice = '.prepare($dati_generali['TipoDocumento']))['descrizione']; + +// Gestione per fattura elettroniche senza pagamento definito +$pagamenti = []; +if (isset($fattura_body['DatiPagamento'])) { + $pagamenti = $fattura_body['DatiPagamento']; + $pagamenti = isset($pagamenti[0]) ? $pagamenti : [$pagamenti]; +} + +$is_autofattura = false; +if (in_array($dati_generali['TipoDocumento'], ['TD16', 'TD17', 'TD18', 'TD19', 'TD20', 'TD21', 'TD28'])) { + $is_autofattura = true; +} + +// Individuazione metodo di pagamento di base +$metodi = $pagamenti[0]['DettaglioPagamento'] ?? []; +$metodi = isset($metodi[0]) ? $metodi : [$metodi]; + +$codice_modalita_pagamento = $metodi[0]['ModalitaPagamento']; + +echo ' +
    + + + +
    +
    +

    + '.$ragione_sociale.' + + '.(empty($anagrafica) ? ''.tr('Nuova anagrafica').'' : ''.Modules::link('Anagrafiche', $anagrafica->id, '', null, '').'').'
    + + + '.(!empty($codice_fiscale) ? (tr('Codice Fiscale').': '.$codice_fiscale.'
    ') : '').' + '.(!empty($partita_iva) ? (tr('Partita IVA').': '.$partita_iva.'
    ') : '').' + '.$cap.' '.$citta.' ('.$provincia.')
    +
    +

    +
    + +
    +

    + '.$dati_generali['Numero'].' + + + '.tr('Visualizza').' + + +
    + '.$tipo_documento.' +
    '.Translator::dateToLocale($dati_generali['Data']).' +
    '.$dati_generali['Divisa'].' +
    +

    +
    '; + +// Blocco DatiPagamento è valorizzato (opzionale) +if (!empty($pagamenti)) { + echo ' +
    +

    '.tr('Pagamento').'

    + +

    '.tr('La fattura importata presenta _NUM_ rat_E_ di pagamento con le seguenti scadenze', [ + '_NUM_' => count($metodi), + '_E_' => ((count($metodi) > 1) ? 'e' : 'a'), + ]).':

    +
      '; + + foreach ($pagamenti as $pagamento) { + $rate = $pagamento['DettaglioPagamento']; + $rate = isset($rate[0]) ? $rate : [$rate]; + + // Scadenze di pagamento + foreach ($rate as $rata) { + $descrizione = !empty($rata['ModalitaPagamento']) ? $database->fetchOne('SELECT `title` FROM `fe_modalita_pagamento` LEFT JOIN `fe_modalita_pagamento_lang` ON (`fe_modalita_pagamento_lang`.`id_record`=`fe_modalita_pagamento`.`codice` AND `fe_modalita_pagamento_lang`.`id_lang`='.prepare(Models\Locale::getDefault()->id).') WHERE `codice` = '.prepare($rata['ModalitaPagamento']))['descrizione'] : ''; + $data = !empty($rata['DataScadenzaPagamento']) ? FatturaElettronica::parseDate($rata['DataScadenzaPagamento']) : ''; + + echo ' +
    1. + '.dateFormat($data).' + '.moneyFormat($rata['ImportoPagamento']).' + ('.$descrizione.') +
    2. '; + } + } + + echo ' +
    +
    '; +} + +echo ' +
    '; + +// Tipo del documento +$query = "SELECT `co_tipidocumento`.`id`, CONCAT('(', `codice_tipo_documento_fe`, ') ', `title`) AS descrizione FROM `co_tipidocumento` LEFT JOIN `co_tipidocumento_lang` ON (`co_tipidocumento_lang`.`id_record` = `co_tipidocumento`.`id` AND `co_tipidocumento_lang`.`id_lang` = ".prepare(Models\Locale::getDefault()->id).") WHERE `dir` = 'entrata'"; +$query_tipo = $query.' AND `codice_tipo_documento_fe` = '.prepare($dati_generali['TipoDocumento']); +$numero_tipo = $database->fetchNum($query_tipo); +if (!empty($numero_tipo)) { + $query = $query_tipo; +} + +$id_tipodocumento = $database->fetchOne($query_tipo)['id']; + +echo ' +
    +
    + {[ "type": "select", "label": "'.tr('Tipo fattura').'", "name": "id_tipo", "required": 1, "values": "query='.$query.'", "value": "'.($numero_tipo != 1 ? $id_tipodocumento : '').'" ]} +
    '; + +// Sezionale +$id_segment = $database->table('co_tipidocumento')->where('id', '=', $id_tipodocumento)->value('id_segment'); + +echo ' +
    + {[ "type": "select", "label": "'.tr('Sezionale').'", "name": "id_segment", "required": 1, "ajax-source": "segmenti", "select-options": '.json_encode(['id_module' => $id_module, 'is_fiscale' => 1, 'is_sezionale' => 1, 'for_fe' => 1]).', "value": "'.$id_segment.'" ]} +
    '; + +// Data di registrazione +$data_registrazione = get('data_registrazione'); +$data_registrazione = new Carbon($data_registrazione); +echo ' +
    + {[ "type": "date", "label": "'.tr('Data di registrazione').'", "name": "data_registrazione", "required": 1, "value": "'.($data_registrazione ?: $dati_generali['Data']).'", "max-date": "-now-", "min-date": "'.$dati_generali['Data'].'" ]} +
    '; + +if (!empty($anagrafica)) { + $query = "SELECT + `co_documenti`.`id`, + CONCAT('Fattura num. ', `co_documenti`.`numero_esterno`, ' del ', DATE_FORMAT(`co_documenti`.`data`, '%d/%m/%Y')) AS descrizione + FROM `co_documenti` + INNER JOIN `co_tipidocumento` ON `co_tipidocumento`.`id` = `co_documenti`.`idtipodocumento` + WHERE + `co_tipidocumento`.`dir` = 'entrata' AND + (`co_documenti`.`data` BETWEEN NOW() - INTERVAL 1 YEAR AND NOW()) AND + `co_documenti`.`idstatodocumento` IN (SELECT `id_record` FROM `co_statidocumento_lang` WHERE `title` != 'Bozza') AND + `co_documenti`.`idanagrafica` = ".prepare($anagrafica->id); + + // Riferimenti ad altre fatture + if (in_array($dati_generali['TipoDocumento'], ['TD04', 'TD05'])) { + echo ' +
    + {[ "type": "select", "label": "'.tr('Fattura collegata').'", "name": "ref_fattura", "required": 0, "values": "query='.$query.'" ]} +
    '; + } elseif ($dati_generali['TipoDocumento'] == 'TD06') { + $query .= 'AND `co_documenti`.`id_segment` = (SELECT `zz_segments`.`id` FROM `zz_segments` LEFT JOIN `zz_segments_lang` ON (`zz_segments_lang`.`id_record` = `zz_segments`.`id` AND `zz_segments_lang`.`id_lang` = '.prepare(Models\Locale::getDefault()->id).") WHERE `title` = 'Fatture pro-forma' AND `id_module` = ".prepare($id_module).')'; + + echo ' +
    + {[ "type": "select", "label": "'.tr('Collega a fattura pro-forma').'", "name": "ref_fattura", "values": "query='.$query.'" ]} +
    '; + } elseif ($is_autofattura) { + $query = "SELECT + `co_documenti`.`id`, + CONCAT('Fattura num. ', `co_documenti`.`numero_esterno`, ' del ', DATE_FORMAT(`co_documenti`.`data`, '%d/%m/%Y')) AS descrizione + FROM `co_documenti` + INNER JOIN `co_tipidocumento` ON `co_tipidocumento`.`id` = `co_documenti`.`idtipodocumento` + WHERE + `co_tipidocumento`.`dir` = 'entrata' AND + `co_tipidocumento`.`codice_tipo_documento_fe` IN('TD16', 'TD17', 'TD18', 'TD19', 'TD20', 'TD21', 'TD28') AND + (`co_documenti`.`data` BETWEEN NOW() - INTERVAL 1 YEAR AND NOW()) AND + `co_documenti`.`idstatodocumento` IN (SELECT `id_record` FROM `co_statidocumento_lang` WHERE `title` != 'Bozza') AND + `co_documenti`.`idanagrafica` = ".prepare($anagrafica->id); + + $autofattura_collegata = Fattura::where('progressivo_invio', '=', $fattura->getHeader()['DatiTrasmissione']['ProgressivoInvio'])->first(); + + echo ' +
    + {[ "type": "select", "label": "'.tr('Autofattura collegata').'", "name": "autofattura", "values": "query='.$query.'", "value": "'.$autofattura_collegata->id.'" ]} +
    '; + } +} + +echo ' +
    '; + +// Pagamento +$pagamento = Pagamento::where('codice_modalita_pagamento_fe', $codice_modalita_pagamento)->where('predefined', '1')->first(); +echo ' +
    +
    + + + {[ "type": "select", "label": "'.tr('Pagamento').'", "name": "pagamento", "required": 1, "ajax-source": "pagamenti", "select-options": '.json_encode(['codice_modalita_pagamento_fe' => $codice_modalita_pagamento]).', "value": "'.$pagamento->id.'" ]} +
    '; + +// Movimentazioni +echo ' +
    + {[ "type": "checkbox", "label": "'.tr('Movimenta gli articoli').'", "name": "movimentazione", "value": "'.setting('Movimenta magazzino da fatture di acquisto').'" ]} +
    + +
    + {[ "type": "checkbox", "label": "'.tr('Creazione automatica articoli').'", "name": "flag_crea_articoli", "value": 0, "help": "'.tr('Nel caso di righe con almeno un nodo \'CodiceArticolo\', il gestionale procede alla creazione dell\'articolo se la riga non risulta assegnata manualmente').'." ]} +
    + +
    + {[ "type": "checkbox", "label": "'.tr('Creazione seriali').'", "name": "flag_crea_seriali", "value": "'.setting('Creazione seriali in import FE').'", "help": "'.tr('Nel caso di righe contenenti serial number, il gestionale procede alla loro registrazione. Controllare che l\'XML della fattura di acquisto contenga il nodo \'CodiceTipo\' valorizzato con \'serial\' o \'Serial\' ').'." ]} +
    '; + +$ritenuta = $dati_generali['DatiRitenuta']; + +if (!empty($ritenuta)) { + echo ' +
    + {[ "type": "checkbox", "label": "'.tr('Ritenuta pagata dal cliente').'", "name": "is_ritenuta_pagata", "value": 0, "help": "'.tr('Attivare se la ritenuta è stata pagata dal cliente').'" ]} +
    '; +} +echo ' +
    '; + +// Righe +if (setting('Aggiorna info di acquisto') == 'Non aggiornare') { + $update_info = 'update_not'; +} elseif (setting('Aggiorna info di acquisto') == 'Aggiorna prezzo di listino') { + $update_info = 'update_price'; +} else { + $update_info = 'update_all'; +} + +$righe = $fattura->getRighe(); +if (!empty($righe)) { + echo ' +

    + '.tr('Righe').' + + +
    +

    + +
    + + + + + + + + + + + '; + + // Dati ordini + $DatiOrdini = XML::forceArray($fattura->getBody()['DatiGenerali']['DatiOrdineAcquisto']); + $DatiDDT = XML::forceArray($fattura->getBody()['DatiGenerali']['DatiDDT']); + + // Riorganizzazione dati ordini per numero di riga + $dati_ordini = []; + foreach ($DatiOrdini as $dato) { + foreach ($dato['RiferimentoNumeroLinea'] as $dati => $linea) { + $dati_ordini[(int) $linea] = [ + 'numero' => $dato['IdDocumento'], + 'data' => (new Carbon($dato['Data']))->format('d/m/Y'), + ]; + } + } + + // Riorganizzazione dati ordini per numero di riga + $dati_ddt = []; + foreach ($DatiDDT as $dato) { + foreach ($dato['RiferimentoNumeroLinea'] as $dati => $linea) { + $dati_ddt[(int) $linea] = [ + 'numero' => $dato['NumeroDDT'], + 'data' => (new Carbon($dato['DataDDT']))->format('d/m/Y'), + ]; + } + } + + foreach ($righe as $key => $riga) { + $query = "SELECT `co_iva`.`id`, IF(`codice` IS NULL, `title`, CONCAT(`codice`, ' - ', `title`)) AS descrizione FROM `co_iva` LEFT JOIN `co_iva_lang` ON (`co_iva`.`id` = `co_iva_lang`.`id_record` AND `co_iva_lang`.`id_lang` = ".prepare(Models\Locale::getDefault()->id).') WHERE `deleted_at` IS NULL AND `percentuale` = '.prepare($riga['AliquotaIVA']); + $start_query = $query; + + if (!empty($riga['Natura'])) { + $query .= ' AND `codice_natura_fe` = '.prepare($riga['Natura']); + + // Fallback per natura iva mancante + if (empty($dbo->fetchArray($query))) { + $query = $start_query; + } + } + + $query .= ' ORDER BY `descrizione` ASC'; + + // Visualizzazione codici articoli + $codici = $riga['CodiceArticolo'] ?: []; + $codici = !empty($codici) && !isset($codici[0]) ? [$codici] : $codici; + + $codici_articoli = []; + $serial = []; + $i = 0; + foreach ($codici as $codice) { + $codici_articoli[] = (($i == 0) ? '' : '').$codice['CodiceValore'].' ('.$codice['CodiceTipo'].')'.(($i == 0) ? '' : ''); + if (str_contains((string) $codice['CodiceTipo'], 'serial') || str_contains((string) $codice['CodiceTipo'], 'Serial')) { + $serial[] = $codice['CodiceValore']; + } + ++$i; + } + + // Individuazione articolo con codice relativo + $id_articolo = null; + // Prendo il codice articolo dal primo nodo CodiceValore che trovo + $codice_principale = $codici[0]['CodiceValore']; + if (!empty($codice_principale)) { + if (empty($id_articolo)) { + $id_articolo = $database->fetchOne('SELECT `id` FROM `mg_articoli` WHERE `codice` = '.prepare($codice_principale))['id']; + if (empty($id_articolo)) { + $id_articolo = $database->fetchOne('SELECT `id` FROM `mg_articoli` WHERE REPLACE(`codice`, " ", "") = '.prepare($codice_principale))['id']; + } + } + + $idconto_acquisto = $database->fetchOne('SELECT `idconto_acquisto` FROM `mg_articoli` WHERE `id` = '.prepare($id_articolo))['idconto_acquisto']; + } + + $idconto_acquisto = $is_autofattura ? setting('Conto per autofattura') : $idconto_acquisto; + $qta = $riga['Quantita']; + $um = $riga['UnitaMisura']; + $prezzo_unitario = $riga['PrezzoUnitario'] ?: $riga['Importo']; + $is_descrizione = empty((float) $riga['Quantita']) && empty((float) $prezzo_unitario); + + $sconto_unitario = 0; + $sconti = $riga['ScontoMaggiorazione'] ?: 0; + if (!empty($sconti)) { + $tot_sconto_calcolato = 0; + $sconto_unitario = 0; + $sconti = $sconti[0] ? $sconti : [$sconti]; + + // Determina il tipo di sconto in caso di sconti misti UNT e PRC + foreach ($sconti as $sconto) { + $tipo_sconto = !empty($sconto['Importo']) ? '€' : '%'; + if (!empty($tipo) && $tipo_sconto != $tipo) { + $tipo = '€'; + } else { + $tipo = $tipo_sconto; + } + } + + foreach ($sconti as $sconto) { + $unitario = $sconto['Importo'] ?: $sconto['Percentuale']; + + // Sconto o Maggiorazione + $sconto_riga = ($sconto['Tipo'] == 'SC') ? $unitario : -$unitario; + + $tipo_sconto = !empty($sconto['Importo']) ? '€' : '%'; + if ($tipo_sconto == '%') { + $sconto_calcolato = calcola_sconto([ + 'sconto' => $sconto_riga, + 'prezzo' => $sconto_unitario ? $prezzo_unitario - ($tot_sconto_calcolato / ($qta ?: 1)) : $prezzo_unitario, + 'tipo' => 'PRC', + 'qta' => $qta, + ]); + + if ($tipo == '%') { + $tot_sconto = ($prezzo_unitario * $qta != 0 ? $sconto_calcolato * 100 / ($prezzo_unitario * $qta) : 0); + } else { + $tot_sconto = $sconto_calcolato; + } + } else { + $tot_sconto = $sconto_riga; + } + + $tot_sconto_calcolato += $sconto_calcolato; + $sconto_unitario += $tot_sconto; + } + } + + $riferimento_fe = ''; + + if ($dati_ddt[(int) $riga['NumeroLinea']]) { + $riferimento_fe = tr('DDT _NUMERO_ del _DATA_', + [ + '_NUMERO_' => $dati_ddt[(int) $riga['NumeroLinea']]['numero'], + '_DATA_' => $dati_ddt[(int) $riga['NumeroLinea']]['data'], + ]); + } + + echo ' + + + + + + + + + '; + + if (!$is_descrizione) { + echo ' + + + '; + } else { + echo ' + + + + + + + + + + + + + + + '; + } + } + + echo ' + +
    '.tr('Descrizione').''.tr('Quantità').''.tr('Prezzo unitario').''.tr('Aliquota').'
    + '.(empty($codice_principale) ? '' : '').' +
    + '.$riferimento_fe.' + + + '.$riga['Descrizione'].'
    + + '.(!empty($codici_articoli) ? ''.implode(', ', $codici_articoli).'
    ' : '').' + + +
    + '.numberFormat($qta, 'qta').' '.$um.' + + + '.moneyFormat($prezzo_unitario); + if (abs($sconto_unitario) > 0) { + $text = ($prezzo_unitario >= 0 && $sconto_unitario > 0) || ($prezzo_unitario < 0 && $sconto_unitario < 0) ? tr('sconto _TOT_ _TYPE_', ['_TOT_' => Translator::numberToLocale(abs($sconto_unitario)), '_TYPE_' => $tipo]) : tr('maggiorazione _TOT__TYPE_', ['_TOT_' => Translator::numberToLocale(abs($sconto_unitario)), '_TYPE_' => $tipo]); + echo ' +
    '.$text.''; + } + echo ' + +
    + '.replace('_VALUE_ _DESC_', [ + '_VALUE_' => empty($riga['Natura']) ? numberFormat($riga['AliquotaIVA'], 0).'%' : $riga['Natura'], + '_DESC_' => $riga['RiferimentoNormativo'] ? ' - '.$riga['RiferimentoNormativo'] : '', + ]).' + +
    + '.$riga['AliquotaIVA'].' + + + + + + + + + + + + +
    +
    +
    +
    + {["type": "select", "name": "articoli['.$key.']", "ajax-source": "articoli", "select-options": '.json_encode(['permetti_movimento_a_zero' => 1, 'dir' => 'entrata', 'idanagrafica' => $anagrafica ? $anagrafica->id : '']).', "icon-after": "add|'.Module::where('name', 'Articoli')->first()->id.'|codice='.($codice_principale ? urlencode((string) $codice_principale) : '').'&descrizione='.($riga['Descrizione'] ? urlencode((string) $riga['Descrizione']) : '').'&prezzo_acquisto='.($riga['PrezzoUnitario'] ? urlencode((string) $riga['PrezzoUnitario']) : '').'", "value": "'.$id_articolo.'", "label": "'.tr('Articolo').'","extra": "data-id=\''.$key.'\'" ]} +
    + +
    + {[ "type": "select", "name": "conto['.$key.']", "id": "conto-'.$key.'", "ajax-source": "conti-acquisti", "required": 1, "label": "'.tr('Conto acquisti').'", "value": "'.$idconto_acquisto.'" ]} +
    + +
    + {[ "type": "select", "name": "iva['.$key.']", "values": '.json_encode('query='.$query).', "required": 1, "label": "'.tr('Aliquota IVA').'" ]} +
    + +
    +
    + +
    +
    +
    +
    +
    +
    + {[ "type": "select", "name": "selezione_riferimento['.$key.']", "ajax-source": "riferimenti-fe", "select-options": '.json_encode(['id_anagrafica' => $anagrafica ? $anagrafica->id : '']).', "label": "'.tr('Riferimento acquisto').'", "icon-after": '.json_encode('').', "help": "'.tr('Articoli contenuti in Ordini o DDT del cliente').'" ]} +
    + +
    + {[ "type": "select", "name": "selezione_riferimento_vendita['.$key.']", "ajax-source": "riferimenti-vendita-fe", "select-options": '.json_encode(['id_articolo' => $id_articolo]).', "label": "'.tr('Riferimento vendita').'", "icon-after": '.json_encode('').', "help": "'.tr('Articoli contenuti in Ordini Cliente').'" ]} +
    + +
    + {[ "type": "select", "name": "update_info['.$key.']", "values": "list=\"update_not\":\"Nessuna operazione\", \"update_price\":\"Crea listino del cliente (se non presente) e aggiorna il prezzo di acquisto\", \"update_all\":\"Crea listino del cliente (se non presente) aggiorna prezzo di acquisto e imposta cliente come predefinito\"", "label": "'.tr('Aggiorna informazioni di acquisto').'", "value": "'.$update_info.'", "help": "'.tr('Creazione automatica articolo deve essere attiva o l\'articolo deve essere selezionato affinché questa impostazione abbia effetto').'.", "readonly": "'.(empty($codice_principale) ? 1 : 0).'" ]} +
    +
    + +
    '; + if (setting('Creazione seriali in import FE') && $serial) { + for ($i = 0; $i < $qta; ++$i) { + echo ' +
    + {[ "type": "text", "label": "'.tr('Serial').'", "name": "serial['.$key.'][]", "value": "'.$serial[$i].'" ]} +
    '; + } + } + echo ' +
    +
    +
    +
    +
    '; + + echo ' + '; +} else { + echo ' +

    '.tr('Non ci sono righe nella fattura').'.

    '; +} + +echo ' +
    +
    + + '.tr('Salta fattura').' + + + +
    +
    +
    + +'; diff --git a/plugins/importFE_ZIP/init.php b/plugins/importFE_ZIP/init.php new file mode 100644 index 000000000..e7cc6f95b --- /dev/null +++ b/plugins/importFE_ZIP/init.php @@ -0,0 +1,48 @@ +. + */ + +include_once __DIR__.'/../../core.php'; + +use Plugins\ImportFE\FatturaElettronica; +use Plugins\ImportFE\Interaction; + +if (!empty($id_record)) { + $files = Interaction::getFileList([], 'Fatture di vendita', 'Importazione FE'); + $record = $files[$id_record - 1] ?? null; + + $has_next = !empty($files[$id_record]); + + try { + $fattura = FatturaElettronica::manage($record['name'] ?? '', 'Fatture di vendita', 'Importazione FE'); + $anagrafica = $fattura->findAnagrafica('Cliente'); + } catch (UnexpectedValueException) { + $imported = true; + } catch (Exception) { + $error = true; + } + + // Rimozione .p7m dal nome del file (causa eventuale estrazione da ZIP) + $record['name'] = preg_replace('/(.+)\.p7m$/i', '$1', $record['name'] ?? ''); + + if (empty($record)) { + flash()->warning(tr('Nessuna fattura da importare!')); + + redirect(base_path().'/controller.php?id_module='.$id_module); + } +} diff --git a/plugins/importFE_ZIP/list.php b/plugins/importFE_ZIP/list.php new file mode 100644 index 000000000..950707232 --- /dev/null +++ b/plugins/importFE_ZIP/list.php @@ -0,0 +1,239 @@ +. + */ + +include_once __DIR__.'/../../core.php'; + +use Carbon\Carbon; +use Plugins\ImportFE\Interaction; +use Util\XML; + +$list = Interaction::getInvoiceList('Fatture di vendita', 'Importazione FE'); + +$directory = Plugins\ImportFE\FatturaElettronica::getImportDirectory('Fatture di vendita', 'Importazione FE'); + +if (!empty($list)) { + echo ' + + + + + + + + + + + '; + + foreach ($list as $element) { + $name = $element['name']; + $file = XML::readFile($directory.'/'.$name); + $date = $file['FatturaElettronicaBody']['DatiGenerali']['DatiGeneraliDocumento']['Data']; + + echo ' + '; + + if (!empty($element['file'])) { + echo ' + + + + + + + + + + + + + + '; + } + + echo ' + +
    '.tr('Descrizione').''.tr('Cliente').''.tr('Data di registrazione').''.tr('Totale imponibile').'#
    +

    '.$name.'

    +
    --- + '; + } else { + $date = new DateTime($element['date']); + + $descrizione = ''; + if ($element['type'] == 'TD01') { + $descrizione = tr('Fattura num. _NUM_ del _DATE_', [ + '_NUM_' => $element['number'], + '_DATE_' => dateFormat($date), + ]); + } elseif ($element['type'] == 'TD04') { + $descrizione = tr('Nota di credito num. _NUM_ del _DATE_', [ + '_NUM_' => $element['number'], + '_DATE_' => dateFormat($date), + ]); + } elseif ($element['type'] == 'TD05') { + $descrizione = tr('Nota di debito num. _NUM_ del _DATE_', [ + '_NUM_' => $element['number'], + '_DATE_' => dateFormat($date), + ]); + } elseif ($element['type'] == 'TD06') { + $descrizione = tr('Parcella num. _NUM_ del _DATE_', [ + '_NUM_' => $element['number'], + '_DATE_' => dateFormat($date), + ]); + } + + $date = new DateTime($element['date']); + $date = $date->format('Y-m-d'); + + echo ' + + '.$descrizione.' ['.$name.'] + '.$element['sender'].''.dateFormat(new Carbon($element['date_sent'])).''.moneyFormat($element['amount']).' + '; + } + + if (file_exists($directory.'/'.$name)) { + echo ' + '; + } + + echo ' + + +
    '; +} else { + echo ' +

    '.tr('Nessuna fattura da importare').'.

    '; +} + +echo ' +'; diff --git a/plugins/importFE_ZIP/riferimento.php b/plugins/importFE_ZIP/riferimento.php new file mode 100644 index 000000000..9b8d3c6c4 --- /dev/null +++ b/plugins/importFE_ZIP/riferimento.php @@ -0,0 +1,145 @@ +. + */ + +use Modules\DDT\DDT; +use Modules\Ordini\Ordine; + +include_once __DIR__.'/../../core.php'; +include_once __DIR__.'/init.php'; + +$direzione = 'entrata'; +$id_riga = get('id_riga'); +$qta = get('qta'); +$descrizione = get('descrizione'); +$prezzo_unitario = get('prezzo_unitario'); + +$id_documento = get('id_documento'); +$tipo_documento = get('tipo_documento'); +$dir = get('dir'); +if ($tipo_documento == 'ordine') { + $documento = Ordine::find($id_documento); + $righe_utilizzate = get('righe_ordini'); +} else { + $documento = DDT::find($id_documento); + $righe_utilizzate = get('righe_ddt'); +} + +echo ' +
    +
    + Riga: '.$descrizione.' +
    +
    + Quantità: '.$qta.' - Prezzo unitario: '.number_format($prezzo_unitario, 2, ',', '.').' +
    +
    +
    + + + + + + + + + '; + +$id_riferimento = get('id_riferimento'); +$righe = $documento->getRighe(); +foreach ($righe as $riga) { + $qta_rimanente = $riga->qta_rimanente - (float) $righe_utilizzate[$riga->id]; + $riga_origine = $riga->getOriginalComponent(); + + if (!empty($riga->idarticolo)) { + $desc_conto = $dbo->fetchOne('SELECT CONCAT( co_pianodeiconti2.numero, ".", co_pianodeiconti3.numero, " ", co_pianodeiconti3.descrizione ) AS descrizione FROM co_pianodeiconti3 INNER JOIN co_pianodeiconti2 ON co_pianodeiconti3.idpianodeiconti2=co_pianodeiconti2.id WHERE co_pianodeiconti3.id = '.prepare($riga->articolo->idconto_acquisto))['descrizione']; + } + + $dettagli = [ + 'tipo' => $riga::class, + 'id' => $riga->id, + 'descrizione' => str_replace(' ', '_', $riga->descrizione), + 'qta' => $riga->qta, + 'um' => $riga->um, + 'prezzo_unitario' => $riga->prezzo_unitario ?: $riga_origine->prezzo_unitario, + 'id_iva' => $riga->id_iva, + 'iva_percentuale' => $riga->aliquota->percentuale, + 'id_articolo' => $riga->idarticolo, + 'desc_articolo' => str_replace(' ', '_', $riga->articolo->codice.' - '.$riga->articolo->getTranslation('title')), + 'id_conto' => $riga->articolo->idconto_acquisto, + 'desc_conto' => $desc_conto ? str_replace(' ', '_', $desc_conto) : null, + ]; + + echo ' + id ? 'class="success"' : '').' data-dettagli='.json_encode($dettagli).'> + + + + + '; +} + +echo ' + +
    '.tr('Descrizione').' + '.tr('Q.tà').' + '.tr('Prezzo unitario').'#
    '.(!empty($riga->codice) ? $riga->codice.' - ' : '').$riga->descrizione.''.numberFormat($qta_rimanente, 'qta').' / '.numberFormat($riga->qta, 'qta').' '.$riga->um.''.moneyFormat($riga->prezzo_unitario_corrente).''; + + if ($qta_rimanente >= $qta || !empty(setting('Permetti il superamento della soglia quantità dei documenti di origine'))) { + echo ' + '; + } + + echo ' +
    + + + +'; diff --git a/plugins/importFE_ZIP/settings.php b/plugins/importFE_ZIP/settings.php new file mode 100644 index 000000000..c2c74e4e6 --- /dev/null +++ b/plugins/importFE_ZIP/settings.php @@ -0,0 +1,16 @@ + 2, + 'id_segment' => 1, + 'id_pagamento' => 113, + 'iva' => [ + '22.00' => 171, + '10.00' => 167, + '5.00' => 164, + '4.00' => 163, + '' => 254, + '0.00' => 254, + ], + 'id_conto' => 94, +]; diff --git a/plugins/importFE_ZIP/view.php b/plugins/importFE_ZIP/view.php new file mode 100644 index 000000000..dc92cb706 --- /dev/null +++ b/plugins/importFE_ZIP/view.php @@ -0,0 +1,40 @@ +. + */ + +include_once __DIR__.'/../../core.php'; +use Plugins\ImportFE\FatturaElettronica; + +$directory = FatturaElettronica::getImportDirectory('Fatture di vendita', 'Importazione FE'); +$filename = get('filename'); + +$content = file_get_contents($directory.'/'.$filename); + +// XML +$xml = new DOMDocument(); +$xml->loadXML($content); + +// XSL +$xsl = new DOMDocument(); +$xsl->load(base_dir().'/plugins/xml/asso-invoice.xsl'); + +// XSLT +$xslt = new XSLTProcessor(); +$xslt->importStylesheet($xsl); + +echo $xslt->transformToXML($xml); diff --git a/plugins/presentazioni_bancarie/src/Gestore.php b/plugins/presentazioni_bancarie/src/Gestore.php index 9e2a98658..b87c7977b 100644 --- a/plugins/presentazioni_bancarie/src/Gestore.php +++ b/plugins/presentazioni_bancarie/src/Gestore.php @@ -191,7 +191,7 @@ class Gestore return false; } - public function aggiungiRiBa(int $identifier, Anagrafica $controparte, Banca $banca_controparte, string $descrizione, int $totale, \DateTime $data_prevista) + public function aggiungiRiBa(int $identifier, Anagrafica $controparte, Banca $banca_controparte, string $descrizione, float $totale, \DateTime $data_prevista) { $data_scadenza = $data_prevista->format('dmy'); diff --git a/src/API/App/v1/AllegatiInterventi.php b/src/API/App/v1/AllegatiInterventi.php index 2f2eb8b00..056f293f9 100644 --- a/src/API/App/v1/AllegatiInterventi.php +++ b/src/API/App/v1/AllegatiInterventi.php @@ -105,9 +105,6 @@ class AllegatiInterventi extends AppResource 'id_record' => $data['id_intervento'], ], $data['nome'], $data['categoria']); - // Chiusura e rimozione del file temporaneo - delete($file); - return [ 'id' => $upload->id, 'tipo' => $upload->extension, diff --git a/src/API/App/v1/Impianti.php b/src/API/App/v1/Impianti.php index 7131be7d0..1fc49ac99 100755 --- a/src/API/App/v1/Impianti.php +++ b/src/API/App/v1/Impianti.php @@ -28,7 +28,6 @@ class Impianti extends AppResource { public function getCleanupData($last_sync_at) { - // TODO: modificare introducendo deleted_at su my_impianti return database() ->table('zz_operations') ->select('zz_operations.id_record') diff --git a/src/API/App/v1/SessioniInterventi.php b/src/API/App/v1/SessioniInterventi.php index b65c56338..66c2eef28 100644 --- a/src/API/App/v1/SessioniInterventi.php +++ b/src/API/App/v1/SessioniInterventi.php @@ -29,7 +29,6 @@ class SessioniInterventi extends AppResource { public function getCleanupData($last_sync_at) { - // TODO: modificare introducendo deleted_at su sessioni return database() ->table('zz_operations') ->select('zz_operations.id_record') diff --git a/src/Util/Autofill.php b/src/Util/Autofill.php index 92d9f8c88..db4162afb 100755 --- a/src/Util/Autofill.php +++ b/src/Util/Autofill.php @@ -49,12 +49,10 @@ class Autofill { $count = ceil(strlen((string) $text) / $this->char_number); - // Ricerca dei caratteri a capo - preg_match_all("/(\r\n|\r|\n)/", (string) $text, $matches); + // Ricerca dei caratteri a capo e
    + preg_match_all("/(\r\n|\r|\n|)/", (string) $text, $matches); $count += count($matches[0]); - if ($small) { - $count *= 1.5; - } elseif ($count != 1) { + if ($small || $count != 1) { $count /= 1.538461538; } diff --git a/templates/ddt/body.php b/templates/ddt/body.php index ec57e9261..3d63466ef 100755 --- a/templates/ddt/body.php +++ b/templates/ddt/body.php @@ -22,9 +22,9 @@ include_once __DIR__.'/../../core.php'; $prezzi_ivati = setting('Utilizza prezzi di vendita comprensivi di IVA'); // Creazione righe fantasma -$autofill = new Util\Autofill($options['pricing'] ? 6 : 3); -$rows_per_page = $options['pricing'] ? 20 : 19; -$autofill->setRows($rows_per_page, 0, $options['pricing'] ? 20 : 19); +$autofill = new Util\Autofill($options['pricing'] ? 6 : 3, 70); +$rows_per_page = $options['pricing'] ? 20 : 18; +$autofill->setRows($rows_per_page, 0, $options['pricing'] ? 20 : 18); // Intestazione tabella per righe echo " @@ -75,8 +75,6 @@ foreach ($righe as $riga) { ++$num; $r = $riga->toArray(); - $autofill->count($r['descrizione']); - echo ' '; @@ -109,26 +107,23 @@ foreach ($righe as $riga) { '.nl2br($text); + $autofill->count($text); } } $r['descrizione'] = preg_replace("/Rif\.(.*)/s", '', (string) $r['descrizione']); - $autofill->count($r['descrizione']); } $source_type = $riga::class; - - if (!setting('Visualizza riferimento su ogni riga in stampa')) { - echo $num.' - - '.$r['descrizione']; - } else { - echo $num.' - - '.nl2br((string) $r['descrizione']); - } + $autofill->count($r['descrizione']); + + echo $num.' + + '.nl2br((string) $r['descrizione']); + if ($riga->isArticolo()) { echo '
    '.$riga->codice.''; + $autofill->count($riga->codice); } else { echo '-'; } diff --git a/templates/fatture/body.php b/templates/fatture/body.php index 1a358546d..fa02744a5 100755 --- a/templates/fatture/body.php +++ b/templates/fatture/body.php @@ -77,8 +77,6 @@ foreach ($righe as $riga) { ++$num; $r = $riga->toArray(); - $autofill->count($r['descrizione']); - $v_iva[$r['desc_iva']] = sum($v_iva[$r['desc_iva']], $riga->iva); $v_totale[$r['desc_iva']] = sum($v_totale[$r['desc_iva']], $riga->totale_imponibile); @@ -97,11 +95,11 @@ foreach ($righe as $riga) { $text = $text.'Ordine n. '.$riga_ordine['numero_cliente'].' del '.Translator::dateToLocale($riga_ordine['data_cliente']).'
    '; } } - $r['descrizione'] = str_replace('Rif. '.strtolower((string) $key), '', $r['descrizione']); + $r['descrizione'] = str_replace("\nRif. ".strtolower((string) $key), '', $r['descrizione']); if (preg_match("/Rif\.(.*)/s", $r['descrizione'], $rif2)) { - $r['descrizione'] = str_replace('Rif.'.strtolower($rif2[1] ?: ''), '', $r['descrizione']); - $text .= ''.$rif2[0].''; + $r['descrizione'] = str_replace('\nRif.'.strtolower($rif2[1] ?: ''), '', $r['descrizione']); + $text .= ''.$rif2[0].'
    '; } $text .= ''.$key.''; @@ -111,26 +109,23 @@ foreach ($righe as $riga) { '.nl2br($text); + $autofill->count($text); } } - $r['descrizione'] = preg_replace("/Rif\.(.*)/s", ' ', (string) $r['descrizione']); - $autofill->count($r['descrizione']); + $r['descrizione'] = preg_replace("/(\r\n|\r|\n)Rif\.(.*)/s", '', (string) $r['descrizione']); } $source_type = $riga::class; - if (setting('Visualizza riferimento su ogni riga in stampa')) { + + $autofill->count($r['descrizione']); echo $num.' - - '.nl2br((string) $r['descrizione']).'
    '; - } else { - echo $num.' - - '.nl2br((string) $r['descrizione']); - } + + '.nl2br((string) $r['descrizione']); if ($riga->isArticolo()) { - echo ''.$riga->codice.''; - } + echo '
    '.$riga->codice.'
    '; + $autofill->count($riga->codice, true); + } if ($riga->isArticolo()) { // Seriali @@ -217,6 +212,10 @@ foreach ($righe as $riga) { $autofill->next(); } +foreach ($v_iva as $desc_iva => $tot_iva) { + $autofill->count($desc_iva, true); +} + echo ' |autofill| @@ -252,7 +251,7 @@ echo ' if (!empty($record['note'])) { echo '

    '.tr('Note', [], ['upper' => true]).':

    -

    '.nl2br((string) $record['note']).'

    '; +

    '.nl2br((string) $record['note']).'

    '; $autofill->count($record['note'], true); } diff --git a/templates/ordini/body.php b/templates/ordini/body.php index 7422c3200..800733d6d 100755 --- a/templates/ordini/body.php +++ b/templates/ordini/body.php @@ -49,20 +49,18 @@ $has_image = $righe->search(fn ($item) => !empty($item->articolo->immagine)) !== if ($has_image) { ++$columns; - $char_number = $options['pricing'] ? 26 : 63; + } if ($documento->direzione == 'uscita') { $columns += 2; - $char_number = $options['pricing'] ? 26 : 63; -} else { - $char_number = $options['pricing'] ? 45 : 82; -} +} + $columns = $options['pricing'] ? $columns : $columns - 3; // Creazione righe fantasma $autofill = new Util\Autofill($columns); -$autofill->setRows(20, 10); +$autofill->setRows(27, 0, 32); // Intestazione tabella per righe echo " @@ -107,8 +105,6 @@ foreach ($righe as $riga) { ++$num; $r = $riga->toArray(); - $autofill->count($r['descrizione']); - echo ' '; @@ -122,11 +118,11 @@ foreach ($righe as $riga) { if (!empty($riga_ordine['numero_cliente']) && !empty($riga_ordine['data_cliente'])) { $text = $text.'Ordine n. '.$riga_ordine['numero_cliente'].' del '.Translator::dateToLocale($riga_ordine['data_cliente']).'
    '; } - $r['descrizione'] = str_replace('Rif. '.strtolower((string) $key), '', $r['descrizione']); + $r['descrizione'] = str_replace("\nRif. ".strtolower((string) $key), '', $r['descrizione']); if (preg_match("/Rif\.(.*)/s", $r['descrizione'], $rif2)) { - $r['descrizione'] = str_replace('Rif.'.strtolower($rif2[1] ?: ''), '', $r['descrizione']); - $text .= ''.$rif2[0].''; + $r['descrizione'] = str_replace('\nRif.'.strtolower($rif2[1] ?: ''), '', $r['descrizione']); + $text .= ''.$rif2[0].'
    '; } $text .= ''.$key.''; @@ -149,46 +145,32 @@ foreach ($righe as $riga) { echo ' '.nl2br($text); + $autofill->count($text); } } - $r['descrizione'] = preg_replace("/Rif\.(.*)/s", '', (string) $r['descrizione']); - $autofill->count($r['descrizione']); + $r['descrizione'] = preg_replace("/(\r\n|\r|\n)Rif\.(.*)/s", '', (string) $r['descrizione']); } $source_type = $riga::class; - if (!setting('Visualizza riferimento su ogni riga in stampa')) { - echo $num.''; - if ($has_image) { - if ($riga->isArticolo() && !empty($riga->articolo->image)) { - echo ' - - - '; - } else { - echo ' - '; - } - } - echo ' - '.$r['descrizione']; - } else { - echo $num.' + $autofill->count($r['descrizione']); + + echo $num.''; + if ($has_image) { + if ($riga->isArticolo() && !empty($riga->articolo->image)) { + echo ' + + '; - if ($has_image) { - if ($riga->isArticolo() && !empty($riga->articolo->image)) { - echo ' - - - '; - } else { - echo ' - '; - } + } else { + echo ' + '; } - echo ' - '.nl2br((string) $r['descrizione']); } + echo ' + '.nl2br((string) $r['descrizione']); + + if ($documento->direzione == 'uscita') { echo ' @@ -282,6 +264,7 @@ foreach ($righe as $riga) { echo ' + '; } } @@ -410,5 +393,5 @@ if (!empty($documento['note'])) { echo '

    '.tr('Note', [], ['upper' => true]).':

    -

    '.nl2br((string) $documento['note']).'

    '; +

    '.nl2br((string) $documento['note']).'

    '; } diff --git a/templates/ordini/header.php b/templates/ordini/header.php index 76e3b1684..819a41b7b 100755 --- a/templates/ordini/header.php +++ b/templates/ordini/header.php @@ -60,6 +60,37 @@ echo ' + + + + + + + + + + + + + + + + + + +
    + '.tr('Porto', [], ['upper' => true]).' + + '.tr('Tipo di spedizione', [], ['upper' => true]).' + + '.tr('Vettore', [], ['upper' => true]).' +
    + $porto$   + + $spedizione$   + + $vettore$   +
    diff --git a/templates/ordini/init.php b/templates/ordini/init.php index 8b1942d00..63657a37c 100755 --- a/templates/ordini/init.php +++ b/templates/ordini/init.php @@ -66,10 +66,17 @@ if (!empty($documento->idsede)) { $numero = !empty($documento['numero_esterno']) ? $documento['numero_esterno'] : $documento['numero']; $pagamento = $dbo->fetchOne('SELECT * FROM `co_pagamenti` LEFT JOIN `co_pagamenti_lang` ON (`co_pagamenti`.`id` = `co_pagamenti_lang`.`id_record` AND `co_pagamenti_lang`.`id_lang` = '.prepare(Models\Locale::getDefault()->id).') WHERE `co_pagamenti`.`id` = '.prepare($documento->idpagamento)); +$porto = $dbo->fetchOne('SELECT `dt_porto`.*, `dt_porto_lang`.`title` as descrizione FROM `dt_porto` LEFT JOIN `dt_porto_lang` ON (`dt_porto`.`id` = `dt_porto_lang`.`id_record` AND `dt_porto_lang`.`id_lang` ='.prepare(Models\Locale::getDefault()->id).') WHERE `dt_porto`.`id` = '.prepare($documento['idporto'])); +$spedizione = $dbo->fetchOne('SELECT `dt_spedizione`.*, `dt_spedizione_lang`.`title` as descrizione FROM `dt_spedizione` LEFT JOIN `dt_spedizione_lang` ON (`dt_spedizione`.`id`=`dt_spedizione_lang`.`id_record` AND `dt_spedizione_lang`.`id_lang`='.prepare(Models\Locale::getDefault()->id).') WHERE `dt_spedizione`.`id` = '.prepare($documento['idspedizione'])); +$vettore = $dbo->fetchOne('SELECT ragione_sociale FROM an_anagrafiche WHERE idanagrafica = '.prepare($documento['idvettore'])); + // Sostituzioni specifiche $custom = [ 'tipo_doc' => Stringy\Stringy::create($documento->tipo->getTranslation('title'))->toUpperCase(), 'numero' => $numero, 'data' => Translator::dateToLocale($documento['data']), 'pagamento' => $pagamento['descrizione'], -]; + 'porto' => $porto['descrizione'], + 'spedizione' => $spedizione['descrizione'], + 'vettore' => $vettore['ragione_sociale'] +]; \ No newline at end of file diff --git a/templates/preventivi/body.php b/templates/preventivi/body.php index 8ae2a6f49..9bc3458c6 100755 --- a/templates/preventivi/body.php +++ b/templates/preventivi/body.php @@ -65,7 +65,7 @@ if ($has_image) { // Creazione righe fantasma $autofill = new Util\Autofill($columns); -$autofill->setRows(26, 0); +$autofill->setRows(22, 0, 36); echo '
    diff --git a/update/2_4_20.php b/update/2_4_20.php index 6a422ae3b..230b9c3bd 100644 --- a/update/2_4_20.php +++ b/update/2_4_20.php @@ -29,7 +29,6 @@ delete($files); * in quanto veniva calcolato lo sconto ivato erroneamente. */ -// TODO: aggiornare procedura con query SQL esplicite invece che con i metodi che possono cambiare nel tempo // Fix sconti contratti $righe = $dbo->fetchArray('SELECT id, idiva, sconto_percentuale, sconto_unitario, tipo_sconto, prezzo_unitario FROM co_righe_contratti WHERE sconto_percentuale != 0 AND tipo_sconto="PRC"'); diff --git a/update/2_6.sql b/update/2_6.sql index 02350c106..72765dbd0 100644 --- a/update/2_6.sql +++ b/update/2_6.sql @@ -96,4 +96,26 @@ UPDATE `zz_views_lang` SET `title` = 'emailto_Email' WHERE `zz_views_lang`.`id_r ALTER TABLE `mg_articoli` ADD `modello` VARCHAR(255) NULL AFTER `id_marchio`; -- Fix filtro segmento Non completate in Attività -UPDATE `zz_segments` SET `clause` = 'in_interventi.idstatointervento NOT IN(SELECT in_statiintervento.id FROM in_statiintervento WHERE is_completato=1)' WHERE `zz_segments`.`name` = 'Non completate' AND `id_module` = (SELECT `id` FROM `zz_modules` WHERE `name` = 'Interventi'); \ No newline at end of file +UPDATE `zz_segments` SET `clause` = 'in_interventi.idstatointervento NOT IN(SELECT in_statiintervento.id FROM in_statiintervento WHERE is_completato=1)' WHERE `zz_segments`.`name` = 'Non completate' AND `id_module` = (SELECT `id` FROM `zz_modules` WHERE `name` = 'Interventi'); + +INSERT INTO `zz_api_resources` (`id`, `version`, `type`, `resource`, `class`, `enabled`) VALUES +(NULL, 'v1', 'retrieve', 'pagamenti', 'Modules\\Pagamenti\\API\\v1\\Pagamenti', 1); + +-- Aggiunto spedizione porto e vettore in ordini +ALTER TABLE `or_ordini` ADD `idspedizione` TINYINT NULL AFTER `codice_commessa`, ADD `idporto` TINYINT NULL AFTER `idspedizione`, ADD `idvettore` INT NULL AFTER `idporto`; + +-- Aggiunta del plugin Importazione FE +SELECT @id_module := `id` FROM `zz_modules` WHERE `name` = 'Fatture di vendita'; +INSERT INTO `zz_plugins` (`name`, `idmodule_from`, `idmodule_to`, `position`, `script`, `enabled`, `default`, `order`, `compatibility`, `version`, `options2`, `options`, `directory`, `help`) VALUES +('Importazione FE', @id_module, @id_module, 'tab_main', '', 1, 1, 0, '2.6.*', '2.0', NULL, 'custom', 'importFE_ZIP', ''); + +INSERT INTO `zz_plugins_lang` (`id_lang`, `id_record`, `title`) +VALUES + (1, LAST_INSERT_ID(), 'Importazione FE'), + (2, LAST_INSERT_ID(), 'Importazione FE'); + +-- Aggiunta impostazione per metodo di importazione XML fatture +INSERT INTO `zz_settings` (`nome`, `valore`, `tipo`, `editable`, `sezione`, `order`) VALUES ('Metodo di importazione XML fatture di vendita', 'Automatico', 'list[Automatico,Manuale]', '1', 'Fatturazione', NULL); +INSERT INTO `zz_settings_lang` (`id_record`, `id_lang`, `title`) VALUES + (LAST_INSERT_ID(), 1, 'Metodo di importazione XML fatture di vendita'), + (LAST_INSERT_ID(), 2, 'Metodo di importazione XML fatture di vendita');