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 ' +
'.tr('Per vedere le fatture da importare utilizza il pulsante _BUTTON_', [ + '_BUTTON_' => '"'.tr('Ricerca fatture di acquisto').'"', + ]).'.
'; +} else { + include $structure->filepath('list.php'); +} + +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 ' +'.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 ' + + + | +
'.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 ' +'.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 ' + | +