Files correlati : Ricompilazione Demo : [ ] Commento : eliminati riferimenti ai files accoppati colldich e caradd; si continuano a compilare AB0 (che sta nel cd) e AB1 (che non sta da nessuna parte), mentre non si compilano AB2 e AB3 (che però non si compilavano neanche prima) git-svn-id: svn://10.65.10.50/trunk@20660 c028cbd2-c16b-5b4b-a496-9718f37d4682
		
			
				
	
	
		
			2113 lines
		
	
	
		
			64 KiB
		
	
	
	
		
			C++
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			2113 lines
		
	
	
		
			64 KiB
		
	
	
	
		
			C++
		
	
	
		
			Executable File
		
	
	
	
	
| 
 | |
| #include "ablib01.h"
 | |
| /**************************************************************************************************/
 | |
| //  TAlbero_AB
 | |
| /**************************************************************************************************/
 | |
| 
 | |
| TAlbero_AB::TAlbero_AB(int filenum)
 | |
|              :TRectype(filenum), _inter_tree(NULL), _newrec(NULL)
 | |
| {            
 | |
| }
 | |
| 
 | |
| TAlbero_AB::~TAlbero_AB()
 | |
| {
 | |
| }
 | |
| 
 | |
| int TAlbero_AB::write(TBaseisamfile &analisi) const
 | |
| {                      
 | |
|   int err = TRectype::write(analisi);
 | |
|   if (err==NOERR)
 | |
|   {
 | |
|     commit_body();
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     err = rewrite(analisi);
 | |
|   }
 | |
|   if (err)
 | |
|     CHECK (FALSE,"Errore di scrittura del record sull'isamfile");
 | |
|   return err;
 | |
| }
 | |
| 
 | |
| int TAlbero_AB::rewrite(TBaseisamfile &analisi) const
 | |
| {                      
 | |
| //  _inter_tree->write_cache(); //Solo debug
 | |
|   int err = TRectype::rewrite(analisi);
 | |
|   if (err==NOERR)
 | |
|   {
 | |
|     commit_body();
 | |
|   }
 | |
|   if (err)
 | |
|     CHECK (FALSE,"Errore di scrittura del record sull'isamfile");
 | |
|   return err;
 | |
| }
 | |
| 
 | |
| //Leggo la "testata" del record
 | |
| int TAlbero_AB::read(TBaseisamfile& saldi, word isop, word lockop)
 | |
| {         
 | |
|   int err=TRectype::read(saldi, isop, lockop);
 | |
|   if (err==NOERR) //Eliminare questa lettura con isop = 0 o isop = 1
 | |
|   {               
 | |
|     read_body(FALSE);
 | |
|     _inter_tree->make_root(saldi);
 | |
|   }
 | |
|   _inter_tree->write_cache(); //solo debig
 | |
|   return err;
 | |
| } 
 | |
| 
 | |
| //Leggo la testata del record basandomi sulla posizione
 | |
| int TAlbero_AB::readat(TBaseisamfile& analisi, TRecnotype nrec, word lockop)
 | |
| {
 | |
|   int err=TRectype::readat(analisi, nrec, lockop);
 | |
| 
 | |
| // E' stata tolta la read_body perchè la readat viene richiamata
 | |
| // prima di fare una write della relazione: si perdono le modifiche
 | |
| // effettuate in quanto la read_body ripristina la vecchia situazione
 | |
| 
 | |
| //  if (err==NOERR)
 | |
| //    read_body(FALSE);
 | |
|   return err;
 | |
| }
 | |
| 
 | |
| 
 | |
| int TAlbero_AB::remove(TBaseisamfile& f) const
 | |
| {
 | |
| /*Abilitare questa parte solo per effettuare un debug della delete_node
 | |
| //  anche se messa in questa posizione, la chiamata non ha senso
 | |
|  // TLocalisamfile lf(LF_RELANA);
 | |
|   TNumeric_id id;
 | |
|   id = 4;
 | |
|   _inter_tree->delete_node(id);
 | |
|   int errc = commit_body();
 | |
|   if (errc != NOERR)
 | |
|   {             
 | |
|     message_box("Errore nell'aggiornameno del file dati");
 | |
|     //return errc;
 | |
|   }
 | |
|   return 0;
 | |
| */  
 | |
| 
 | |
| 
 | |
| //*Abilitare solo questa parte per il test della remove_tree  
 | |
|   _inter_tree->delete_tree();
 | |
|   int errc = commit_body();
 | |
|   if (errc != NOERR)
 | |
|   {             
 | |
|     message_box("Errore nell'aggiornameno del file dati");
 | |
|     //return errc;
 | |
|   }
 | |
|   int err = TRectype::remove(f);
 | |
|   return err;
 | |
| } 
 | |
| 
 | |
| void TAlbero_AB::zero() 
 | |
| {
 | |
|   _inter_tree->zero();
 | |
|   TRectype::zero() ;
 | |
| } 
 | |
|   
 | |
| 
 | |
| /**************************************************************************************************/
 | |
| //  TABsaldo
 | |
| /**************************************************************************************************/
 | |
| TABsaldo::TABsaldo()
 | |
|              :TAlbero_AB(LF_ABSALDI), _saldi(NULL), _movdett(NULL)
 | |
| {
 | |
|   _saldi   = new TLocalisamfile(LF_ABSALDI);
 | |
|   _movdett = new TLocalisamfile(LF_MOVDETT);
 | |
| //  _relaz  = NULL;
 | |
| //  _ana    = NULL;
 | |
|   _inter_tree = new TLocal_balance3();
 | |
|   _newrec = new TRectype(LF_MOVDETT);
 | |
| }
 | |
| 
 | |
| TABsaldo::~TABsaldo()
 | |
| {
 | |
|   delete _saldi;
 | |
|   delete _movdett;
 | |
|   delete _newrec;
 | |
|   delete _inter_tree;
 | |
|   _saldi = NULL;
 | |
|   _movdett = NULL;
 | |
|   _newrec = NULL;
 | |
|   _inter_tree = NULL;
 | |
| }
 | |
| 
 | |
| void TABsaldo::put_headkey(TRectype &rec) const
 | |
| {
 | |
|   rec.put(ABMD_CODDITTA, get(ABS_CODDITTA));
 | |
|   rec.put(ABMD_ANNO, get(ABS_ANNO));
 | |
|   rec.put(ABMD_CODPDB, get(ABS_CODPDB));
 | |
|   rec.put(ABMD_TIPOBIL, get(ABS_TIPOBIL));
 | |
|   rec.put(ABMD_CODCBL, get(ABS_CODCBL));
 | |
| }
 | |
| 
 | |
| //Leggo il "corpo" del record
 | |
| int TABsaldo::read_body(bool lockstruct)
 | |
