// cg5200.cpp  - Visualizzazione saldi

#include <brwapp.h>
#include <msksheet.h> 
#include <tabutil.h>
#include <urldefid.h>
#include <utility.h>

#include <saldi.h>
#include <clifo.h>
#include <pconti.h>

#include "cglib02.h"
#include "cg5200.h"

class TRiga_array : public TArray
{
public:
  bool add_riga(const TRectype& rec_saldi); 
};

bool TRiga_array::add_riga(const TRectype& rec_saldi) 
{
  bool flsca, found = FALSE;
  int annoes, annoesr;
  char flag = ' ', flagr = ' ';    
  char flagfin = ' ', flagfinr = ' ';
  TDate udata, udatar;
  long  unum, unumr;
  real saldofin,pdare,pavere,saldoini,pdarepro,paverepro;
  real saldofinr,pdarer,paverer,saldoinir,pdarepror,paverepror;
  
  
  for (int i = 0; i < items(); i++)
  {
    TRectype& r = (TRectype&)(*this)[i];
    annoes    = r.get_int(SLD_ANNOES);
    annoesr   = rec_saldi.get_int(SLD_ANNOES);

// TBI occhio al flag

    if ( annoes == annoesr ) 
    {           
      flsca     = r.get_bool(SLD_FLSCA);
      flag      = r.get_char(SLD_FLAGSALINI);
      flagfin   = r.get_char(SLD_FLAGSALFIN);
      pdare     = r.get_real(SLD_PDARE);
      pavere    = r.get_real(SLD_PAVERE);
      saldoini  = r.get_real(SLD_SALDO);   
      saldofin  = r.get_real(SLD_SALDOFIN);
      pdarepro  = r.get_real(SLD_PDAREPRO);
      paverepro = r.get_real(SLD_PAVEREPRO);
      udata     = r.get(SLD_DATAULMOV);
      unum      = r.get_long(SLD_NUMULTMOV);     
      flagr     = rec_saldi.get_char(SLD_FLAGSALINI);
      flagfinr  = rec_saldi.get_char(SLD_FLAGSALFIN);
      pdarer    = rec_saldi.get_real(SLD_PDARE);
      pdarer   += pdare;
      paverer   = rec_saldi.get_real(SLD_PAVERE);
      paverer  += pavere;
      saldoinir = rec_saldi.get_real(SLD_SALDO);
      saldofinr = rec_saldi.get_real(SLD_SALDOFIN);
      udatar    = rec_saldi.get(SLD_DATAULMOV);
      unumr     = rec_saldi.get_long(SLD_NUMULTMOV);
      udatar    = fnc_max(udatar,udata);
      //unumr     = (unum > unumr) ? unum : unumr;
      if (udata > udatar)
        unumr = unum;
      if (flagr == flag)
        saldoinir += saldoini;
      else saldoinir -= saldoini;
      if (saldoinir < ZERO) // saldoinir e' piu' piccolo di saldoini => vince il flag di saldoini
      {
        flagr = flag;
        saldoinir = -saldoinir;
      }      
      pdarepror  = rec_saldi.get_real(SLD_PDAREPRO);
      pdarepror  += pdarepro;
      paverepror = rec_saldi.get_real(SLD_PAVEREPRO);
      paverepror += paverepro;
      found = TRUE;                
      r.put(SLD_FLSCA, flsca);
      r.put(SLD_ANNOES, annoes);
      r.put(SLD_PDARE, pdarer);
      r.put(SLD_PAVERE,paverer);
      r.put(SLD_FLAGSALINI,flagr);
      r.put(SLD_FLAGSALFIN,flagfinr);
      r.put(SLD_SALDO, saldoinir);
      r.put(SLD_SALDOFIN, saldofinr);
      r.put(SLD_PDAREPRO, pdarepror);
      r.put(SLD_PAVEREPRO,paverepro);
      r.put(SLD_DATAULMOV, udatar);
      r.put(SLD_NUMULTMOV, unumr);
    }
  }
  if (!found)
  {
    add(rec_saldi);
  }
  return found;
}             

///////////////////////////////////////////////////////////
// Visualizzazione saldi
///////////////////////////////////////////////////////////

