2017-08-04 16:28:16 +02:00
< ? php
2020-09-07 15:04:06 +02:00
/*
* OpenSTAManager : il software gestionale open source per l ' assistenza tecnica e la fatturazione
2021-01-20 15:08:51 +01:00
* Copyright ( C ) DevCode s . r . l .
2020-09-07 15:04:06 +02:00
*
* This program is free software : you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation , either version 3 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program . If not , see < https :// www . gnu . org / licenses />.
*/
2017-08-04 16:28:16 +02:00
include_once __DIR__ . '/../../core.php' ;
2019-07-15 12:27:12 +02:00
use Modules\Anagrafiche\Anagrafica ;
2020-11-06 10:46:42 +01:00
use Modules\Articoli\Articolo as ArticoloOriginale ;
2019-08-29 10:25:14 +02:00
use Modules\Emails\Mail ;
use Modules\Emails\Template ;
2019-07-24 15:29:03 +02:00
use Modules\Fatture\Components\Descrizione ;
use Modules\Fatture\Components\Riga ;
2019-07-24 15:41:04 +02:00
use Modules\Fatture\Fattura ;
2019-07-15 12:27:12 +02:00
use Modules\Interventi\Components\Sessione ;
use Modules\Interventi\Intervento ;
2021-02-25 15:16:05 +01:00
use Util\Generator ;
2020-09-09 15:12:02 +02:00
use Util\Ini ;
2019-07-15 12:27:12 +02:00
2017-08-04 16:28:16 +02:00
/**
* Recupera il totale delle ore spese per un intervento .
*
2019-07-24 15:29:03 +02:00
* @ param int $id_intervento
2020-08-17 09:47:18 +02:00
*
* @ deprecated
2017-08-04 16:28:16 +02:00
*/
2019-07-24 15:29:03 +02:00
function get_ore_intervento ( $id_intervento )
2019-07-24 15:41:04 +02:00
{
$intervento = Intervento :: find ( $id_intervento );
2017-08-04 16:28:16 +02:00
2019-07-24 15:41:04 +02:00
return $intervento -> ore_totali ;
2017-08-04 16:28:16 +02:00
}
/**
* Funzione per collegare gli articoli , usati in un intervento , ai rispettivi impianti .
*
2020-09-09 15:12:02 +02:00
* @ param int $id_intervento
* @ param int $id_impianto
* @ param int $id_articolo
2018-06-27 18:58:50 +02:00
* @ param int $qta
2021-08-05 18:17:26 +02:00
*
* @ deprecated 2.4 . 25
2017-08-04 16:28:16 +02:00
*/
2020-09-09 15:12:02 +02:00
function link_componente_to_articolo ( $id_intervento , $id_impianto , $id_articolo , $qta )
2017-08-04 16:28:16 +02:00
{
2020-09-09 15:12:02 +02:00
if ( empty ( $id_impianto ) || empty ( $id_intervento )) {
return ;
}
2017-08-04 16:28:16 +02:00
2020-09-09 15:12:02 +02:00
$dbo = database ();
$intervento = Intervento :: find ( $id_intervento );
2017-08-04 16:28:16 +02:00
2020-09-09 15:12:02 +02:00
// Data di inizio dell'intervento (data_richiesta in caso di assenza di sessioni)
$data = $intervento -> inizio ? : $intervento -> data_richiesta ;
// Se l'articolo aggiunto è collegato a un componente, aggiungo il componente all'impianto selezionato
$componente_articolo = $dbo -> fetchOne ( 'SELECT componente_filename, contenuto FROM mg_articoli WHERE id = ' . prepare ( $id_articolo ));
if ( ! empty ( $componente_articolo ) && ! empty ( $componente_articolo [ 'componente_filename' ])) {
$contenuto_ini = Ini :: read ( $componente_articolo [ 'contenuto' ]);
$nome_componente = Ini :: getValue ( $contenuto_ini , 'Nome' );
$dati = [
'idimpianto' => $id_impianto ,
'idintervento' => $id_intervento ,
'nome' => $nome_componente ,
'data' => $data ,
'filename' => $componente_articolo [ 'componente_filename' ],
'contenuto' => $componente_articolo [ 'contenuto' ],
];
// Inserisco il componente tante volte quante la quantità degli articoli inseriti
for ( $q = 0 ; $q < $qta ; ++ $q ) {
$dbo -> insert ( 'my_impianto_componenti' , $dati );
2017-08-04 16:28:16 +02:00
}
}
}
2020-09-09 15:12:02 +02:00
function add_tecnico ( $id_intervento , $idtecnico , $inizio , $fine , $idcontratto = null )
2017-08-04 16:28:16 +02:00
{
2020-09-09 15:12:02 +02:00
$intervento = Intervento :: find ( $id_intervento );
2019-07-15 12:27:12 +02:00
$anagrafica = Anagrafica :: find ( $idtecnico );
2017-09-14 16:20:11 +02:00
2019-07-15 12:27:12 +02:00
$sessione = Sessione :: build ( $intervento , $anagrafica , $inizio , $fine );
2018-08-31 17:33:33 +02:00
2018-09-20 11:39:03 +02:00
// Notifica nuovo intervento al tecnico
2020-06-23 16:29:56 +02:00
if ( setting ( 'Notifica al tecnico l\'assegnazione all\'attività' )) {
2020-06-22 17:45:10 +02:00
if ( ! empty ( $anagrafica [ 'email' ])) {
2020-09-23 13:36:37 +02:00
$template = Template :: pool ( 'Notifica intervento' );
2020-06-22 17:45:10 +02:00
2020-06-30 13:26:15 +02:00
if ( ! empty ( $template )) {
2020-09-09 15:12:02 +02:00
$mail = Mail :: build ( auth () -> getUser (), $template , $id_intervento );
2020-06-22 17:45:10 +02:00
$mail -> addReceiver ( $anagrafica [ 'email' ]);
$mail -> save ();
}
}
2018-09-20 11:39:03 +02:00
}
2018-08-31 17:33:33 +02:00
return true ;
2017-08-04 16:28:16 +02:00
}
2017-09-06 15:29:51 +02:00
2018-07-17 12:05:21 +02:00
/**
* Calcola le ore presenti tra due date .
*
* @ param string $orario_inizio
* @ param string $orario_fine
*
* @ return float
2020-08-17 09:47:18 +02:00
*
* @ deprecated
2018-07-17 12:05:21 +02:00
*/
function calcola_ore_intervento ( $orario_inizio , $orario_fine )
{
$inizio = new DateTime ( $orario_inizio );
$diff = $inizio -> diff ( new DateTime ( $orario_fine ));
2018-09-18 18:21:54 +02:00
$ore = $diff -> i / 60 + $diff -> h + ( $diff -> days * 24 );
2018-07-17 12:05:21 +02:00
return $ore ;
}
2018-09-03 16:01:05 +02:00
2018-09-26 16:28:02 +02:00
function aggiungi_intervento_in_fattura ( $id_intervento , $id_fattura , $descrizione , $id_iva , $id_conto , $id_rivalsa_inps = false , $id_ritenuta_acconto = false , $calcolo_ritenuta_acconto = false )
2018-08-31 12:33:48 +02:00
{
2018-09-20 12:05:22 +02:00
$dbo = database ();
2018-08-31 12:33:48 +02:00
2019-01-22 15:07:09 +01:00
$id_rivalsa_inps = $id_rivalsa_inps !== false ? $id_rivalsa_inps : setting ( 'Percentuale rivalsa' );
2018-09-26 16:28:02 +02:00
$id_ritenuta_acconto = $id_ritenuta_acconto !== false ? $id_ritenuta_acconto : setting ( " Percentuale ritenuta d'acconto " );
2018-10-04 17:41:31 +02:00
$calcolo_ritenuta_acconto = $calcolo_ritenuta_acconto !== false ? $calcolo_ritenuta_acconto : setting ( " Metodologia calcolo ritenuta d'acconto predefinito " );
2018-09-04 17:59:12 +02:00
2019-07-24 15:29:03 +02:00
$fattura = Fattura :: find ( $id_fattura );
$intervento = Intervento :: find ( $id_intervento );
2018-08-31 12:33:48 +02:00
2021-03-09 12:54:11 +01:00
$data = $intervento -> fine ;
2019-07-24 15:29:03 +02:00
$codice = $intervento -> codice ;
2018-08-31 12:33:48 +02:00
2019-07-24 15:29:03 +02:00
// Riga di descrizione
$riga = Descrizione :: build ( $fattura );
$riga -> descrizione = $descrizione ;
$riga -> idintervento = $id_intervento ;
$riga -> save ();
// Ore di lavoro raggruppate per costo orario
$sessioni = $intervento -> sessioni ;
2018-08-31 12:33:48 +02:00
2019-07-24 15:29:03 +02:00
if ( empty ( $sessioni )) {
2021-07-28 11:50:02 +02:00
flash () -> warning ( tr ( " L'attività _NUM_ non ha sessioni di lavoro! " , [
2019-07-24 15:29:03 +02:00
'_NUM_' => $codice ,
]));
} else {
2020-11-20 15:50:15 +01:00
$decimals = setting ( 'Cifre decimali per quantità' );
2019-07-24 15:29:03 +02:00
$ore_di_lavoro = $sessioni -> groupBy ( function ( $item , $key ) {
return $item [ 'prezzo_orario' ] . '|' . $item [ 'sconto_unitario' ] . '|' . $item [ 'tipo_sconto' ];
});
2019-07-24 15:41:04 +02:00
foreach ( $ore_di_lavoro as $gruppo ) {
2019-07-24 15:29:03 +02:00
$sessione = $gruppo -> first ();
$riga = Riga :: build ( $fattura );
2021-07-28 11:50:02 +02:00
$riga -> descrizione = tr ( " Ore di lavoro dell'attività _NUM_ del _DATE_ " , [
2019-07-24 15:29:03 +02:00
'_NUM_' => $codice ,
'_DATE_' => dateFormat ( $data ),
]);
$riga -> idintervento = $id_intervento ;
$riga -> um = 'ore' ;
$riga -> id_iva = $id_iva ;
$riga -> idconto = $id_conto ;
$riga -> calcolo_ritenuta_acconto = $calcolo_ritenuta_acconto ;
$riga -> id_ritenuta_acconto = $id_ritenuta_acconto ;
$riga -> id_rivalsa_inps = $id_rivalsa_inps ;
2020-02-14 17:43:39 +01:00
$riga -> prezzo_unitario = $sessione -> prezzo_orario ;
2021-09-27 10:27:17 +02:00
//Calcolo lo sconto unitario della sessione in base all'impostazione sui prezzi ivati
2021-10-18 10:23:39 +02:00
$iva = $dbo -> table ( 'co_iva' ) -> where ( 'id' , $id_iva ) -> first ();
if ( $sessione -> tipo_sconto == 'UNT' && setting ( 'Utilizza prezzi di vendita comprensivi di IVA' )) {
$sconto_unitario = $sessione -> sconto_unitario + (( $sessione -> sconto_unitario * $iva -> percentuale ) / 100 );
} else {
2021-09-27 10:27:17 +02:00
$sconto_unitario = $sessione -> sconto_unitario ;
}
$riga -> setSconto ( $sconto_unitario , $sessione -> tipo_sconto );
2019-07-24 15:29:03 +02:00
2020-11-20 15:50:15 +01:00
$qta_gruppo = $gruppo -> sum ( 'ore' );
$riga -> qta = round ( $qta_gruppo , $decimals );
2019-07-24 15:29:03 +02:00
2021-09-17 10:30:36 +02:00
// Riferimento al documento di origine
$riga -> original_document_type = get_class ( $intervento );
$riga -> original_document_id = $intervento -> id ;
2019-07-24 15:29:03 +02:00
$riga -> save ();
2018-08-31 12:33:48 +02:00
}
2019-07-24 15:29:03 +02:00
// Diritti di chiamata raggruppati per costo
2019-07-24 15:41:04 +02:00
$diritti_chiamata = $sessioni -> where ( 'prezzo_diritto_chiamata' , '>' , 0 ) -> groupBy ( function ( $item , $key ) {
2019-07-24 15:29:03 +02:00
return $item [ 'prezzo_diritto_chiamata' ];
});
2019-07-24 15:41:04 +02:00
foreach ( $diritti_chiamata as $gruppo ) {
2019-07-24 15:29:03 +02:00
$diritto_chiamata = $gruppo -> first ();
$riga = Riga :: build ( $fattura );
2021-07-28 11:50:02 +02:00
$riga -> descrizione = tr ( " Diritto di chiamata dell'attività _NUM_ del _DATE_ " , [
2019-07-24 15:29:03 +02:00
'_NUM_' => $codice ,
'_DATE_' => dateFormat ( $data ),
]);
$riga -> idintervento = $id_intervento ;
2020-02-14 13:21:11 +01:00
//$riga->um = 'ore';
2019-07-24 15:29:03 +02:00
$riga -> id_iva = $id_iva ;
$riga -> idconto = $id_conto ;
2018-08-31 12:33:48 +02:00
2019-07-24 15:29:03 +02:00
$riga -> calcolo_ritenuta_acconto = $calcolo_ritenuta_acconto ;
$riga -> id_ritenuta_acconto = $id_ritenuta_acconto ;
$riga -> id_rivalsa_inps = $id_rivalsa_inps ;
2018-08-31 12:33:48 +02:00
2020-02-14 17:43:39 +01:00
$riga -> prezzo_unitario = $diritto_chiamata -> prezzo_diritto_chiamata ;
2019-07-24 15:29:03 +02:00
$riga -> qta = $gruppo -> count ();
2021-09-17 10:30:36 +02:00
// Riferimento al documento di origine
$riga -> original_document_type = get_class ( $intervento );
$riga -> original_document_id = $intervento -> id ;
2019-07-24 15:29:03 +02:00
$riga -> save ();
}
2020-01-24 12:24:20 +01:00
// Viaggi raggruppati per costo
$viaggi = $sessioni -> where ( 'prezzo_km_unitario' , '>' , 0 ) -> groupBy ( function ( $item , $key ) {
2020-05-18 09:33:43 +02:00
return $item [ 'prezzo_km_unitario' ] . '|' . $item [ 'scontokm_unitario' ] . '|' . $item [ 'tipo_scontokm' ];
2020-01-24 12:24:20 +01:00
});
foreach ( $viaggi as $gruppo ) {
2020-11-20 15:23:52 +01:00
$qta_trasferta = $gruppo -> sum ( 'km' );
2020-11-20 15:50:15 +01:00
if ( $qta_trasferta == 0 ) {
2020-11-20 15:23:52 +01:00
continue ;
}
2020-01-24 12:24:20 +01:00
$viaggio = $gruppo -> first ();
$riga = Riga :: build ( $fattura );
2021-07-28 11:50:02 +02:00
$riga -> descrizione = tr ( " Trasferta dell'attività _NUM_ del _DATE_ " , [
2020-01-24 12:24:20 +01:00
'_NUM_' => $codice ,
'_DATE_' => dateFormat ( $data ),
]);
$riga -> idintervento = $id_intervento ;
$riga -> um = 'km' ;
$riga -> id_iva = $id_iva ;
$riga -> idconto = $id_conto ;
$riga -> calcolo_ritenuta_acconto = $calcolo_ritenuta_acconto ;
$riga -> id_ritenuta_acconto = $id_ritenuta_acconto ;
$riga -> id_rivalsa_inps = $id_rivalsa_inps ;
2020-02-14 17:43:39 +01:00
$riga -> prezzo_unitario = $viaggio -> prezzo_km_unitario ;
2020-05-18 09:33:43 +02:00
$riga -> setSconto ( $viaggio -> scontokm_unitario , $viaggio -> tipo_scontokm );
2020-01-24 12:24:20 +01:00
2021-09-17 10:30:36 +02:00
// Riferimento al documento di origine
$riga -> original_document_type = get_class ( $intervento );
$riga -> original_document_id = $intervento -> id ;
2020-11-20 15:23:52 +01:00
$riga -> qta = $qta_trasferta ;
2020-01-24 12:24:20 +01:00
$riga -> save ();
}
2018-08-31 12:33:48 +02:00
}
2019-07-24 15:29:03 +02:00
// Articoli, righe, sconti e descrizioni collegati all'intervento
$righe = $intervento -> getRighe ();
foreach ( $righe as $riga ) {
$qta = $riga -> qta ;
$copia = $riga -> copiaIn ( $fattura , $qta );
2020-11-09 20:03:47 +01:00
2019-07-24 15:29:03 +02:00
$copia -> id_conto = $id_conto ;
$copia -> calcolo_ritenuta_acconto = $calcolo_ritenuta_acconto ;
$copia -> id_ritenuta_acconto = $id_ritenuta_acconto ;
$copia -> id_rivalsa_inps = $id_rivalsa_inps ;
// Aggiornamento seriali dalla riga dell'ordine
if ( $copia -> isArticolo ()) {
$copia -> serials = $riga -> serials ;
2020-11-04 11:18:37 +01:00
$articolo = ArticoloOriginale :: find ( $copia -> idarticolo );
$copia -> id_conto = ( $articolo -> idconto_vendita ? $articolo -> idconto_vendita : $id_conto );
2018-08-31 12:33:48 +02:00
}
2019-07-24 15:29:03 +02:00
$copia -> save ();
2018-08-31 12:33:48 +02:00
}
// Ricalcolo inps, ritenuta e bollo
2019-07-24 15:29:03 +02:00
ricalcola_costiagg_fattura ( $id_fattura );
2018-08-31 12:33:48 +02:00
// Metto l'intervento in stato "Fatturato"
2019-11-13 12:03:29 +01:00
$dbo -> query ( " UPDATE in_interventi SET idstatointervento=(SELECT idstatointervento FROM in_statiintervento WHERE codice='FAT') WHERE id= " . prepare ( $id_intervento ));
2018-08-31 12:33:48 +02:00
}
2021-02-25 15:16:05 +01:00
/**
* Verifica che il numero_esterno della fattura indicata sia correttamente impostato , a partire dai valori delle fatture ai giorni precedenti .
* Restituisce il numero_esterno mancante in caso di numero errato .
*
* @ return bool | string
*/
function verifica_numero_intervento ( Intervento $intervento )
{
if ( empty ( $intervento -> codice )) {
return null ;
}
$data = $intervento -> data_richiesta ;
2021-07-30 16:16:27 +02:00
$documenti = Intervento :: whereDate ( 'data_richiesta' , '=' , $data -> format ( 'Y-m-d' ))
2021-02-25 15:16:05 +01:00
-> get ();
// Recupero maschera per questo segmento
$maschera = setting ( 'Formato codice attività' );
if (( strpos ( $maschera , 'YYYY' ) !== false ) or ( strpos ( $maschera , 'yy' ) !== false )) {
$ultimo = Generator :: getPreviousFrom ( $maschera , 'in_interventi' , 'codice' , [
2021-07-30 16:16:27 +02:00
'DATE(data_richiesta) < ' . prepare ( $data -> format ( 'Y-m-d' )),
'YEAR(data_richiesta) = ' . prepare ( $data -> format ( 'Y' )),
2021-02-25 15:16:05 +01:00
], $data );
} else {
$ultimo = Generator :: getPreviousFrom ( $maschera , 'in_interventi' , 'codice' , [
2021-07-30 16:16:27 +02:00
'DATE(data_richiesta) < ' . prepare ( $data -> format ( 'Y-m-d' )),
2021-02-25 15:16:05 +01:00
]);
}
do {
$numero = Generator :: generate ( $maschera , $ultimo , 1 , Generator :: dateToPattern ( $data ), $data );
$filtered = $documenti -> reject ( function ( $item , $key ) use ( $numero ) {
return $item -> codice == $numero ;
});
if ( $documenti -> count () == $filtered -> count ()) {
return $numero ;
}
$documenti = $filtered ;
$ultimo = $numero ;
} while ( $numero != $intervento -> codice );
return null ;
}