| {                      
 | |
|   TString codditta = get(ABS_CODDITTA);
 | |
|   TToken_string key(codditta.right_just(5,'0'));
 | |
|   key.add(get(ABS_ANNO));
 | |
|   key.add(get(ABS_CODPDB));
 | |
|   key.add(get(ABS_TIPOBIL));
 | |
|   key.add(get(ABS_CODCBL));
 | |
|   _mov = new TAlbero_movdett(key);
 | |
|   
 | |
|   if (_mov->goto_root())
 | |
|   {
 | |
|     naviga_movdett();
 | |
|   }
 | |
|   //_inter_tree->write_cache(); //solo per debug
 | |
|   //prova_inserimento(); //Solo debug per inserimento diretto di un nodo
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| //Naviga l'albero dei movimenti-dettagli
 | |
| void TABsaldo::naviga_movdett()
 | |
| {
 | |
|   TString key_rel;
 | |
|   _mov->curr_id(key_rel);
 | |
|   TNumeric_id lbrother_id, pather_id, my_id;
 | |
|    
 | |
|   my_id = _mov->curr().get_real(ABMD_ID);
 | |
|   pather_id = _mov->curr().get_real(ABMD_IDPADRE);
 | |
|   lbrother_id = _mov->curr().get_real(ABMD_IDPREC);
 | |
|   *_newrec = _mov->curr(); 
 | |
|   _inter_tree->insert_node(*_newrec,lbrother_id,pather_id,my_id);
 | |
|   
 | |
|   if (_mov->has_rbrother())
 | |
|   {
 | |
|     if (_mov->goto_rbrother())
 | |
|     { //Fratello di movdett trovato
 | |
|       naviga_movdett();
 | |
|       _mov->goto_node(key_rel);
 | |
|     }
 | |
|   }
 | |
|   
 | |
|   if (_mov->has_son())
 | |
|   {
 | |
|     if (_mov->goto_firstson())
 | |
|     { //Fratello di movdett trovato
 | |
|       naviga_movdett();
 | |
|       _mov->goto_node(key_rel);
 | |
|     }
 | |
|   }
 | |
| }           
 | |
| 
 | |
| 
 | |
| // Questa rudimentale funzione serve solo per il debug: richiama direttamente
 | |
| //la insert_node di un saldo IMMETTENDO un nuovo nodo nella struttura.
 | |
| // Cambiando i parametri id posiziona il nodo in posizione diverse
 | |
| void TABsaldo::prova_inserimento()
 | |
| { 
 | |
|   TRectype prova(LF_MOVDETT);
 | |
|   TNumeric_id lbrother_id,pather_id,my_id;
 | |
|   lbrother_id = 13;
 | |
|   pather_id = 0;
 | |
|   prova.put(ABMD_CODDITTA,"00001");
 | |
|   prova.put(ABMD_ANNO,"1996");
 | |
|   prova.put(ABMD_CODPDB,"001");
 | |
|   prova.put(ABMD_TIPOBIL,"E2");
 | |
|   prova.put(ABMD_CODCBL,"000000000200");
 | |
|   _inter_tree->insert_node(prova,lbrother_id,pather_id,my_id,IMMESSO);
 | |
| }
 | |
| 
 | |
| /**************************************************************************************************/
 | |
| //  TAnalisi_bil
 | |
| /**************************************************************************************************/
 | |
| 
 | |
| 
 | |
| TAnalisi_bil::TAnalisi_bil()
 | |
|              :TAlbero_AB(LF_ANALISI), _voci(NULL), _analisi(NULL)
 | |
| {
 | |
|   _voci   = new TLocalisamfile(LF_VOCI);
 | |
|   _analisi= new TLocalisamfile(LF_ANALISI);
 | |
|   _relaz  = NULL;
 | |
|   _ana    = NULL;
 | |
|   _inter_tree = new TLocal_relana3();
 | |
|   _inter_tree_relvoci = new TAlbero_locale_relvoci();
 | |
|   _newrec = new TRectype(LF_RELANA);
 | |
| 
 | |
| }
 | |
| 
 | |
| 
 | |
| TAnalisi_bil::~TAnalisi_bil() 
 | |
| {
 | |
|   delete _newrec;
 | |
|   delete _inter_tree;
 | |
|   _analisi = NULL;
 | |
|   _voci = NULL;
 | |
|   _newrec = NULL;
 | |
|   _inter_tree = NULL;
 | |
| }
 | |
| 
 | |
| void TAnalisi_bil::put_headkey(TRectype &rec) const
 | |
| {
 | |
|     rec.put(ABAN_CODAN, get(ABAN_CODAN));
 | |
| }
 | |
| //Scrive tutto il contenuto della cache sull'isamfile, aggiungendo, modificando
 | |
| //o rimuovendo i nodi necessari
 | |
| 
 | |
| int TAlbero_AB::commit_body() const
 | |
| {                         
 | |
|   TRectype rec(_inter_tree->current().num());
 | |
|   TLocalisamfile filebody(rec.num());
 | |
|   TString status;
 | |
|   //Estraggo un nodo per volta basandomi sull'assoc_array degli status  
 | |
|   while (_inter_tree->dirty_nodes() > 0)                                              
 | |
|   {
 | |
|     TString id = ID_NULLO_STR;
 | |
|     int prova = _inter_tree->dirty_nodes(); //solo debug
 | |
| 
 | |
|     rec= _inter_tree->extract_dirtynode(id, status);
 | |
|     put_headkey(rec);
 | |
|     
 | |
|     //Ho un rectype e uno status node
 | |
|     if (status == NODO_AGGIUNTO)
 | |
|     { 
 | |
|       if (filebody.curr().empty())
 | |
|       { //Se il cursor di relana è posizionato in modo errato:
 | |
|         //lo sistemo sull'ultimo record del file
 | |
|         //DOMANDA: - Perchè ???
 | |
|         //TENATATIVO DI RISPOSTA: - Quando l'ultima azione effettuata (prima di questa "aggiunta") è la rimozione 
 | |
|         //  dell'ultimo record probabilmente (ma non sono sicuro) il cursore non è posizionato (o forse
 | |
|         //  è posizionato in modo errato come ad es. dopo l'EOF, visto che l'EOF era sull'ultimo record rimosso)
 | |
|         //  e quando eseguo un tentativo di scrittura da un errore di chiave duplicata (anche se questa non esiste);
 | |
|         //  In tutti gli altri casi (quando l'ultima azione è stata una write o una rewrite o una remove di un record
 | |
|         //  che non sia l'ultimo) questo problema non esiste.
 | |
|         //RISPOSTA ESATTA: - Bho !!!
 | |
|         filebody.last();
 | |
|       }      
 | |
|       int err = filebody.write(rec);
 | |
|       if (err != NOERR)
 | |
|         return err;
 | |
|     }
 | |
|     if (status == NODO_MODIFICATO)
 | |
|     {            
 | |
|       int err = filebody.rewrite(rec);
 | |
|       if (err != NOERR)
 | |
|         return err;
 | |
|     }
 | |
|     if (status == NODO_RIMOSSO)
 | |
|     {
 | |
|       filebody.read(rec); //Devo leggere sul file per posizionare il cursore: stesso motivo di prima
 | |
|       int err = filebody.remove(rec);
 | |
|       if (err != NOERR)
 | |
|         return err;
 | |
|     }
 | |
|     // In tutti gli altri casi il nodo non viene modificato
 | |
|   }
 | |
|   return NOERR;
 | |
| }
 | |
| 
 | |
| //Leggo il "corpo" del record
 | |
| int TAnalisi_bil::read_body(bool lock_struct )
 | |
| {   
 | |
|   //Vuoto la cache: da eventuali residui
 | |
|   _inter_tree->clear_cache();
 | |
|   if (_ana) //Azzero l'id del TAlbero_relana: deve essere ricostruito
 | |
|     _ana->zero_id();
 | |
|   if (_relaz) 
 | |
|   {
 | |
|     delete _relaz;
 | |
|     delete _ana;
 | |
|     _relaz=NULL;
 | |
|     _ana=NULL;
 | |
|     
 | |
| /*    if (_relaz->codtab() != get(ABAN_CODAN))
 | |
|     {                 
 | |
|       delete _relaz;
 | |
|       delete _ana;
 | |
|       _relaz=NULL;
 | |
|       _ana=NULL;
 | |
|     } */
 | |
|   }
 | |
|   if (!_relaz) 
 | |
|   {
 | |
|     _relaz  = new TAlbero_relvoci(get(ABAN_CODAN));
 | |
|     _ana    = new TAlbero_relana(get(ABAN_CODAN));
 | |
|   }                    
 | |
|   
 | |
|   // carica la copia locale
 | |
|   TNumeric_id id_relana; 
 | |
|   if (lock_struct)
 | |
|     _relaz->lock();
 | |
| 
 | |
|   if (_relaz->goto_root())
 | |
|   {
 | |
|     if (_ana->goto_root())
 | |
|       id_relana=_ana->curr().get_real(ABRA_ID);
 | |
|     naviga_relazioni(id_relana,ID_NULLO,ID_NULLO);
 | |
|   }
 | |
|   //prova_inserimento();  //Solo debug per inserimento diretto di un nodo
 | |
|   _inter_tree->write_cache(); //solo debug
 | |
|   _inter_tree_relvoci->write_cache(); //solo debug
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| //Naviga l'albero delle relazioni
 | |
| void TAnalisi_bil::naviga_relazioni(const TNumeric_id & begin_relana, const TNumeric_id & id_prec,const TNumeric_id & id_padre)
 | |
| {
 | |
|   TNumeric_id id_relana, currid;
 | |
|   TString key_rel;
 | |
|   _relaz->curr_id(key_rel);
 | |
|   _newrec = sincronizza_relana(begin_relana, id_relana);
 | |
|   currid=_newrec->get_real(ABRA_ID);
 | |
|   // Inserico relana
 | |
|   _inter_tree->insert_node(*_newrec,id_prec,id_padre,currid);
 | |
|   // Inserico relvoci 
 | |
|   _inter_tree_relvoci->insert_node(_relaz->curr(),ID_NULLO,ID_NULLO,ID_NULLO);
 | |
|   //Controllo se c'è un fratello di rel_voci
 | |
|   if (_relaz->has_rbrother())
 | |
|   {
 | |
|     if (_relaz->goto_rbrother())
 | |
|     { //Fratello di rel_voci trovato
 | |
|       //Cerco un fratello anche su rel_ana: se non si trova c'è una inconsistenza nei dati
 | |
|       //fra relvoci e relana
 | |
|       TNumeric_id frat_relana;
 | |
|       naviga_relazioni(frat_relana,currid , id_padre);
 | |
|       _relaz->goto_node(key_rel);
 | |
|     }
 | |
|     else
 | |
|     {//Fratello di rel_voci non  trovato
 | |
|       error_box ("Errore di consistenza nei dati di relvoci: è previsto un fratello ma non è stato trovato");
 | |
|     }  
 | |
|   } // Il record di _relaz non ha un fratello destro
 | |
|   
 | |
|   _inter_tree->link_succ_zero(currid);  //Azzera link fratello
 | |
|   
 | |
|   
 | |
|   if (_relaz->goto_firstson())
 | |
|   { //Figlio di rel_voci trovato
 | |
|     //Cerco il figlio di rel_ana: se non si trova c'è una inconsistenza nei dati
 | |
|     //fra relvoci e relana
 | |
|     if (id_relana!=ID_NULLO)
 | |
|     { 
 | |
|       _ana->goto_id(id_relana);
 | |
|       if (_ana->goto_firstson())
 | |
|         id_relana=_ana->curr().get_real(ABRA_ID);
 | |
|     } 
 | |
|     naviga_relazioni(id_relana, ID_NULLO , currid);
 | |
|   }//Se non trovo il figlio di una relazione, potrebbe essere una foglia di un ramo dell'albero
 | |
|   _relaz->goto_node(key_rel); //Per risalire al nodo che mi ha chiamato in relvoci
 | |
|   
 | |
|   _inter_tree->link_son_zero(currid); //azzera link filgio
 | |
| }
 | |
|                        
 | |
| //Sincronizza relana con quello che trovo in relaz: relaz è navigato da naviga_relazioni
 | |
| TRectype *TAnalisi_bil::sincronizza_relana(const TNumeric_id &begin_relana,TNumeric_id &id_relana)
 | |
| {
 | |
|   id_relana=begin_relana;
 | |
|   TToken_string key;
 | |
|   // ****** caricamento array dei nodi di questo livello dall'isam file LF_RELANA ....
 | |
|   if (!_relaz->has_lbrother())
 | |
|   {                                                  
 | |
|     key = _relaz->curr().get(ABRL_CODVC);
 | |
|     if (id_relana != ID_NULLO)
 | |
|     { //carica l'array con i nodi componenti di questa voce (analisi)
 | |
|       _ana->goto_id(id_relana);
 | |
|       _nodifratelli.destroy();
 | |
|       do
 | |
|       {
 | |
|         TString codvc =_ana->curr().get(ABRA_CODVC);
 | |
|         TString idcomp =_ana->curr().get(ABRA_NCOMP);
 | |
|         _nodifratelli.setkey(codvc,idcomp,key);
 | |
|         _nodifratelli.add(key,_ana->curr());
 | |
|       } while (_ana->has_rbrother() && _ana->goto_rbrother());
 | |
|     }
 | |
|   }
 | |
|   
 | |
|   // ****** ... ricerca il nodo letto da LF_RELVOCI sull'array del livello ...
 | |
|   // rintraccia il record di relana sull'array 
 | |
|   TString codvc = _relaz->curr().get(ABRL_CODVC);
 | |
|   TString idcomp = _relaz->curr().get(ABRL_IDCOMP);
 | |
|   if (_nodifratelli.find1stkey(codvc,idcomp, key))
 | |
|   {
 | |
|     *_newrec = ((TRectype &)_nodifratelli[key]);
 | |
|     _nodifratelli.mark_used(key); //segna il record nell'array come utilizzato
 | |
|     id_relana=_newrec->get_real(ABRA_ID);
 | |
|   }
 | |
|   else
 | |
|   {//Il record viene rigenerato con i dati minimi per ripristinare l'integrità degli archivi
 | |
|     TLocalisamfile voc(LF_VOCI);
 | |
|     _newrec->zero(); //Azzero l'archivio
 | |
|     //Lo riempio con i dati basilari
 | |
|     _newrec->put(ABRA_CODAN,get(ABAN_CODAN));
 | |
|     _newrec->put(ABRA_ID,ID_NULLO);                 
 | |
|     _newrec->put(ABRA_TIPOCOD,_relaz->curr().get(ABRL_TIPOCOD));
 | |
|     _newrec->put(ABRA_CODVC,_relaz->curr().get(ABRL_CODVC));
 | |
|     _newrec->put(ABRA_NCOMP,_relaz->curr().get(ABRL_IDCOMP));
 | |
|     // Prelevo la descrizione di default
 | |
|     voc.put(ABVC_CODVC,_relaz->curr().get(ABRL_CODCOMP));
 | |
|     voc.read();
 | |
|     _newrec->put(ABRA_DESCRIZ,voc.get(ABVC_DESCR));
 | |
|     id_relana = ID_NULLO; // Serve per eseguire correttamente la new_id
 | |
|     _newrec->put(ABRA_ID,_ana->new_id(id_relana)); //Questa è stata spostata alla insert_node
 | |
|     _inter_tree->set_status_node(_newrec->get(ABRA_ID),NODO_AGGIUNTO);
 | |
|   }
 | |
|   
 | |
|   // ******** ...scarica l'array
 | |
|   if (!_relaz->has_rbrother()) // ultimo figlio
 | |
|   {
 | |
|     TObject *o;
 | |
|     _nodifratelli.restart();
 | |
|     while ((o = _nodifratelli.get()) != NULL)
 | |
|     {
 | |
|       if (!_nodifratelli.used(o))  //Marca il record in più su relana come da cancellare
 | |
|         _inter_tree->set_status_node(((TRectype *)o)->get(ABRA_ID),NODO_RIMOSSO);
 | |
|     }
 | |
|   }
 | |
|   return _newrec;
 | |
| }
 | |
| 
 | |
| // ************
 | |
| // array interno
 | |
| 
 | |
| void TAnalisi_bil::TArray_fratelli::setkey(const char *tipocod, const char *codvc, TToken_string &key) const
 | |
| {           
 | |
|   key.cut(0);
 | |
|   key.add(tipocod);
 | |
|   key.add(codvc);
 | |
| }
 | |
| 
 | |
| bool TAnalisi_bil::TArray_fratelli::find1stkey(const char *tipocod, const char *codvc, TToken_string &key)
 | |
| {
 | |
|   TObject *o;
 | |
|   setkey(tipocod,codvc,key);
 | |
|   o = objptr((const char *)key);
 | |
|   return ((o != NULL) && (!((TRectype *)o)->empty()));
 | |
| }
 | |
| 
 | |
| void TAnalisi_bil::TArray_fratelli::mark_used(const char *key)
 | |
| { 
 | |
|   ((TRectype *)objptr(key))->zero();  // segna il record nell'array come utilizzato
 | |
| }
 | |
| 
 | |
| bool TAnalisi_bil::TArray_fratelli::used(TObject *o)
 | |
| { 
 | |
|   return ((TRectype *)o)->empty();
 | |
| }
 | |
| 
 | |
| // Questa rudimentale funzione serve solo per il debug: richiama direttamente
 | |
| //la insert_node di un saldo IMMETTENDO un nuovo nodo nella struttura.
 | |
| // Cambiando i parametri id posiziona il nodo in posizione diverse
 | |
| void TAnalisi_bil::prova_inserimento()
 | |
| { 
 | |
|   TRectype prova(LF_RELANA);
 | |
|   TNumeric_id lbrother_id,pather_id,my_id;
 | |
|   lbrother_id = 0;
 | |
|   pather_id = 14;
 | |
|   prova.put(ABRA_CODAN,"A1");
 | |
|   _inter_tree->insert_node(prova,lbrother_id,pather_id,my_id,IMMESSO);
 | |
| }
 | |
| 
 | |
| 
 | |
| /**************************************************************************************************/
 | |
| // TLocal_relana3 & TLocal_balance3
 | |
| /**************************************************************************************************/
 | |
| TAlbero_locale::TAlbero_locale(int filenum)
 | |
| 
 | |
| {
 | |
| // versione definitiva:
 | |
| //  _currnode = new TRectype(filenum);
 | |
| //  _local_cache = new TRWrecord_cache(new TIsamtempfile(filenum,"",TRUE,TRUE));
 | |
| 
 | |
| // debug mods:
 | |
|   _currnode = new TRectype(filenum);
 | |
|   
 | |
|   
 | |
|   
 | |
|   if (filenum==LF_RELANA)
 | |
|   {
 | |
|     _f= new TIsamtempfile(filenum,"/com/relana",FALSE,FALSE);
 | |
|   }
 | |
| 
 | |
|   if (filenum == LF_MOVDETT)
 | |
|   {
 | |
|     _f= new TIsamtempfile(filenum,"/com/movdett",FALSE,FALSE);
 | |
|   }
 | |
|   
 | |
|   if (filenum==LF_RELVOCI)
 | |
|   {
 | |
|     _f= new TIsamtempfile(filenum,"/com/relvoci",FALSE,FALSE);
 | |
|   }
 | |
|   
 | |
|   _local_cache = new TRWrecord_cache(_f);
 | |
|   
 | |
| }                       
 | |
| 
 | |
| TAlbero_locale::~TAlbero_locale()
 | |
| {               
 | |
|   delete _currnode;     
 | |
|   delete _local_cache;
 | |
|   delete _f;
 | |
|   _currnode = NULL;
 | |
|   _local_cache = NULL;
 | |
|   _f = NULL;
 | |
| }                       
 | |
| 
 | |
| TAlbero_locale_link::TAlbero_locale_link(int filenum) :
 | |
|   TAlbero_locale(filenum)
 | |
| {
 | |
| }
 | |
| 
 | |
| TAlbero_locale_relvoci::TAlbero_locale_relvoci() :
 | |
|   TAlbero_locale(LF_RELVOCI)
 | |
| {
 | |
| }
 | |
| 
 | |
| TLocal_relana3::TLocal_relana3() :
 | |
|   TAlbero_locale_link(LF_RELANA)
 | |
| { 
 | |
|   ;
 | |
| }                       
 | |
| 
 | |
| TLocal_relana3::~TLocal_relana3()
 | |
| {
 | |
| }
 | |
| 
 | |
| TLocal_balance3::TLocal_balance3() :
 | |
|   TAlbero_locale_link(LF_MOVDETT)
 | |
| {
 | |
|   ;
 | |
| }                       
 | |
| 
 | |
| TLocal_balance3::~TLocal_balance3()
 | |
| {
 | |
| }                       
 | |
|                 
 | |
| bool TLocal_relana3::get_description(TString& desc) const
 | |
| {                  
 | |
|   desc.cut(0);
 | |
| //  curr_id(desc); 
 | |
|   desc << " " << current().get(ABRA_DESCRIZ);
 | |
|   return desc.not_empty();
 | |
| }
 | |
| 
 | |
| bool TLocal_relana3::make_root(TBaseisamfile& saldi)
 | |
| {                        
 | |
|   TString descriz;
 | |
|   descriz << "-- " << saldi.get(ABAN_DESCRIZ);
 | |
|   goto_son_first_level();           //Vado sul primo figlio
 | |
|   TNumeric_id temp_id = current().get_real(id_fieldname());  // e ne estraggo l'id
 | |
|   current().zero();                //Vuoto il record
 | |
|   current().put(id_fieldname(),ID_NULLO);   //ID
 | |
|   current().put(idson_fieldname(),temp_id); //Assegno il link al primo figlio
 | |
|   current().put(ABRA_DESCRIZ,descriz);  //Aggiungo la descrizione alla radice
 | |
|   current_cache().put(current());    //Aggiungo alla cache il record della radice principale
 | |
|   return TRUE;  
 | |
| }
 | |
| 
 | |
| bool TLocal_balance3::get_description(TString& desc) const
 | |
| {  
 | |
|   curr_id(desc); 
 | |
| //  desc << "|" << curnode.get(); 
 | |
|   return desc.not_empty();
 | |
| }
 | |
| 
 | |
| void TLocal_balance3::put_headkey(TRectype & node) 
 | |
| {
 | |
|   TToken_string s(get_headkey());
 | |
|   node.put(ABMD_CODDITTA,s.get());
 | |
|   node.put(ABMD_ANNO,s.get());
 | |
|   node.put(ABMD_CODPDB,s.get());
 | |
|   node.put(ABMD_TIPOBIL,s.get());
 | |
|   node.put(ABMD_CODCBL,s.get());
 | |
| }
 | |
| 
 | |
| bool TLocal_balance3::make_root(TBaseisamfile& saldi)
 | |
| {
 | |
|   goto_son_first_level();           //Vado sul primo figlio
 | |
|   TNumeric_id temp_id = current().get_real(id_fieldname());  // e ne estraggo l'id
 | |
|   current().zero();                //Vuoto il record
 | |
|   current().put(id_fieldname(),ID_NULLO);
 | |
|   current().put(idson_fieldname(),temp_id); //Assegno il link al primo figlio
 | |
|   current_cache().put(current());    //Aggiungo alla cache il record della radice principale
 | |
|   return TRUE;  
 | |
| }
 | |
| 
 | |
| 
 | |
| /**************************************************************************************************/
 | |
| // TAlbero_locale
 | |
| /**************************************************************************************************/
 | |
| 
 | |
| 
 | |
| //Estraggo un nodo: se specifico l'id in ingresso estrae il nodo con quell'id;
 | |
| //altrimenti ne sceglie uno a caso tra quelli "dirty" 
 | |
| //in output ho il record e le informazioni per sapere cosa fare di quel record
 | |
| TRectype TAlbero_locale_link::extract_dirtynode(TString &id, TString &status)
 | |
| { 
 | |
|   TToken_string key = get_headkey();
 | |
|   TRectype rec(current());
 | |
|   THash_object *row_status;
 | |
|   
 | |
|   if (id == ID_NULLO_STR)
 | |
|   { //Gestione interna
 | |
|     _status_node.restart();
 | |
|     row_status = _status_node.get_hashobj();
 | |
|     status = (TString&)row_status->obj();
 | |
|     id = (TString)row_status->key();
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     status = *(TString*)_status_node.objptr(id);
 | |
|   }
 | |
|   
 | |
|   //Posso accedere alla cache (con la get) solo se il nodo non
 | |
|   //è stato rimosso: se il nodo è stato rimosso, non sarà presente
 | |
|   //nella cache, e quindi la get di un nodo che non esiste è un record
 | |
|   //vuoto; questo record vuoto causerà errori nel momento in cui chiamerò
 | |
|   //la flush per il distruttore; devo quindi evitare questo 
 | |
|   //caso l'utilizzo della get
 | |
|   if (status != NODO_RIMOSSO)  
 | |
|   {
 | |
|     key.add(id); //La chiave è completamente ricostruita per la ricerca sulla cache
 | |
|     rec = _local_cache->get(key); //passo la chiave (formata da codtab e id) e ritorna il rectype
 | |
|   }
 | |
|   else
 | |
|   { //Riempio manualmente il record che deve essere rimosso: inserisco solo la chiave
 | |
|     //rec.put(ABRA_CODAN, ??? );
 | |
|     rec.put(id_fieldname(),id);
 | |
|   }
 | |
|   //A questo punto il record è completo e so già quale operazione devo compierci sopra
 | |
|   
 | |
|   //Questa parte serve per gestire l'assoc_array degli status
 | |
|   if ((rec.empty()) && (status != NODO_RIMOSSO))
 | |
|   { //Chiave non trovata: la chiave ricercata non è presente nella cache
 | |
|     fatal_box("Errore nella lettura della cache o nella costruzione del record.");
 | |
|   }
 | |
|   _status_node.remove(id);
 | |
|   return rec; //Ritorno un Trectype e passo lo stauts node
 | |
| }
 | |
| 
 | |
| //Setta lo stato del nodo: cioè decide che cosa dovrò fare con questo nodo
 | |
| void TAlbero_locale::set_status_node(const TString &id, const char *status)
 | |
| {
 | |
|   // STATUS A priorità assoluta e rimane sempre A
 | |
|   // STATUS M priorità su R ma non su A
 | |
|   // STATUS R priorità sul vuoto ma non su M o A
 | |
|   // STATUS vuoto: nessuna priorità
 | |
| 
 | |
|   // In TInser_albero::insert_node  posso capire solo A o M ma non R
 | |
|   // In TAlbero_relana::sincronizza posso capire solo R ma non A e M
 | |
|   TString key = id;
 | |
|   const TString *prec_status = (const TString *)_status_node.objptr(key);
 | |
|   
 | |
|   if (!prec_status)
 | |
|   { //Elemento ancora non presente nell'array
 | |
|     _status_node.add(key,TString (status));
 | |
|   }
 | |
|   else
 | |
|   { //Elemento già inserito
 | |
|     if (stricmp(status,NODO_AGGIUNTO)==0 && (*prec_status != NODO_AGGIUNTO))
 | |
|       _status_node.add(key,TString (status),TRUE); //Forzo la sovrascrittura in ogni caso
 | |
|     
 | |
|     if (stricmp(status,NODO_MODIFICATO)== 0 && (*prec_status != NODO_AGGIUNTO))
 | |
|     {
 | |
|       //Se il nodo era già inserito e il suo flag è diverso da NODO_AGGIUNTO
 | |
|       //allora devo settare che questo nodo è modificato (se non lo è ancora)
 | |
|       if (*prec_status != NODO_MODIFICATO)
 | |
|       {
 | |
|         _status_node.add(key,TString (status),TRUE); //Forzo la sovrascrittura in ogni caso
 | |
|       } 
 | |
|       // Arriva qui se il nodo è inserito e il flag era già stato settato da una
 | |
|       //operazione precedente a NODO_MODIFICATO
 | |
|     }
 | |
|     
 | |
|     if (stricmp(status,NODO_RIMOSSO)==0 && *prec_status != NODO_RIMOSSO)
 | |
|     { //Sto rimuovento un nodo che prima era stato modificato
 | |
|       //message_box("Sto rimuovendo un nodo appena modificato e non ancora registrato");
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| //Restituisce un nuovo id
 | |
| TNumeric_id &TAlbero_locale_link::new_id(TNumeric_id id)
 | |
| {
 | |
|   if (id == ID_NULLO)
 | |
|     _last_insert_id=_last_insert_id+1;
 | |
|   else
 | |
|     _last_insert_id = max(_last_insert_id,id);
 | |
|   return _last_insert_id;
 | |
| }
 | |
| 
 | |
| //Inserisce il nodo (che ho in ingresso) nella struttura ad albero doppiamente linkata
 | |
| //di relana aggiornando tutti i link
 | |
| bool TAlbero_locale_link::insert_node(TRectype & node,const TNumeric_id id_prec, const TNumeric_id id_padre, TNumeric_id id, const char flag_it)
 | |
| {
 | |
|   const char* dirty_value=NODO_MODIFICATO;
 | |
|   TNumeric_id id_succ;
 | |
|   if (id == ID_NULLO) 
 | |
|   {
 | |
|     id = new_id(id); 
 | |
|     set_status_node(id.string(),NODO_AGGIUNTO);
 | |
|   } 
 | |
|   else
 | |
|     new_id(id);
 | |
| 
 | |
|   //Cerco il mio fratello precedente 
 | |
|   if (id_prec != ID_NULLO)
 | |
|   { //Aggiorno il successivo del mio precedente dicendogli che il suo successivo sono io
 | |
|     goto_node(id_prec);
 | |
|     if (_currnode->get_real(idsucc_fieldname()) != id)
 | |
|     {// Il nodo è effettivamente modificato: apporto le modifiche e setto lo status
 | |
|      // Se il nodo è IMMESSO devo mantenere il link con il successivo per mantenere integra la struttutra
 | |
|       if (flag_it == IMMESSO)
 | |
|       {
 | |
|         id_succ = _currnode->get_real(idsucc_fieldname());
 | |
|       }
 | |
|        _currnode->put(idsucc_fieldname(),id.string());
 | |
|       _local_cache->put(*_currnode);
 | |
|       set_status_node(id_prec.string(),dirty_value); //Setto l'eventuale flag di array modificato per il fratello
 | |
|     }
 | |
|   }
 | |
|   else
 | |
|   { //Se non ho il mio precedente, allora o sono una radice principale o divento il primo figlio:
 | |
|     if (id_padre != ID_NULLO)
 | |
|     { //Sono un primo figlio: rintraccio mio padre e gli dico che il suo primo figlio ora, sono io
 | |
|       goto_node(id_padre);
 | |
|       if(_currnode->get_real(idson_fieldname()) != id)
 | |
|       {// Il nodo è effettivamente modificato: apporto le modifiche e setto lo status
 | |
|         if (flag_it == IMMESSO)
 | |
|         {
 | |
|           id_succ = _currnode->get_real(idson_fieldname());
 | |
|         }
 | |
|         _currnode->put(idson_fieldname(),id.string());
 | |
|         _local_cache->put(*_currnode);
 | |
|         set_status_node(id_padre.string(),dirty_value); //Setto l'eventuale flag di array modificato per il padre
 | |
|       }
 | |
|     } // Sono una radice principale e quindi sono orfana di padre: posso essere solo un nuovo nodo  IMMESSO direttamente
 | |
|     else
 | |
|     {
 | |
|       if (flag_it == IMMESSO)
 | |
|       {
 | |
|         goto_son_first_level();
 | |
|         id_succ = _currnode->get_real(id_fieldname());
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   
 | |
|   // Se il nodo è immesso controllo che ci sia un link al successivo da ripristinare
 | |
|   if (flag_it == IMMESSO)
 | |
|   {
 | |
|     if (id_succ != ID_NULLO)
 | |
|     { //Esiste un nodo che prima era il successivo dell'attuale mio precedente: devo modificarlo per
 | |
|       //dirgli che ora il suo precedente sono io
 | |
|       goto_node(id_succ);
 | |
|       // Il mio successivo deve essere effettivamente modificato
 | |
|       if (_currnode->get_real(idprec_fieldname()) != id)
 | |
|       {
 | |
|         _currnode->put(idprec_fieldname(),id.string());
 | |
|         _local_cache->put(*_currnode);
 | |
|         set_status_node(id_succ.string(),dirty_value); //Setto l'eventuale flag di array modificato per il fratello
 | |
|       }// Altrimenti il nodo non è da modificare
 | |
|       
 | |
|       //Aggiorno me stesso
 | |
|       node.put(idsucc_fieldname(),id_succ.string());
 | |
|       node.put(idson_fieldname(),ID_NULLO);
 | |
|       //set_current(node);
 | |
|     }
 | |
|   }
 | |
|   //Ora aggiorno me stesso: solo quella parte che rimane da aggiornare
 | |
|   put_headkey(node);
 | |
|   node.put(id_fieldname(),id.string());
 | |
|   if ((node.get_real(idprec_fieldname()) != id_prec) ||
 | |
|       (node.get_real(idfather_fieldname()) != id_padre))
 | |
|   {// Il nodo è effettivamente modificato: apporto le modifiche e setto lo status
 | |
|     node.put(idprec_fieldname(),id_prec.string());
 | |
|     node.put(idfather_fieldname(),id_padre.string());
 | |
|     set_status_node(id.string(),dirty_value);
 | |
|   }
 | |
|   _local_cache->put(node);
 | |
|   goto_node(node);
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| 
 | |
| //Questa funzione aggiorna i link dei "parenti" di un nodo che sta per essere rimosso
 | |
| bool TAlbero_locale_link::delete_node(const TNumeric_id &id)
 | |
| {                 
 | |
|   TRectype parente(current());
 | |
|   TNumeric_id id_padre, id_figlio, id_prec, id_succ;
 | |
|   
 | |
|   goto_node(id);
 | |
|   TToken_string key;
 | |
|   
 | |
| //  id = _currnode->get_real(ABRA_ID);
 | |
|   id_padre = _currnode->get_real(idfather_fieldname());
 | |
|   id_figlio = _currnode->get_real(idson_fieldname());
 | |
|   id_prec = _currnode->get_real(idprec_fieldname());
 | |
|   id_succ = _currnode->get_real(idsucc_fieldname());
 | |
|   
 | |
| //  if (id_padre != ID_NULLO)
 | |
| //  {
 | |
| 
 | |
| // Il padre esiste sempre: al limite è la radice stessa
 | |
|   key.cut(0);
 | |
|   key = get_headkey();
 | |
|   key.add(id_padre.string());
 | |
|   parente=_local_cache->get(key); //padre
 | |
|   if (parente.get(idson_fieldname()) == id.string())
 | |
|   { //Il nodo che sta per essere rimosso è il primo figlio di un padre
 | |
|     parente.put(idson_fieldname(),id_succ.string());  //Aggiorno il padre
 | |
|     if (id_padre != ID_NULLO)
 | |
|     {
 | |
|       // Se l'id del padre = ID_NULLO vuol dire che è la radice e non devo settare
 | |
|       //nessun status perchè questa non dovrà essere salvata nel DB
 | |
|       set_status_node(id_padre.string(), NODO_MODIFICATO);       //Aggiorno lo status
 | |
|     }
 | |
|     _local_cache->put(parente);
 | |
|   }
 | |
| //  }
 | |
|   
 | |
|   //Ricerco il mio fratello precedente per dirgli che io non sono più
 | |
|   //suo fratello successivo, e gli dico anche quale sarà il suo successivo
 | |
|   if (id_prec != ID_NULLO)
 | |
|   {                     
 | |
|     key.cut(0);
 | |
|     key = get_headkey();
 | |
|     key.add(id_prec.string());
 | |
|     parente=_local_cache->get(key); //fratello sinistro
 | |
|     parente.put(idsucc_fieldname(),id_succ.string()); //Assegno al mio fratello sinistro il mio fratello destro
 | |
|     set_status_node(id_prec.string(),NODO_MODIFICATO);         //Aggiorno lo status
 | |
|     _local_cache->put(parente);
 | |
|   }
 | |
|   
 | |
|   //Dico a mio fratello successivo chi il suo precedente non sono più io,
 | |
|   //e gli dico che il suo precedente è il MIO precedente
 | |
|   if (id_succ != ID_NULLO)
 | |
|   {                         
 | |
|     key.cut(0);
 | |
|     key = get_headkey();
 | |
|     key.add(id_succ.string());
 | |
|     parente=_local_cache->get(key); //fratello destro
 | |
|     parente.put(idprec_fieldname(),id_prec.string()); //Assegno al mio fratello destro il mio fratello sinistro
 | |
|     set_status_node(id_succ.string(),NODO_MODIFICATO);         //Aggiorno lo status
 | |
|     _local_cache->put(parente);
 | |
|   }
 | |
|   
 | |
|   //I miei vicini mi hanno già escluso: non sono più raggiungibile
 | |
|   //Ricostruisco la chiave
 | |
|   //Elimino i miei link dai fratelli
 | |
|   key.cut(0);
 | |
|   key = get_headkey();
 | |
|   key.add(id.string());
 | |
|   //node.put(ABRA_IDPADRE,ID_NULLO);
 | |
|   _currnode->put(idprec_fieldname(),ID_NULLO);
 | |
|   _currnode->put(idsucc_fieldname(),ID_NULLO);
 | |
|   _local_cache->put(*_currnode);
 | |
|   //Elimino i miei figli
 | |
|   
 | |
|   TNumeric_id nextnode_id;
 | |
|   if (id_prec != ID_NULLO)
 | |
|   {
 | |
|     nextnode_id = id_prec;
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     if (id_succ != ID_NULLO)
 | |
|     {
 | |
|       nextnode_id = id_succ;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|       if (id_padre != ID_NULLO)
 | |
|       {
 | |
|         nextnode_id = id_padre;
 | |
|       }
 | |
|       else
 | |
|       {
 | |
|         CHECK (FALSE,"NON SO DOVE POSIZIONARMI !!!!");
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   remove_subtree(nextnode_id);
 | |
|   //A questo punto l'assoc_array degli status è già completamente aggiornato
 | |
|   //write_cache();  //Solo per debug
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| void TAlbero_locale_link::zero()
 | |
| {
 | |
|   delete_tree();  // cancella l'albero
 | |
|   _status_node.destroy(); 
 | |
|   _currnode->put(id_fieldname(),ID_NULLO);
 | |
| }
 | |
| 
 | |
| bool TAlbero_locale_link::delete_tree()
 | |
| {
 | |
|   //write_cache();
 | |
|   goto_son_first_level();
 | |
|   remove_subtree(ID_NULLO); //Non eliminando il link coi fratelli della radice, remove_subtree naviga tutto l'albero
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| //Questa funzione elimina il nodo corrente e tutti i nodi sottostanti 
 | |
| bool TAlbero_locale_link::remove_subtree(const TNumeric_id return_id)
 | |
| {                                   
 | |
|   TNumeric_id myself_id;
 | |
|   //cerco il figlio
 | |
|   if (has_son())
 | |
|   {                
 | |
|     myself_id = _currnode->get_real(id_fieldname());
 | |
|     goto_firstson();
 | |
|     remove_subtree(myself_id);
 | |
|   }
 | |
|   
 | |
|   //cerco il fratello
 | |
|   if (has_rbrother())
 | |
|   {
 | |
|     myself_id = _currnode->get_real(id_fieldname());
 | |
|     goto_rbrother();
 | |
|     remove_subtree(myself_id);
 | |
|   }                              
 | |
|   TNumeric_id id = _currnode->get(id_fieldname());
 | |
|   
 | |
|   //Questa parte esegue la gestione dell'assoc_array degli stati
 | |
|   //in modo tale da mantenerlo allineato con il contenuto della cache
 | |
|   TString *status;
 | |
|   status = (TString*)_status_node.objptr(id.string());
 | |
|   if ((status == NULL) || (*status != NODO_AGGIUNTO))
 | |
|   {//In questi due casi il nodo esisteva già nel file originario
 | |
|     _status_node.add(id.string(),(TString)NODO_RIMOSSO,TRUE); //Forzo la sovrascrittura in ogni caso
 | |
|   }
 | |
|   else
 | |
|   {//Nel caso di NODO_AGGIUNTO
 | |
|     _status_node.remove(id.string());
 | |
|   }
 | |
|   
 | |
|   if (return_id != ID_NULLO)
 | |
|     goto_node(return_id);
 | |
|   else
 | |
|     _currnode->put(id_fieldname(),ID_NULLO);
 | |
| 
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| bool TAlbero_locale_link::has_son() const
 | |
| {
 | |
|   return (_currnode->get_real(idson_fieldname()) != ID_NULLO);
 | |
| }
 | |
| 
 | |
| bool TAlbero_locale_link::goto_firstson()
 | |
| {                        
 | |
|   if (has_son())
 | |
|   {
 | |
|     goto_node(_currnode->get_real(idson_fieldname()));
 | |
|     return TRUE;
 | |
|   }
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| 
 | |
| bool TAlbero_locale_link::has_father() const
 | |
| {
 | |
| //  return (_currnode->get_real(idfather_fieldname()) != ID_NULLO); 
 | |
| //Se l'id = 0 allora sono la radice
 | |
|   return (_currnode->get_real(id_fieldname()) != ID_NULLO);
 | |
| }
 | |
| 
 | |
| bool TAlbero_locale_link::goto_father()
 | |
| {                        
 | |
|   if (has_father())
 | |
|   {
 | |
|     goto_node(_currnode->get_real(idfather_fieldname()));
 | |
|     return TRUE;
 | |
|   }
 | |
|   return FALSE;
 | |
| } 
 | |
| 
 | |
| bool TAlbero_locale_link::has_rbrother() const
 | |
| {
 | |
|   return (_currnode->get_real(idsucc_fieldname()) != ID_NULLO);
 | |
| }
 | |
| 
 | |
| bool TAlbero_locale_link::goto_rbrother()
 | |
| {
 | |
|   if (has_rbrother())
 | |
|   {
 | |
|     goto_node(_currnode->get_real(idsucc_fieldname()));
 | |
|     return TRUE;
 | |
|   }
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| bool TAlbero_locale_link::has_lbrother() const
 | |
| {
 | |
|   return (_currnode->get_real(idprec_fieldname()) != ID_NULLO);
 | |
| }
 | |
| 
 | |
| bool TAlbero_locale_link::goto_lbrother()
 | |
| {
 | |
|   if (has_lbrother())
 | |
|   {
 | |
|     goto_node(_currnode->get_real(idprec_fieldname()));
 | |
|     return TRUE;
 | |
|   }
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| bool TAlbero_locale_link::goto_root()
 | |
| {
 | |
|   if (_currnode->get_real(id_fieldname()) != ID_NULLO )
 | |
|   { //Significa che non sono sulla radice: sono su un livello sottostante
 | |
|     goto_son_first_level();
 | |
|     goto_father();
 | |
|   }
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| bool TAlbero_locale_link::goto_son_first_level()
 | |
| { //Il _current sarà posizionato sull'ultimo nodo che è stato visitato
 | |
|   if (current().get_real(id_fieldname())==ID_NULLO)
 | |
|   {
 | |
|     CHECK (FALSE,"Non so dove posizionarmi !*!*!");
 | |
|     return FALSE;
 | |
|   }
 | |
|   TToken_string key;
 | |
|   TNumeric_id id_prec, id_padre;
 | |
|   key=get_headkey();
 | |
|   key.add(_currnode->get(id_fieldname()));
 | |
|   do
 | |
|   {
 | |
|     _currnode->zero();
 | |
|     *_currnode = _local_cache->get(key);
 | |
|     id_prec = _currnode->get_real(idprec_fieldname());
 | |
|     id_padre = _currnode->get_real(idfather_fieldname());
 | |
|     if (id_prec != ID_NULLO)
 | |
|     {
 | |
|       key.cut(0);
 | |
|       key=get_headkey();
 | |
|       key.add(id_prec.string());
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|       if (id_padre != ID_NULLO)
 | |
|       {
 | |
|         key.cut(0);
 | |
|         key=get_headkey();
 | |
|         key.add(id_padre.string());
 | |
|       }
 | |
|     }
 | |
|   } while (id_prec != ID_NULLO || id_padre != ID_NULLO);
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| void TAlbero_locale_link::goto_node(const TNumeric_id id)
 | |
| {
 | |
| /*  if (id != ID_NULLO)
 | |
|   { */
 | |
|     TToken_string key;
 | |
|     key=get_headkey();
 | |
|     key.add(id.string());
 | |
|     *_currnode = _local_cache->get(key);
 | |
| /*  }
 | |
|   else
 | |
|     _currnode = NULL;*/
 | |
| }
 | |
| 
 | |
| void TAlbero_locale_link::link_succ_zero(const TNumeric_id currid) 
 | |
| { // Se il nodo è da rimuovere non eseguo nessuna operazione: potrebbe causre problemi alla cache
 | |
|   TString *status = get_status_node(currid.string());
 | |
| //  if ((status != NULL) && (*status != NODO_RIMOSSO)) 
 | |
|   if ((status == NULL) || (*status != NODO_RIMOSSO))
 | |
|   { //Posiziono il nodo che sto aggiornando
 | |
|     goto_node(currid);
 | |
|     //Tento di posizionarmi sul fratello se questo può avere senso
 | |
|     if (current().get_real(idsucc_fieldname()) != ID_NULLO)
 | |
|     { //Secondo il link, ci dovrebbe essere un fratello
 | |
|       status = get_status_node(current().get(idsucc_fieldname()));
 | |
|       if ((status != NULL) && (*status != NODO_RIMOSSO))
 | |
|       { // Entro qui solo se il nodo esite, e lo status è stato fissato ad un valore diverso dal RIMOSSO
 | |
|         goto_node(current().get_real(idsucc_fieldname()));
 | |
|         if (current().empty())
 | |
|         { // Se però non esiste modifico il record
 | |
|           goto_node(currid);
 | |
|           current().put(idsucc_fieldname(),ID_NULLO);
 | |
|           _local_cache->put(current());
 | |
|           set_status_node(currid.string(),NODO_MODIFICATO);
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void TAlbero_locale_link::link_son_zero(const TNumeric_id currid) 
 | |
| { // Se il nodo è da rimuovere non eseguo nessuna operazione: potrebbe causre problemi alla cache
 | |
|   TString *status = get_status_node(currid.string());
 | |
| //  if ((status != NULL) && (*status != NODO_RIMOSSO)) 
 | |
|   if ((status == NULL) || (*status != NODO_RIMOSSO))
 | |
|   { //Posiziono il nodo che sto aggiornando
 | |
|     goto_node(currid);
 | |
|     //Tento di posizionarmi sul fratello se questo può avere senso
 | |
|     if (current().get_real(idson_fieldname()) != ID_NULLO)
 | |
|     { //Secondo il link, ci dovrebbe essere un fratello
 | |
|       status = get_status_node(current().get(idson_fieldname()));
 | |
|       if ((status != NULL) && (*status != NODO_RIMOSSO))
 | |
|       { // Entro qui solo se il nodo esite, e lo status è stato fissato ad un valore diverso dal RIMOSSO
 | |
|         goto_node(current().get_real(idson_fieldname()));
 | |
|         if (current().empty())
 | |
|         { // Se però non esiste modifico il record
 | |
|           goto_node(currid);
 | |
|           current().put(idson_fieldname(),ID_NULLO);
 | |
|           _local_cache->put(current());
 | |
|           set_status_node(currid.string(),NODO_MODIFICATO);
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /**************************************************************************************************/
 | |
| //  TAlbero_relana
 | |
| /**************************************************************************************************/
 | |
| //In questo costruttore cerco di capire anche quale è l'ultimo id utilizzato
 | |
| TAlbero_relana::TAlbero_relana( const char * tabcode)
 | |
| {
 | |
|   _codtab=tabcode;
 | |
|   _relana=new TLocalisamfile(LF_RELANA);
 | |
|   _relana->put(ABRA_CODAN,_codtab);
 | |
|   _relana->put(ABRA_ID,MAX_ID_REL);
 | |
|   
 | |
|   //Trova l'ultimo id usato in relana: DA COMPLETARE !!!
 | |
|   int err = _relana->read(_isgteq);  //Codifica errori in TIsamerr
 | |
|   if (err == NOERR)
 | |
|   { // Ho trovato il primo della tabella successiva: leggendo il precedente
 | |
|     // ho l'ultimo record dalla tabella cercata
 | |
|     _relana->prev();
 | |
|     if (_relana->get(ABRA_CODAN) != _codtab)
 | |
|     { // Non esiste la tabella
 | |
|       ; //_last_id viene lasciato a 0
 | |
|     }
 | |
|     _last_id = _relana->get_real(ABRA_ID);
 | |
|   }
 | |
|   
 | |
|   if (err == _isemptyfile)
 | |
|   { //Il file è vuoto
 | |
|     ;//_last_id viene lasciato a 0
 | |
|   }
 | |
|   
 | |
|   if (err == _iseof)
 | |
|   { // La tabella successiva non esiste: provo a leggere l'ultimo record
 | |
|     _relana->last();
 | |
|     if (_relana->get(ABRA_CODAN) != _codtab)
 | |
|     { //Tabella non trovata
 | |
|       ; //_last_id viene lasciato a 0
 | |
|     }
 | |
|     //Tabella trovata: prendo l'id dell'ultimo record
 | |
|     _last_id = _relana->get_real(ABRA_ID);
 | |
|   }
 | |
| }
 | |
| 
 | |
| TAlbero_relana::~TAlbero_relana() 
 | |
| {
 | |
|   delete _relana;
 | |
| }
 | |
|  
 | |
| //Incremento l'id per dare il primo id disponibile
 | |
| TNumeric_id & TAlbero_relana::new_id(TNumeric_id id)
 | |
| {      
 | |
|   if (id == ID_NULLO)
 | |
|     _last_id = _last_id+1;
 | |
|   else
 | |
|     _last_id = max(_last_id,id);
 | |
|   return _last_id;
 | |
| }
 | |
| 
 | |
| //Posiziona relana sul record voluto
 | |
| bool TAlbero_relana::goto_id(const TNumeric_id &id)
 | |
| {
 | |
|   _relana->put(ABRA_CODAN,_codtab);
 | |
|   _relana->put(ABRA_ID,id);
 | |
|   if (_relana->read() == NOERR)
 | |
|     return TRUE;
 | |
|   return FALSE; //Non sono riuscito a posizionarmi sul nodo
 | |
| } 
 | |
|  
 | |
| bool TAlbero_relana::lock()
 | |
| {
 | |
|   //Potrebbe usare come parametro _codtab
 | |
|   return TRUE;
 | |
| } 
 | |
| 
 | |
| void TAlbero_relana::unlock()
 | |
| {                                        
 | |
|   //Potrebbe usare come parametro _codtab
 | |
|   return ;
 | |
| }
 | |
| 
 | |
| //Ricerca la radice di relana
 | |
| bool TAlbero_relana::goto_root()
 | |
| {
 | |
|   _relana->put(ABRA_CODAN,_codtab); //Ricerca il codice della tabella: è l'unica cosa che conosco
 | |
|   //No so nulla sull'id
 | |
|   _relana->read(_isgteq); //Leggo la prima chiave che si avvicina a quella specificata
 | |
|   if (_relana->get(ABRA_CODAN) == _codtab)
 | |
|   {       
 | |
|     while (has_father())
 | |
|     { 
 | |
|       goto_father();
 | |
|     } // Ho raggiunto il primo livello
 | |
|     while (has_lbrother())
 | |
|     {
 | |
|       goto_lbrother();
 | |
|     }// Ho raggiunto la radice
 | |
|     return TRUE;
 | |
|   }        
 | |
|   //Non ho trovato una voce con codice tabella richiesta
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| //Dice se un nodo di relana ha il primo figlio
 | |
| bool TAlbero_relana::has_son() const
 | |
| {                                         
 | |
|   return _relana->get_real(ABRA_IDFIGLIO)!=ID_NULLO;
 | |
| }
 | |
| 
 | |
| //Ricerca il primo filglio di un nodo di relana
 | |
| bool TAlbero_relana::goto_firstson()
 | |
| {                                         
 | |
|   TNumeric_id id_figlio(_relana->get_real(ABRA_IDFIGLIO));
 | |
|   if (id_figlio > 0)
 | |
|   { //Ci deve essere un figlio: lo cerco
 | |
|     //Il CODAN deve rimanere uguale a quello del padre
 | |
|     _relana->put(ABRA_ID,_relana->get(ABRA_IDFIGLIO));
 | |
|     if (_relana->read() == NOERR)
 | |
|       return TRUE;  //Figlio trovato con successo
 | |
|   }
 | |
|   //Non c'è un figlio
 | |
|   return FALSE;
 | |
| }          
 | |
| 
 | |
| //Dice se un nodo di relana ha un fratello destro
 | |
| bool TAlbero_relana::has_rbrother() const 
 | |
| {              
 | |
|   return _relana->get_real(ABRA_IDSUCC) != ID_NULLO;
 | |
| }
 | |
| 
 | |
| //Ricerca il fratello destro di un nodo di relana
 | |
| bool TAlbero_relana::goto_rbrother() 
 | |
| {
 | |
|   TNumeric_id id_fratello(_relana->get_real(ABRA_IDSUCC));
 | |
|   if (id_fratello > 0)
 | |
|   { //Ci deve essere un fratello: lo cerco
 | |
|     //Il CODAN deve rimanere uguale a quello del fratello precedente
 | |
|     _relana->put(ABRA_ID,_relana->get(ABRA_IDSUCC));
 | |
|     if (_relana->read() == NOERR)
 | |
|       return TRUE; //Fratello trovato con successo
 | |
|   }
 | |
|   //Non c'è un fratello: non si dovrebbe mai verificare visto che
 | |
|   //si entra in questa funzione solo quando è previsto che ci sia un fratello.
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| //Dice se un nodo di relana ha un padre
 | |
| bool TAlbero_relana::has_father() const 
 | |
| { 
 | |
|   return _relana->get_real(ABRA_IDPADRE) != ID_NULLO;
 | |
| }
 | |
| 
 | |
| //Ricerca il padre del nodo di relana
 | |
| bool TAlbero_relana::goto_father()
 | |
| {
 | |
|   _relana->put(ABRA_ID,_relana->get(ABRA_IDPADRE));
 | |
|   if (_relana->read() != NOERR)
 | |
|     CHECK (FALSE,"ERRORE DI PADRE"); //*!*!*!*! DA gestire
 | |
|   else
 | |
|     return TRUE;
 | |
|   return FALSE;
 | |
| }
 | |
|  
 | |
| //Ritrona il nodo corrente di relana
 | |
| TObject *TAlbero_relana::curr_node() const 
 | |
| {
 | |
|   return &_relana->curr();
 | |
| }
 | |
| 
 | |
| //Dice se un nodo di relana ha un fratello sinistro
 | |
| bool TAlbero_relana::has_lbrother() const 
 | |
| {
 | |
|   return _relana->get_real(ABRA_IDPREC) != ID_NULLO;
 | |
| }
 | |
| 
 | |
| //Ricerca il fratello sinistro di un nodo di relana
 | |
| bool TAlbero_relana::goto_lbrother()
 | |
| {
 | |
|   TNumeric_id id_fratello(_relana->get_real(ABRA_IDPREC));
 | |
|   if (id_fratello > 0)
 | |
|   {//Ci deve essere un fratello: lo cerco
 | |
|     //Il CODAN deve rimanere uguale a quello del fratello successivo
 | |
|     _relana->put(ABRA_ID,_relana->get(ABRA_IDPREC));
 | |
|     if (_relana->read() == NOERR)
 | |
|       return TRUE; //Fratello trovato con successo
 | |
|   }
 | |
|   //ERRORE !!!
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| void TAlbero_relana::node2id(const TObject * node,TString & id) const
 | |
| {
 | |
|   id.format("%ld",(long)_relana->recno()); 
 | |
| }
 | |
| 
 | |
| /**************************************************************************************************/
 | |
| //  TAlbero_relvoci
 | |
| /**************************************************************************************************/
 | |
| TAlbero_relvoci::TAlbero_relvoci(const char * tabcode)
 | |
| {
 | |
|   _codtab=tabcode;
 | |
|   _relvoci=new TLocalisamfile(LF_RELVOCI);
 | |
| }
 | |
| 
 | |
| TAlbero_relvoci::~TAlbero_relvoci() 
 | |
| {
 | |
|   delete _relvoci;
 | |
| }
 | |
|   
 | |
| //Ricerca la radice di relvoci
 | |
| bool TAlbero_relvoci::goto_root()
 | |
| {
 | |
|   _relvoci->put(ABRL_TIPOCOD,TIPO_ANALISI);             // La radice ha tipocod = "A"
 | |
|   TString codtab = _codtab;                           //Mi serve per eseguire una formattazione senza modificare _codtab
 | |
|   _relvoci->put(ABRL_CODVC,codtab.right_just(10,' '));  //Inserisco il codice della tabella su cui sto lavorando
 | |
|   //No so nulla sull'id
 | |
|   _relvoci->read(_isgteq);                              //Leggo la prima chiave che si avvicina a quella specificata
 | |
|   if ((_relvoci->get(ABRL_TIPOCOD) == TIPO_ANALISI) || (_relvoci->get(ABRL_CODVC) == _codtab))
 | |
|   {
 | |
|     while (has_lbrother())
 | |
|     {
 | |
|       goto_lbrother();
 | |
|     }
 | |
|     return TRUE;
 | |
|   }                                                     
 | |
|   //Non ho trovato una voce di analisi richiesta o
 | |
|   //non ho trovato una voce con codice tabella richiesta
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| bool TAlbero_relvoci::lock()
 | |
| {
 | |
|   //Potrebbe usare come parametro _codtab
 | |
|   return TRUE;
 | |
| } 
 | |
| 
 | |
| 
 | |
| void TAlbero_relvoci::unlock()
 | |
| {                                        
 | |
|   //Potrebbe usare come parametro _codtab
 | |
|   return ;
 | |
| }
 | |
| 
 | |
| //Dice se il nodo ha un primo figlio; DA DEFINIRE
 | |
| bool TAlbero_relvoci::has_son() const
 | |
| {
 | |
|   CHECK(FALSE,"!!!");
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| //Ricerca il primo figlio di un nodo di relvoci
 | |
| bool TAlbero_relvoci::goto_firstson()
 | |
| {               
 | |
|   TRecnotype temp_pos = _relvoci->recno();
 | |
|   
 | |
|   _relvoci->put(ABRL_TIPOCOD,TIPO_VOCE); //Il tipo codice del filgio di una radice o di un livello inferiore è sempre "V"
 | |
|   TString controllo_comp = _relvoci->get(ABRL_CODCOMP);
 | |
|   _relvoci->put(ABRL_CODVC,controllo_comp); //Setto il figlio
 | |
|   //Non posso sapere nulla sull'id corretto: lo azzero perchè il record contiene già l'id del record letto in precedenza
 | |
|   //e leggo la prima chiave che assomiglia a quella settata
 | |
|   _relvoci->put(ABRL_IDCOMP,0);
 | |
|   _relvoci->read(_isgteq); //Leggo la prima chiave che si avvicina a quella specificata
 | |
|   
 | |
|   if ((_relvoci->get(ABRL_TIPOCOD) == TIPO_VOCE) && (_relvoci->get(ABRL_CODVC) == controllo_comp))
 | |
|   {//Ho trovato un figlio, ma non so se è proprio il primo figlio:
 | |
|    //se non sono sul primo figlio navigo all'indietro
 | |
|     while (has_lbrother())
 | |
|     {
 | |
|       goto_lbrother();
 | |
|     }
 | |
|     return TRUE;
 | |
|   }
 | |
|   else
 | |
|   { //Non ho trovato il filgio: mi riposiziono alla posizione precedente
 | |
|     //Questo non rappresenta un errore perchè potrebbe essere 
 | |
|     //la foglia di un ramo dell'albero
 | |
|     _relvoci->readat(temp_pos);
 | |
|     return FALSE;
 | |
|   }
 | |
| }
 | |
| 
 | |
| //Dice su un nodo di relvoci ha un fratello destro
 | |
| bool TAlbero_relvoci::has_rbrother() const 
 | |
| {                                 
 | |
|   return _relvoci->get_long(ABRL_NEXTCOMP) != ID_NULLO;  
 | |
| }
 | |
| 
 | |
| //Ricerca il fratello destro di un nodo di relvoci
 | |
| bool TAlbero_relvoci::goto_rbrother()
 | |
| {             
 | |
|   //Il tipocod di un fratello deve essere uguale a quello attuale
 | |
|   //Il codice voce di un fratello deve essere uguale a quella attuale
 | |
|   _relvoci->put(ABRL_IDCOMP,_relvoci->get(ABRL_NEXTCOMP)); //Assegno all'id il successivo
 | |
|   if (_relvoci->read() == NOERR)
 | |
|     return TRUE; //Fratello trovato
 | |
|   else
 | |
|     return FALSE; //Fratello non trovato
 | |
| }
 | |
| 
 | |
| //Dice se un nodo di relvoci ha un fratello sinistro
 | |
| bool TAlbero_relvoci::has_lbrother() const 
 | |
| {                                 
 | |
|   return _relvoci->get_long(ABRL_PREVCOMP) != ID_NULLO;  
 | |
| }
 | |
| 
 | |
| //Ricerca il fratello sinistro di un nodo di relvoci
 | |
| bool TAlbero_relvoci::goto_lbrother()
 | |
| {             
 | |
|   //Il tipocod di un fratello deve essere uguale a quello attuale
 | |
|   //Il codice voce di un fratello deve essere uguale a quella attuale
 | |
|   _relvoci->put(ABRL_IDCOMP,_relvoci->get(ABRL_PREVCOMP)); //Assegno all'id il successivo
 | |
|   if (_relvoci->read() == NOERR)
 | |
|     return TRUE; //Fratello trovato
 | |
|   else
 | |
|     return FALSE; //Fratello non trovato
 | |
| }  
 | |
| 
 | |
| //Ritorna il nodo corrente di relvoci
 | |
| TObject *TAlbero_relvoci::curr_node() const 
 | |
| {
 | |
|   return &_relvoci->curr();
 | |
| }
 | |
| 
 | |
| void TAlbero_relvoci::node2id(const TObject * node,TString &  id) const
 | |
| {
 | |
|   id.format("%ld",_relvoci->recno()); 
 | |
| }
 | |
| 
 | |
| /**************************************************************************************************/
 | |
| //  TAlbero_movdett
 | |
| /**************************************************************************************************/
 | |
| TAlbero_movdett::TAlbero_movdett( const char * tabcode)
 | |
| {
 | |
|   _codtab = tabcode;
 | |
|   _movdett=new TLocalisamfile(LF_MOVDETT);
 | |
|   
 | |
| /*  _movdett->put(ABMD_CODDITTA,_codtab.get(0));
 | |
|   _movdett->put(ABMD_ANNO,_codtab.get(1));
 | |
|   _movdett->put(ABMD_CODPDB,_codtab.get(2));
 | |
|   _movdett->put(ABMD_TIPOBIL,_codtab.get(3));
 | |
|   _movdett->put(ABMD_CODCBL,_codtab.get(4));
 | |
|   _movdett->put(ABMD_ID,MAX_ID_MOVDETT);
 | |
| 
 | |
|   //Trova l'ultimo id usato in MOVDETT: DA COMPLETARE !!!
 | |
|   int err = _movdett->read(_isgteq);  //Codifica errori in TIsamerr
 | |
|   if (err == NOERR)
 | |
|   { // Ho trovato il primo della "categoria" successiva: leggendo il precedente
 | |
|     // ho l'ultimo record cercato
 | |
|     _movdett->prev();
 | |
|     TString16 format =_movdett->get(ABMD_CODDITTA);
 | |
|     TToken_string controllo = format.right_just(5,'0');
 | |
|     controllo.add(_movdett->get(ABMD_ANNO));
 | |
|     controllo.add(_movdett->get(ABMD_CODPDB));
 | |
|     controllo.add(_movdett->get(ABMD_TIPOBIL));
 | |
|     controllo.add(_movdett->get(ABMD_CODCBL));
 | |
|     
 | |
|     if (controllo != _codtab)
 | |
|     { // Non esiste la tabella
 | |
|       ; //_last_id viene lasciato a 0
 | |
|     }
 | |
|     _last_id = _movdett->get_real(ABMD_ID);
 | |
|   }
 | |
|   
 | |
|   if (err == _isemptyfile)
 | |
|   { //Il file è vuoto
 | |
|     ;//_last_id viene lasciato a 0
 | |
|   }
 | |
|   
 | |
|   if (err == _iseof)
 | |
|   { // La tabella successiva non esiste: provo a leggere l'ultimo record
 | |
|     _movdett->last();
 | |
|     TString16 format =_movdett->get(ABMD_CODDITTA);
 | |
|     TToken_string controllo = format.right_just(5,'0');
 | |
|     controllo.add(_movdett->get(ABMD_ANNO));
 | |
|     controllo.add(_movdett->get(ABMD_CODPDB));
 | |
|     controllo.add(_movdett->get(ABMD_TIPOBIL));
 | |
|     controllo.add(_movdett->get(ABMD_CODCBL));
 | |
|     if (controllo != _codtab)
 | |
|     { //Tabella non trovata
 | |
|       ; //_last_id viene lasciato a 0
 | |
|     }
 | |
|     //Tabella trovata: prendo l'id dell'ultimo record
 | |
|     _last_id = _movdett->get_real(ABMD_ID);
 | |
|   } */
 | |
| }
 | |
| 
 | |
| TAlbero_movdett::~TAlbero_movdett()
 | |
| {
 | |
|   delete _movdett;
 | |
| }
 | |
| 
 | |
| 
 | |
| //Posiziona relana sul record voluto
 | |
| bool TAlbero_movdett::goto_id(const TNumeric_id &id)
 | |
| {
 | |
|   _movdett->put(ABMD_CODDITTA,_codtab);
 | |
|   _movdett->put(ABMD_ID,id);
 | |
|   if (_movdett->read() == NOERR)
 | |
|     return TRUE;
 | |
|   return FALSE; //Non sono riuscito a posizionarmi sul nodo
 | |
| } 
 | |
|  
 | |
| //Ricerca la radice di relana
 | |
| bool TAlbero_movdett::goto_root()
 | |
| {
 | |
|   _movdett->put(ABMD_CODDITTA,_codtab.get(0));
 | |
|   _movdett->put(ABMD_ANNO,_codtab.get(1));
 | |
|   _movdett->put(ABMD_CODPDB,_codtab.get(2));
 | |
|   _movdett->put(ABMD_TIPOBIL,_codtab.get(3));
 | |
|   _movdett->put(ABMD_CODCBL,_codtab.get(4));
 | |
|   //No so nulla sull'id
 | |
|   _movdett->read(_isgteq); //Leggo la prima chiave che si avvicina a quella specificata
 | |
|   TString fctrl = _movdett->get(ABMD_CODDITTA);
 | |
|   TToken_string controllo = fctrl.right_just(5,'0');
 | |
|   controllo.add(_movdett->get(ABMD_ANNO));
 | |
|   controllo.add(_movdett->get(ABMD_CODPDB));
 | |
|   controllo.add(_movdett->get(ABMD_TIPOBIL));
 | |
|   controllo.add(_movdett->get(ABMD_CODCBL));
 | |
|   
 | |
|   if (controllo == _codtab)
 | |
|   {       
 | |
|     while (has_father())
 | |
|     { 
 | |
|       goto_father();
 | |
|     } // Ho raggiunto il primo livello
 | |
|     while (has_lbrother())
 | |
|     {
 | |
|       goto_lbrother();
 | |
|     }// Ho raggiunto la radice
 | |
|     return TRUE;
 | |
|   }        
 | |
|   //Non ho trovato una voce con codice tabella richiesta
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| //Dice se un nodo di relana ha il primo figlio
 | |
| bool TAlbero_movdett::has_son() const
 | |
| {                                         
 | |
|   return _movdett->get_real(ABMD_IDFIGLIO)!=ID_NULLO;
 | |
| }
 | |
| 
 | |
| //Ricerca il primo filglio di un nodo di relana
 | |
| bool TAlbero_movdett::goto_firstson()
 | |
| {                                         
 | |
|   TNumeric_id id_figlio(_movdett->get_real(ABMD_IDFIGLIO));
 | |
|   if (id_figlio > 0)
 | |
|   { //Ci deve essere un figlio: lo cerco
 | |
|     //Il CODAN deve rimanere uguale a quello del padre
 | |
|     _movdett->put(ABMD_ID,_movdett->get(ABMD_IDFIGLIO));
 | |
|     if (_movdett->read() == NOERR)
 | |
|       return TRUE;  //Figlio trovato con successo
 | |
|   }
 | |
|   //Non c'è un figlio: non si dovrebbe mai verificare visto che
 | |
|   //si entra in questa funzione solo quando è previsto che ci sia un figlio.
 | |
|   return FALSE;
 | |
| }          
 | |
| 
 | |
| //Dice se un nodo di relana ha un fratello destro
 | |
| bool TAlbero_movdett::has_rbrother() const 
 | |
| {              
 | |
|   return _movdett->get_real(ABMD_IDSUCC) != ID_NULLO;
 | |
| }
 | |
| 
 | |
| //Ricerca il fratello destro di un nodo di relana
 | |
| bool TAlbero_movdett::goto_rbrother() 
 | |
| {
 | |
|   TNumeric_id id_fratello(_movdett->get_real(ABMD_IDSUCC));
 | |
|   if (id_fratello > 0)
 | |
|   { //Ci deve essere un fratello: lo cerco
 | |
|     //Il CODAN deve rimanere uguale a quello del fratello precedente
 | |
|     _movdett->put(ABMD_ID,_movdett->get(ABMD_IDSUCC));
 | |
|     if (_movdett->read() == NOERR)
 | |
|       return TRUE; //Fratello trovato con successo
 | |
|   }
 | |
|   //Non c'è un fratello: non si dovrebbe mai verificare visto che
 | |
|   //si entra in questa funzione solo quando è previsto che ci sia un fratello.
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| //Dice se un nodo di relana ha un padre
 | |
| bool TAlbero_movdett::has_father() const 
 | |
| { 
 | |
|   return _movdett->get_real(ABMD_IDPADRE) != ID_NULLO;
 | |
| }
 | |
| 
 | |
| //Ricerca il padre del nodo di relana
 | |
| bool TAlbero_movdett::goto_father()
 | |
| {
 | |
|   _movdett->put(ABMD_ID,_movdett->get(ABMD_IDPADRE));
 | |
|   if (_movdett->read() != NOERR)
 | |
|     CHECK (FALSE,"ERRORE DI PADRE"); //*!*!*!*! DA gestire
 | |
|   else
 | |
|     return TRUE;
 | |
|   return FALSE;
 | |
| }
 | |
|  
 | |
| //Ritrona il nodo corrente di relana
 | |
| TObject *TAlbero_movdett::curr_node() const 
 | |
| {
 | |
|   return &_movdett->curr();
 | |
| }
 | |
| 
 | |
| //Dice se un nodo di relana ha un fratello sinistro
 | |
| bool TAlbero_movdett::has_lbrother() const 
 | |
| {
 | |
|   return _movdett->get_real(ABMD_IDPREC) != ID_NULLO;
 | |
| }
 | |
| 
 | |
| //Ricerca il fratello sinistro di un nodo di relana
 | |
| bool TAlbero_movdett::goto_lbrother()
 | |
| {
 | |
|   TNumeric_id id_fratello(_movdett->get_real(ABMD_IDPREC));
 | |
|   if (id_fratello > 0)
 | |
|   {//Ci deve essere un fratello: lo cerco
 | |
|     //Il CODAN deve rimanere uguale a quello del fratello successivo
 | |
|     _movdett->put(ABMD_ID,_movdett->get(ABMD_IDPREC));
 | |
|     if (_movdett->read() == NOERR)
 | |
|       return TRUE; //Fratello trovato con successo
 | |
|   }
 | |
|   //ERRORE !!!
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| void TAlbero_movdett::node2id(const TObject * node,TString &  id) const
 | |
| {
 | |
|   id.format("%ld",_movdett->recno()); 
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /****************************************/
 | |
| /* TAlbero_locale_relvoci                 */
 | |
| /*********************************/
 | |
| TRectype TAlbero_locale_relvoci::extract_dirtynode(TString &id, TString &status)
 | |
| { 
 | |
|   TToken_string key_rel = (TToken_string&)id;
 | |
|   TRectype rec(current());
 | |
|   THash_object *row_status;
 | |
|   _status_node.restart();
 | |
|   row_status = _status_node.get_hashobj();
 | |
|   status = (TString&)row_status->obj();
 | |
|   key_rel = (TToken_string)row_status->key();
 | |
|   rec = current_cache().get(key_rel);
 | |
|   _status_node.remove(key_rel);
 | |
|   return rec; //Ritorno un Trectype e passo lo stauts node
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| bool TAlbero_locale_relvoci::has_lbrother() const
 | |
| {
 | |
|   return current().get_real(ABRL_PREVCOMP) != ID_NULLO; 
 | |
| }
 | |
| 
 | |
| bool TAlbero_locale_relvoci::goto_lbrother()
 | |
| {
 | |
|   TToken_string key_rel;
 | |
|   key_rel.add(current().get(ABRL_TIPOCOD));
 | |
|   key_rel.add(current().get(ABRL_CODVC));
 | |
|   key_rel.add(current().get(ABRL_PREVCOMP));
 | |
|   return goto_node(key_rel);
 | |
| }
 | |
| 
 | |
| bool TAlbero_locale_relvoci::has_rbrother() const
 | |
| {
 | |
|   return current().get_real(ABRL_NEXTCOMP) != ID_NULLO; 
 | |
| }
 | |
| 
 | |
| bool TAlbero_locale_relvoci::goto_rbrother()
 | |
| {
 | |
|   TToken_string key_rel;
 | |
|   key_rel.add(current().get(ABRL_TIPOCOD));
 | |
|   key_rel.add(current().get(ABRL_CODVC));
 | |
|   key_rel.add(current().get(ABRL_NEXTCOMP));
 | |
|   return goto_node(key_rel);
 | |
| }
 | |
| 
 | |
| 
 | |
| TNumeric_id TAlbero_locale_relvoci::new_id()
 | |
| {   
 | |
|   TNumeric_id max_id;
 | |
|   
 | |
|   // Vado sul primo figlio
 | |
|   while (has_lbrother())
 | |
|   {
 | |
|     goto_lbrother();
 | |
|   }
 | |
|     
 | |
|   max_id = current().get_real(ABRL_IDCOMP);
 | |
|   
 | |
|   while (has_rbrother())
 | |
|   {
 | |
|     goto_rbrother();
 | |
|     if ((current().get_real(ABRL_IDCOMP)) > max_id)
 | |
|       max_id = current().get_real(ABRL_IDCOMP);
 | |
|   }
 | |
|   return max_id + 1;
 | |
| }
 | |
| 
 | |
| bool TAlbero_locale_relvoci::insert_node(TRectype &node, const TNumeric_id id_prec, const TNumeric_id id_padre, TNumeric_id id, const char trasf_immes)
 | |
| {
 | |
|   if (trasf_immes == IMMESSO)
 | |
|   {
 | |
|     TNumeric_id prev_comp = current().get_real(ABRL_IDCOMP);
 | |
|     TNumeric_id next_comp = current().get_real(ABRL_NEXTCOMP);
 | |
|     TNumeric_id new_comp;
 | |
|     new_comp = new_id();    
 | |
|     
 | |
|     TToken_string key_rel;
 | |
|     // Mi riposiziono sul record PRECEDENTE  a quello che sto inserendo
 | |
|     key_rel.cut(0);
 | |
|     key_rel.add(current().get(ABRL_TIPOCOD));
 | |
|     key_rel.add(current().get(ABRL_CODVC));
 | |
|     key_rel.add(prev_comp.string());
 | |
|     goto_node(key_rel);
 | |
|       
 | |
|     // Creo il nuovo record di relvoci e lo inserisco
 | |
|     TRectype new_rel(current());
 | |
|     new_rel.zero();
 | |
|     new_rel.put(ABRL_TIPOCOD,current().get(ABRL_TIPOCOD));
 | |
|     new_rel.put(ABRL_CODVC,current().get(ABRL_CODVC));
 | |
|     new_rel.put(ABRL_IDCOMP,new_comp.string());
 | |
|     new_rel.put(ABRL_PREVCOMP,prev_comp.string());
 | |
|     new_rel.put(ABRL_NEXTCOMP,next_comp.string());
 | |
|     _local_cache->put(new_rel);
 | |
|     
 | |
|     // Creo la chiave corretta per lo status node
 | |
|     key_rel.cut(0);
 | |
|     key_rel.add(new_rel.get(ABRL_TIPOCOD));
 | |
|     key_rel.add(new_rel.get(ABRL_CODVC));
 | |
|     key_rel.add(new_rel.get(ABRL_IDCOMP));
 | |
|     // Setto lo status 
 | |
|     set_status_node(key_rel,NODO_AGGIUNTO);
 | |
|     
 | |
|     //Aggiorno eventualmente il precedente del nodo che sto inserendo
 | |
|     if (prev_comp > ID_NULLO)
 | |
|     {
 | |
|       key_rel.cut(0);
 | |
|       key_rel.add(current().get(ABRL_TIPOCOD));
 | |
|       key_rel.add(current().get(ABRL_CODVC));
 | |
|       key_rel.add(prev_comp.string());
 | |
|       goto_node(key_rel);
 | |
|       current().put(ABRL_NEXTCOMP,new_comp); // Il successivo del precedente sono io !!
 | |
|       _local_cache->put(*_currnode);
 | |
|       set_status_node(key_rel,NODO_MODIFICATO);
 | |
|     }
 | |
|     
 | |
|     //Aggiorno eventualmente il successivo
 | |
|     if (next_comp > ID_NULLO)
 | |
|     {       
 | |
|       key_rel.cut(0);
 | |
|       key_rel.add(current().get(ABRL_TIPOCOD));
 | |
|       key_rel.add(current().get(ABRL_CODVC));
 | |
|       key_rel.add(next_comp.string());
 | |
|       goto_node(key_rel);
 | |
|       current().put(ABRL_PREVCOMP,new_comp);  // Il precedente del successivo sono io !!
 | |
|       _local_cache->put(*_currnode);
 | |
|       set_status_node(key_rel,NODO_MODIFICATO);
 | |
|     }
 | |
|     
 | |
|     //Mi posizione sul nodo di relvoci appena immesso
 | |
|     key_rel.cut(0);
 | |
|     key_rel.add(current().get(ABRL_TIPOCOD));
 | |
|     key_rel.add(current().get(ABRL_CODVC));
 | |
|     key_rel.add(new_comp.string());
 | |
|     goto_node(key_rel);
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     _local_cache->put(node);
 | |
|   }
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| 
 | |
| bool TAlbero_locale_relvoci::delete_node(const TNumeric_id &id_nullo)
 | |
| {
 | |
| //  current_cache().put(current());
 | |
|   TToken_string key = current().get(ABRL_TIPOCOD);
 | |
|   key.add(current().get(ABRL_CODVC));
 | |
|   key.add(current().get(ABRL_IDCOMP));
 | |
|   set_status_node(key,NODO_RIMOSSO);
 | |
|   
 | |
|   TNumeric_id id, next, prev;
 | |
|   id = current().get_real(ABRL_IDCOMP);
 | |
|   prev = current().get_real(ABRL_PREVCOMP);
 | |
|   next = current().get_real(ABRL_NEXTCOMP);
 | |
|     
 | |
|   // Aggiorno eventualmente il precedente
 | |
|   if (prev != ID_NULLO)
 | |
|   { // Costruisco la chiave di ricerca, lo ricerco e gli dico che il suo successivo
 | |
|     //è diventato il mio successivo, aggiorno
 | |
|     key.cut(0);
 | |
|     key.add(current().get(ABRL_TIPOCOD));
 | |
|     key.add(current().get(ABRL_CODVC));
 | |
|     key.add(prev.string());
 | |
|     goto_node(key);
 | |
|     current().put(ABRL_NEXTCOMP,next);
 | |
|     current_cache().put(current());
 | |
|     set_status_node(key,NODO_MODIFICATO);
 | |
|   }
 | |
|   
 | |
|   if (next != ID_NULLO)
 | |
|   {
 | |
|     key.cut(0);
 | |
|     key.add(current().get(ABRL_TIPOCOD));
 | |
|     key.add(current().get(ABRL_CODVC));
 | |
|     key.add(next.string());
 | |
|     goto_node(key);
 | |
|     current().put(ABRL_PREVCOMP,prev);
 | |
|     current_cache().put(current());
 | |
|     set_status_node(key,NODO_MODIFICATO);
 | |
|   }
 | |
|   
 | |
|   return TRUE;
 | |
| } 
 | |
|  
 | |
| bool TAlbero_locale_relvoci::goto_node(const TString &key)
 | |
| {
 | |
|   TToken_string key_rel = (TToken_string &) key;
 | |
|   *_currnode = current_cache().get(key);
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| bool TAnalisi_bil::insert_new_node(TToken_string &codice)
 | |
| {     
 | |
|     TNumeric_id prec_id = codice.get(0);   
 | |
|     user_tree()->goto_node(prec_id);
 | |
|     TToken_string key = user_tree()->current().get(ABRA_TIPOCOD);
 | |
|     key.add(user_tree()->current().get(ABRA_CODVC));
 | |
|     key.add(user_tree()->current().get(ABRA_NCOMP));
 | |
|     
 | |
|     if (user_tree_voc()->goto_node(key))
 | |
|     {
 | |
|       TNumeric_id father_id = user_tree()->current().get(ABRA_IDPADRE);
 | |
|       TRectype new_node(user_tree()->current());
 | |
|       new_node.zero();
 | |
|       //Inserisco il nodo corrispondente nella struttura di relvoci
 | |
|       user_tree_voc()->insert_node(new_node,ID_NULLO,ID_NULLO,ID_NULLO,IMMESSO);
 | |
|       
 | |
|       //Completo le informazioni di new_node con quelle appena inserite in relvoci
 | |
|       new_node.put(ABRA_TIPOCOD,user_tree_voc()->current().get(ABRL_TIPOCOD));
 | |
|       new_node.put(ABRA_CODVC,user_tree_voc()->current().get(ABRL_CODVC));
 | |
|       new_node.put(ABRA_NCOMP,user_tree_voc()->current().get(ABRL_IDCOMP));
 | |
|       
 | |
|       user_tree()->insert_node(new_node,prec_id,father_id,ID_NULLO,IMMESSO);
 | |
|       
 | |
|       return TRUE;
 | |
|     }
 | |
|     return FALSE;
 | |
| }
 | |
| 
 | |
| bool TAnalisi_bil::remove_node(const TNumeric_id &id)
 | |
| {                            
 | |
|   user_tree()->goto_node(id);
 | |
|   TToken_string key_rel; 
 | |
|   key_rel.add(user_tree()->current().get(ABRA_TIPOCOD));
 | |
|   key_rel.add(user_tree()->current().get(ABRA_CODVC));
 | |
|   key_rel.add(user_tree()->current().get(ABRA_NCOMP));
 | |
|   
 | |
|   if (user_tree_voc()->goto_node(key_rel))
 | |
|   { // Questa delete_node vorrebbe un TNumeric_id; ma siccome non
 | |
|     // ho nulla del genere, e sono già sul nodo da rimuovere,
 | |
|     // gli passo ID_NULLO e rimuovo il nodo
 | |
|     user_tree_voc()->delete_node(ID_NULLO);
 | |
|     user_tree()->delete_node(id);
 | |
|     user_tree_voc()->write_cache(); //solo debug
 | |
|     user_tree()->write_cache(); //solo debug
 | |
|     return TRUE;
 | |
|   }
 | |
|   error_box("Sto rimuovendo un nodo di relana che non dovrebbe esserci");
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| int TAnalisi_bil::commit_body() const
 | |
| {
 | |
|   TAlbero_AB::commit_body();
 | |
|   
 | |
|   TRectype rec(_inter_tree_relvoci->current().num());
 | |
|   TLocalisamfile filebody(rec.num());
 | |
|   
 | |
|   TString status;
 | |
| 
 | |
| /**********************************************************/
 | |
| /* PARTE RISERVATA A RELVOCI */  
 | |
| /**********************************************************/
 | |
|   
 | |
|   //Estraggo un nodo per volta basandomi sull'assoc_array degli status  
 | |
|   while (_inter_tree_relvoci->dirty_nodes() > 0)                                              
 | |
|   {
 | |
|     TString id = ID_NULLO_STR;
 | |
|     int prova = _inter_tree_relvoci->dirty_nodes(); //solo debug
 | |
|     
 | |
|     rec= _inter_tree_relvoci->extract_dirtynode(id, status);
 | |
|     
 | |
|     //Ho un rectype e uno status node
 | |
|     if (status == NODO_AGGIUNTO)
 | |
|     { 
 | |
|       if (filebody.curr().empty())
 | |
|       { //Se il cursor di relana è posizionato in modo errato:
 | |
|         //lo sistemo sull'ultimo record del file
 | |
|         //DOMANDA: - Perchè ???
 | |
|         //TENATATIVO DI RISPOSTA: - Quando l'ultima azione effettuata (prima di questa "aggiunta") è la rimozione 
 | |
|         //  dell'ultimo record probabilmente (ma non sono sicuro) il cursore non è posizionato (o forse
 | |
|         //  è posizionato in modo errato come ad es. dopo l'EOF, visto che l'EOF era sull'ultimo record rimosso)
 | |
|         //  e quando eseguo un tentativo di scrittura da un errore di chiave duplicata (anche se questa non esiste);
 | |
|         //  In tutti gli altri casi (quando l'ultima azione è stata una write o una rewrite o una remove di un record
 | |
|         //  che non sia l'ultimo) questo problema non esiste.
 | |
|         //RISPOSTA ESATTA: - Bho !!!
 | |
|         filebody.last();
 | |
|       }      
 | |
|       int err = filebody.write(rec);
 | |
|       if (err != NOERR)
 | |
|         return err;
 | |
|     }
 | |
|     if (status == NODO_MODIFICATO)
 | |
|     {            
 | |
|       int err = filebody.rewrite(rec);
 | |
|       if (err != NOERR)
 | |
|         return err;
 | |
|     }
 | |
|     if (status == NODO_RIMOSSO)
 | |
|     {
 | |
|       filebody.read(rec); //Devo leggere sul file per posizionare il cursore: stesso motivo di prima
 | |
|       int err = filebody.remove(rec);
 | |
|       if (err != NOERR)
 | |
|         return err;
 | |
|     }
 | |
|     // In tutti gli altri casi il nodo non viene modificato
 | |
|   }
 | |
|   return NOERR;
 | |
| }
 | |
| 
 | |
| void TAnalisi_bil::modify_node(TToken_string &codice)
 | |
| {                
 | |
|    //Posiziono il nodo sul record selezionato
 | |
|   TNumeric_id temp_id = codice.get(0);
 | |
|   user_tree()->goto_node(temp_id);
 | |
|   //Aggiorno il contenuto del record
 | |
|   user_tree()->current().put(ABRA_TIPOCOD,codice.get(1));
 | |
|   user_tree()->current().put(ABRA_CODVC,codice.get(2));
 | |
|   user_tree()->current().put(ABRA_NCOMP,codice.get(3));
 | |
|   user_tree()->current().put(ABRA_VOCEINCID,codice.get(4));
 | |
|   user_tree()->current().put(ABRA_DESCRIZ,codice.get(5));
 | |
|   user_tree()->current().put(ABRA_USACARADD,codice.get(6));
 | |
|   user_tree()->current().put(ABRA_IDCARADD,codice.get(8));
 | |
|   user_tree()->current().put(ABRA_IDCOLDICH,codice.get(9));
 | |
|     
 | |
|   // Rintraccio il record corrispondente di relvoci e apporto le modifiche
 | |
|   TToken_string key_rel;
 | |
|   key_rel.add(codice.get(1));
 | |
|   key_rel.add(codice.get(2));
 | |
|   key_rel.add(codice.get(3));
 | |
|   user_tree_voc()->goto_node(key_rel);
 | |
|   user_tree_voc()->current().put(ABRL_CODCOMP,codice.get(7));
 | |
|         
 | |
|   // Salvo le modifiche di relana
 | |
|   user_tree()->current_cache().put(user_tree()->current());
 | |
|   user_tree()->set_status_node(temp_id.string(),NODO_MODIFICATO);
 | |
|   user_tree()->write_cache();  //Solo debug
 | |
|     
 | |
|   // Salvo le modifiche di relvoci
 | |
|   user_tree_voc()->current_cache().put(user_tree_voc()->current());
 | |
|   user_tree_voc()->set_status_node(key_rel,NODO_MODIFICATO);
 | |
|   user_tree_voc()->write_cache();  //Solo debug
 | |
| }
 | |
| 
 | |
| /***********************************************************/
 | |
| /***********************************************************/
 | |
| 
 | |
| TRecord::TRecord(int filenum)
 | |
| 
 | |
| {
 | |
| // debug mods:
 | |
|   _currrec = new TRectype(filenum);
 | |
|   _local_cache_record = new TRWrecord_cache(_r);
 | |
| }                       
 | |
| 
 | |
| TRecord::~TRecord()
 | |
| {               
 | |
|   delete _currrec;
 | |
|   delete _local_cache_record;
 | |
|   delete _r;
 | |
|   _currrec = NULL;
 | |
|   _local_cache_record = NULL;
 | |
|   _r = NULL;
 | |
| } 
 | |
| 
 | |
| TRectype TRecord::extract_dirtynode(TString &id, TString& status)
 | |
| {              
 | |
|   TRectype rec(current_rec());
 | |
|   THash_object *row_status;
 | |
|   _status_node_record.restart();
 | |
|   row_status = _status_node_record.get_hashobj();
 | |
|   status = (TString&)row_status->obj();
 | |
|   id = (TString)row_status->key();
 | |
|   rec = current_cache_record().get(id);
 | |
|   _status_node_record.remove(id);
 | |
|   return rec; //Ritorno un Trectype e passo lo stauts node
 | |
| }     
 | |
| 
 | |
| void TRecord::set_status_node(const TString &id, const char *status)
 | |
| {
 | |
|   TString key = id;
 | |
|   const TString *prec_status = (const TString *)_status_node_record.objptr(key);
 | |
|   
 | |
|   if (!prec_status)
 | |
|   { //Elemento ancora non presente nell'array
 | |
|     _status_node_record.add(key,TString (status));
 | |
|   }
 | |
|   else
 | |
|   { //Elemento già inserito
 | |
|     if (stricmp(status,NODO_AGGIUNTO)==0 && (*prec_status != NODO_AGGIUNTO))
 | |
|       _status_node_record.add(key,TString (status),TRUE); //Forzo la sovrascrittura in ogni caso
 | |
|     
 | |
|     if (stricmp(status,NODO_MODIFICATO)== 0 && (*prec_status != NODO_AGGIUNTO))
 | |
|     {
 | |
|       //Se il nodo era già inserito e il suo flag è diverso da NODO_AGGIUNTO
 | |
|       //allora devo settare che questo nodo è modificato (se non lo è ancora)
 | |
|       if (*prec_status != NODO_MODIFICATO)
 | |
|       {
 | |
|         _status_node_record.add(key,TString (status),TRUE); //Forzo la sovrascrittura in ogni caso
 | |
|       } 
 | |
|       // Arriva qui se il nodo è inserito e il flag era già stato settato da una
 | |
|       //operazione precedente a NODO_MODIFICATO
 | |
|     }
 | |
|     
 | |
|     if (stricmp(status,NODO_RIMOSSO)==0 && *prec_status != NODO_RIMOSSO)
 | |
|     { //Sto rimuovento un nodo che prima era stato modificato
 | |
|       //message_box("Sto rimuovendo un nodo appena modificato e non ancora registrato");
 | |
|     }
 | |
|   }
 | |
| }
 |