class TSaldibrowse_application : public TBrowse_application
{
  TMask*          _msk;
  TCursor       * _cur;
  TRelation     * _rel,* _rel1;
  TRiga_array   _riga;
  int _anno, _annop, _g, _c;
  long _s;
  bool _saldo_conto, _saldo_gruppo, _saldo_sottoc, _scarongly;

  TEsercizi_contabili _ese;
  
protected:
  virtual bool user_create();
  virtual bool user_destroy();
  virtual TRelation*   get_relation() const { return _rel; }
  virtual TMask*       get_mask(int mode)   { return _msk; }
  virtual bool         changing_mask(int mode) {return FALSE; }
  virtual int read(TMask& m);
  virtual void init_query_mode(TMask&);
  virtual void init_modify_mode(TMask&); 

  static int  compare_rows(const TObject **o1, const TObject** o2);
  static bool anno_handler  (TMask_field& f, KEY k);
  static bool flsca_handler (TMask_field& f, KEY k);
  static bool sottoc_handler (TMask_field& f, KEY k);
  static bool sheet_action(TSheet_field& s, int r, KEY k);  
public:
  void add_r(int,int,real&,char,real&,char,real&,real&, real&,real&,
             TDate&,long,real&,real&,real&, real&, real&);
  void compilasheet();
  bool fai_filtro();
  TSheet_field& ss() const { return (TSheet_field&)_msk->field(F_SHEET_SALDI);}

  TSaldibrowse_application() {}
};

HIDDEN TSaldibrowse_application& app() { return (TSaldibrowse_application&) main_app(); }

bool TSaldibrowse_application::sheet_action(TSheet_field& s, int r, KEY k)
{                    
  // non si possono cancellare o aggiungere righe
  return (k != K_DEL && k != K_INS);
}

void TSaldibrowse_application::init_query_mode(TMask& m)
{
  _anno = 0;
  _scarongly = FALSE;
  m.disable(DLG_FINDREC);
  m.enable(F_DESCR_CONTO);
  m.enable(F_ANNO);
  m.enable(F_SCARICATO);
}

void TSaldibrowse_application::init_modify_mode(TMask& m)
{
  m.disable(DLG_FINDREC);
  m.disable(F_ANNO);
  m.disable(F_SCARICATO);
  
  const char tipo = m.get(F_TIPOCF)[0];
  if (tipo=='C')
    m.disable(F_DESCR_CLIENTE);
  else if (tipo=='F')
    m.disable(F_DESCR_FORN);
  else 
    m.disable(F_DESCR_CONTO);
}

bool TSaldibrowse_application::fai_filtro()
{
  TSaldo sld; 
  int annop = 0;
  char tipo;
  
  _saldo_gruppo = _saldo_conto = _saldo_sottoc = FALSE;
  TMask& m = curr_mask();
  tipo  = m.get(F_TIPOCF)[0];
  if (_anno != 0)  
  {
    _annop = _ese.pred(_anno);
  }
  else 
    _annop = 0;
  _g = m.get_int(F_GRUPPO);
  _c = m.get_int(F_CONTO);

  if (tipo == '\0')
    _s = m.get_long(F_SOTTOCONTO);
  else if (tipo == 'C')
    _s = m.get_long(F_SOTTOC_CLIENTE);
  else _s = m.get_long(F_SOTTOC_FORN);
  if (_c == 0)
    _saldo_gruppo = TRUE;
  else if (_s == 0l)
    _saldo_conto = TRUE;
  else _saldo_sottoc = TRUE;   
  
  TRectype from (LF_SALDI);
  TRectype to   (LF_SALDI);
  
  from.put(SLD_GRUPPO,_g);
  if (_c != 0)
    from.put(SLD_CONTO, _c);
  if (_s != 0l)   
    from.put(SLD_SOTTOCONTO, _s);
  
  to.put(SLD_GRUPPO,_g);
  if (_c != 0)
    to.put(SLD_CONTO,_c);
  if (_s != 0l)   
    to.put(SLD_SOTTOCONTO,_s);
  
  _cur->setkey(2);
  
  _cur->setregion(from,to);                      
  
  return TRUE;   
}

bool TSaldibrowse_application::anno_handler(TMask_field& f, KEY key)
{
  if (key == K_TAB && f.focusdirty())
  { 
    const int anno = atoi(f.get());
    if (anno > 0)
    {
      if (!app()._ese.exist(anno))
        return f.error_box(FR("Esercizio %d non presente."), anno);
      else
        app()._anno = anno;
    }  
  }
  return TRUE;
}

bool TSaldibrowse_application::flsca_handler(TMask_field& f, KEY key)
{
  if (key == K_SPACE)
    app()._scarongly = !f.get().blank();
  return TRUE;
}

bool TSaldibrowse_application::sottoc_handler(TMask_field& f, KEY key)
{
  bool ok = TRUE;
  
  if (key == K_TAB)
  {      
    bool stop = FALSE;
    
    TMask& m = f.mask();
    int  gruppo = m.get_int(F_GRUPPO);
    int  conto  = m.get_int(F_CONTO);
    const long sottoconto = atol(f.get());
    const short id  = f.dlg();
    
    if (sottoconto != 0)
    {
      if (id == F_SOTTOCONTO)
      {
        if (gruppo != 0 && conto != 0)
        {
          TString key;
					
					key.format("%d|%d|%ld",  gruppo, conto, sottoconto);
          
					const TRectype & pconti = cache().get(LF_PCON, key);
          
					ok = stop = !pconti.empty();
        }  
      } 
      else
      {
        const char tipo = id == F_SOTTOC_CLIENTE ? 'C' : 'F';
				TString key;

				key.format("%c|%ld",  tipo, sottoconto);
        app()._s = sottoconto;

				const TRectype & clifo = cache().get(LF_CLIFO, key);

        ok = stop = !clifo.empty();
        if (ok && (gruppo == 0 || conto == 0))
        {
          m.set(F_GRUPPO, gruppo = clifo.get_int("GRUPPO"));
          m.set(F_CONTO, conto = clifo.get_int("CONTO"));
          ok = stop = gruppo != 0 && conto != 0;
        }
      }
    }  
    else
      if (gruppo != 0 /* && conto != 0 */)
      {
        TString key;
				
				key.format("%d|%d",  gruppo, conto);
        
				const TRectype & pconti = cache().get(LF_PCON, key);
        
				ok = stop = !pconti.empty();
        if (ok) 
          m.set(F_DESCR_CONTO, pconti.get(PCN_DESCR));
        else 
          m.reset(F_DESCR_CONTO);
      }
    if (!ok)
      f.error_box(TR("Conto errato o incompleto"));
  }
  
  return ok;
}

bool TSaldibrowse_application::user_create()
{
	open_files(LF_TABCOM, LF_TAB, LF_PCON, LF_SALDI, 0);
  _rel   = new TRelation(LF_PCON);
  _rel->add(LF_CLIFO,"TIPOCF=TMCF");

  _rel1  = new TRelation(LF_SALDI);
  _cur   = new TCursor(_rel1, "", 2);
  _msk   = new TMask("cg5200a");
  _msk->set_handler(F_ANNO, anno_handler);
  _msk->set_handler(F_SCARICATO, flsca_handler);
  _msk->set_handler(F_SOTTOCONTO, sottoc_handler);
  _msk->set_handler(F_SOTTOC_CLIENTE, sottoc_handler);
  _msk->set_handler(F_SOTTOC_FORN, sottoc_handler);
  ss().disable();  
  ss().set_notify(sheet_action);      
  
  set_search_field(F_GRUPPO);
  
  return TRUE;
}

bool TSaldibrowse_application::user_destroy()
{
  delete _msk;
  delete _rel;
  delete _rel1;
  delete _cur;
//  delete _esc;
  
  return TRUE;
}

int TSaldibrowse_application::read(TMask& m)
{
  int err = NOERR;
  m.autoload(*_rel);
  m.set(F_ANNO,_anno);
  m.set(F_SCARICATO,_scarongly ? "X" : " ");
  TString tipo(m.get(F_TIPOCF));

  fai_filtro();
  if (_cur->items() > 0)
    compilasheet();
  else 
  {
    err = _iskeynotfound;
    error_box(TR("Saldi non presenti per il conto specificato."));
  }
  
  return err;
}

void TSaldibrowse_application::add_r(int numrig,int a,real& slf,char sff,real& sc,char f,real& si,
                                     real& pd, real& pa,real& saldo,TDate& d,long n,real& pdp,
                                     real& pap,real& sp, real& dsc, real& asc)
{
  TSheet_field& cs = ss();
  char segno = ' ';
  
  TToken_string& riga = cs.row(numrig);
  riga.add(a, 0);                         // 101 - Anno esercizio
  riga.add(si.string(),1);                // 102 - Saldo iniziale
  if (si != ZERO)   
    riga.add(f,2);                        // 103 - Segno saldo iniziale
  
  riga.add(pd.string(),5);                // 106 - Progressivo dare
  riga.add(pa.string(),6);                // 107 - Progressivo avere
  
  if (saldo > ZERO)
    segno = 'D';
  else if (saldo < ZERO)
  {
    saldo = -saldo;
    segno = 'A'; 
  }            
  riga.add(saldo.string(),3);             // 104 - Saldo attuale
  riga.add(segno,4);                      // 105 - Segno saldo attuale 
  
  segno = ' ';
  if (sc > ZERO)
    segno = 'D';
  else if (sc < ZERO)
  {
    sc = -sc;
    segno = 'A';
  }     
  riga.add(slf.string(),7);
  if (slf != ZERO)   
    riga.add(sff,8);
  riga.add(sc.string(),9);
  riga.add(segno,10);
  riga.add(d.string(),11);
  riga.add(n,12);
  riga.add(pdp.string(),13);
  riga.add(pap.string(),14);
  segno = ' ';
  if (sp > ZERO)
    segno = 'D';
  else if (sp < ZERO)
  {
    sp = -sp;
    segno = 'A';
  }        
  riga.add(sp.string(),15);
  riga.add(segno,16); 
  riga.add(dsc.string(),17); 
  riga.add(asc.string(),18); 
}

int TSaldibrowse_application::compare_rows(const TObject** o1, const TObject** o2)
{
  TToken_string* r1 = (TToken_string*)*o1;
  TToken_string* r2 = (TToken_string*)*o2;
  TEsercizi_contabili& ese = app()._ese;       
  int a1 = r1->get_int(0);
  TDate d1 = ese.exist(a1) ? ese[a1].inizio() : TDate(1, 1, a1);
  int a2 = r2->get_int(0);
  TDate d2 = ese.exist(a2) ? ese[a2].inizio() : TDate(1, 1, a2);

  int rt = d1 == d2 ? 0 : (d1 < d2 ? -1 : +1);
  return rt;
}

void TSaldibrowse_application::compilasheet()
{
  char flagsal   = ' ';
  char flagfin   = ' ';
  real saldo = ZERO;
  real saldocompl = ZERO;
//  real saldosca = ZERO; 
  real saldopro = ZERO;
//  real saldo_gc = ZERO;
//  real saldoini_gc     = ZERO;     ??? a che servono????
//  real saldosca_gc     = ZERO;
//  real saldopro_gc     = ZERO;
  real saldoini  = ZERO; 
  real pdare     = ZERO;
  real pavere    = ZERO;
  real pdaresca  = ZERO;
  real paveresca = ZERO;
  real saldofin  = ZERO;
  real pdarepro  = ZERO;
  real paverepro = ZERO;
  int gruppo, conto, anno;
  long sottoconto;
  long ultimo_num = 0l; 
  TDate ultima_data;
  
  // nuova struttura archivio saldi
  
  _riga.destroy();
  
  const TRecnotype items = _cur->items();
  *_cur = 0l;
  
  const TRectype& curr = _cur->curr();
  for (int i = 0; i < items; i++,++(*_cur))
  { 
                     
    const bool scar = curr.get_bool(SLD_FLSCA);
    if ((_scarongly && !scar) || (!_scarongly && scar))
      continue;
      
    anno   = curr.get_int(SLD_ANNOES);
    gruppo = curr.get_int(SLD_GRUPPO);
    conto  = curr.get_int(SLD_CONTO);
    sottoconto = curr.get_long(SLD_SOTTOCONTO);
    
    if (_anno != 0 && _anno != anno && _annop != anno) continue;
    
    if (gruppo != _g) continue;
    
    if (_saldo_conto && conto != _c) continue;
    
    if (_saldo_conto || _saldo_gruppo)
      _riga.add_riga(curr);
    
    if (_saldo_sottoc)
    {
      flagsal   = curr.get_char(SLD_FLAGSALINI);
      flagfin   = curr.get_char(SLD_FLAGSALFIN);
      saldoini  = curr.get_real(SLD_SALDO); 
      saldofin  = curr.get_real(SLD_SALDOFIN); 
      pdare     = curr.get_real(SLD_PDARE);
      pavere    = curr.get_real(SLD_PAVERE);
      pdarepro  = curr.get_real(SLD_PDAREPRO);
      paverepro = curr.get_real(SLD_PAVEREPRO);
      ultima_data = curr.get(SLD_DATAULMOV);
      ultimo_num  = curr.get_long(SLD_NUMULTMOV); 

      pdaresca  = ZERO;
      paveresca = ZERO;    
      
      if (flagsal == 'D')
        saldo = pdare + saldoini - pavere;
      else
        saldo = pdare - saldoini - pavere;

      if (flagfin == 'D')
        saldocompl = saldo + saldofin;
      else 
        saldocompl = saldo - saldofin;
      
      saldopro = pdarepro - paverepro;
      
      add_r(i,anno,saldofin,flagfin,saldocompl,flagsal,saldoini,pdare,pavere,saldo,
            ultima_data,ultimo_num,pdarepro,paverepro,saldopro,pdaresca, paveresca);   
    }
  }
  if (_saldo_gruppo || _saldo_conto)
  {
    for (int j = 0; j < _riga.items(); j++)
    {
      TRectype& riga = (TRectype&)_riga[j];
      anno = riga.get_int(SLD_ANNOES);
      flagsal   = riga.get_char(SLD_FLAGSALINI);
      flagfin   = riga.get_char(SLD_FLAGSALFIN);
      saldoini  = riga.get_real(SLD_SALDO);
      saldofin  = riga.get_real(SLD_SALDOFIN);
      pdare  = riga.get_real(SLD_PDARE);
      pavere = riga.get_real(SLD_PAVERE);
      if (flagsal == 'D')
        saldo = pdare + saldoini - pavere;
      else
        saldo = pdare - saldoini - pavere;

      if (flagfin == 'D')
        saldocompl = saldo + saldofin;
      else 
        saldocompl = saldo - saldofin;

      pdarepro  = riga.get_real(SLD_PDAREPRO);
      paverepro = riga.get_real(SLD_PAVEREPRO);
      saldopro  = pdarepro - paverepro;
      ultima_data = riga.get(SLD_DATAULMOV);
      ultimo_num  = riga.get_long(SLD_NUMULTMOV);

      pdaresca  = ZERO;
      paveresca = ZERO;

      if (!_scarongly && (j < _riga.items() - 1)) 
      {  
        TRectype& riga = (TRectype&)_riga[++j];
        if (riga.get_bool(SLD_FLSCA) &&
            sottoconto == riga.get_long(SLD_SOTTOCONTO) &&   
            conto      == riga.get_int(SLD_CONTO) &&
            gruppo     == riga.get_int(SLD_GRUPPO) &&
            anno       == riga.get_int(SLD_ANNOES))
        {
          pdaresca  = riga.get_real(SLD_PDARE);
          paveresca = riga.get_real(SLD_PAVERE);
        }
        else --j; 
      }
                                         
// come sopra
      add_r(j,anno,saldofin,flagfin,saldocompl,flagsal,saldoini,pdare,pavere,saldo,ultima_data,
            ultimo_num,pdarepro,paverepro,saldopro, pdaresca, paveresca);   
    } 
    _riga.destroy();
  }   
  
  // Ora ordina le righe dello sheet per ordine della data inizio esercizio
  TString_array& righe_sheet = ss().rows_array();
  righe_sheet.TArray::sort(compare_rows);
}     

int cg5200(int argc, char* argv[])
{
  TSaldibrowse_application a;
  a.run(argc, argv, TR("Visualizzazione saldi"));
  return 0;
}