6301d0f6f6
Files correlati : Ricompilazione Demo : [ ] Commento : Corretto calcolo saldi :-) git-svn-id: svn://10.65.10.50/trunk@13220 c028cbd2-c16b-5b4b-a496-9718f37d4682
391 lines
11 KiB
C++
Executable File
391 lines
11 KiB
C++
Executable File
#include <xvt.h>
|
|
|
|
#include "calib01.h"
|
|
#include "calib02.h"
|
|
|
|
#include "movana.h"
|
|
#include "rmovana.h"
|
|
#include "saldana.h"
|
|
|
|
///////////////////////////////////////////////////////////
|
|
// TAnal_report
|
|
///////////////////////////////////////////////////////////
|
|
|
|
bool TAnal_report::get_usr_val(const TString& name, TVariant& var) const
|
|
{
|
|
// CODCONTO:1, CODCMS:3, FASCMS:2, ecc...
|
|
const int namelen = name.len();
|
|
if (namelen >= 6 && name[namelen-2] == ':')
|
|
{
|
|
int logicnum = 0;
|
|
if (name.starts_with("CODCONTO:"))
|
|
logicnum = LF_PCONANA; else
|
|
if (name.starts_with("CODCMS:"))
|
|
logicnum = LF_COMMESSE; else
|
|
if (name.starts_with("CODCOSTO:"))
|
|
logicnum = LF_CDC; else
|
|
if (name.starts_with("FASCMS:"))
|
|
logicnum = LF_FASI;
|
|
if (logicnum > 0)
|
|
{
|
|
const TMultilevel_code_info& mci = ca_multilevel_code_info(logicnum);
|
|
const int level = name[namelen-1]-'1';
|
|
if (level < mci.levels())
|
|
{
|
|
TString16 fldname = name; fldname.cut(namelen-2);
|
|
TFieldref field = mci.fieldref(level);
|
|
field.set_name(fldname);
|
|
field.set_from(0);
|
|
fldname.cut(0) << field; // Trasformo il TFieldref in stringa!
|
|
get_record_field(fldname, var);
|
|
}
|
|
else
|
|
var.set_null();
|
|
return true;
|
|
}
|
|
}
|
|
return TReport::get_usr_val(name, var);
|
|
}
|
|
|
|
size_t TAnal_report::get_usr_words(TString_array& words) const
|
|
{
|
|
TReport::get_usr_words(words);
|
|
|
|
const char* const name[] =
|
|
{
|
|
"CA_FORMAT_COSTO",
|
|
"CA_FORMAT_COMMESSA",
|
|
"CA_FORMAT_FASE",
|
|
"CA_FORMAT_CONTO",
|
|
"CA_FORMAT_CMSCDC",
|
|
NULL
|
|
};
|
|
|
|
((TAnal_report*)this)->_first_msg = words.items(); // Calcola il primo numero disponibile
|
|
size_t i;
|
|
for (i = 0; name[i] != NULL; i++)
|
|
words.add(name[i]);
|
|
|
|
return words.items();
|
|
}
|
|
|
|
void TAnal_report::msg_format(int logicnum, TVariant_stack& stack)
|
|
{
|
|
const TString& str_in = curr_field()->get().as_string();
|
|
if (str_in.not_empty())
|
|
{
|
|
TString80 separator = " ";
|
|
if (stack.items() > 0)
|
|
separator = stack.pop().as_string();
|
|
if (separator.not_empty())
|
|
{
|
|
const TMultilevel_code_info& mci = ca_multilevel_code_info(logicnum);
|
|
TString str_out;
|
|
for (int i = 0; i < mci.levels(); i++)
|
|
{
|
|
const TFieldref& fld = mci.fieldref(i);
|
|
const TString& str = str_in.sub(fld.from(), fld.to());
|
|
if (str.not_empty())
|
|
{
|
|
if (i > 0) str_out << separator;
|
|
str_out << str;
|
|
}
|
|
else
|
|
break;
|
|
}
|
|
curr_field()->set(str_out);
|
|
}
|
|
}
|
|
}
|
|
|
|
void TAnal_report::msg_format_costo (TVariant_stack& stack)
|
|
{ msg_format(LF_CDC, stack); }
|
|
|
|
void TAnal_report::msg_format_commessa(TVariant_stack& stack)
|
|
{ msg_format(LF_COMMESSE, stack); }
|
|
|
|
void TAnal_report::msg_format_fase (TVariant_stack& stack)
|
|
{ msg_format(LF_FASI, stack); }
|
|
|
|
void TAnal_report::msg_format_conto (TVariant_stack& stack)
|
|
{ msg_format(LF_PCONANA, stack); }
|
|
|
|
void TAnal_report::msg_format_commessa_costo(TVariant_stack& stack)
|
|
{
|
|
const TMultilevel_code_info& main_info = ca_multilevel_code_info(LF_FASI);
|
|
msg_format(main_info.parent(), stack);
|
|
}
|
|
|
|
bool TAnal_report::execute_usr_word(unsigned int opcode, TVariant_stack& stack)
|
|
{
|
|
if (opcode < _first_msg)
|
|
return TReport::execute_usr_word(opcode, stack);
|
|
opcode -= _first_msg;
|
|
switch (opcode)
|
|
{
|
|
case 0 : msg_format_costo(stack); break;
|
|
case 1 : msg_format_commessa(stack); break;
|
|
case 2 : msg_format_fase(stack); break;
|
|
case 3 : msg_format_conto(stack); break;
|
|
case 4 : msg_format_commessa_costo(stack); break;
|
|
default: break;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////
|
|
// TAnal_balance
|
|
///////////////////////////////////////////////////////////
|
|
|
|
class TAnal_balance : public TObject
|
|
{
|
|
public:
|
|
TImporto saldo_fine_anno(const char* conto, const char* costo,
|
|
const char* commessa, const char* fase,
|
|
int annofin, word tipo) const;
|
|
bool saldo_movimenti(const char* conto, const char* costo,
|
|
const char* commessa, const char* fase,
|
|
const TDate& dal, const TDate& al, word tipo,
|
|
TImporto& ini, TImporto& dare, TImporto& avere) const;
|
|
bool saldi(const char* conto, const char* costo,
|
|
const char* commessa, const char* fase,
|
|
const TDate& dal, const TDate& al, word tipo,
|
|
TImporto& ini, TImporto& dare, TImporto& avere) const;
|
|
};
|
|
|
|
TImporto TAnal_balance::saldo_fine_anno(const char* conto, const char* costo,
|
|
const char* commessa, const char* fase,
|
|
int anno, word tipo) const
|
|
{
|
|
TImporto saldo;
|
|
|
|
if (anno > 0)
|
|
{
|
|
TString query, select;
|
|
|
|
if (costo && *costo)
|
|
select << "(CODCOSTO=='" << costo << "')";
|
|
if (commessa && *commessa)
|
|
{
|
|
if (select.not_empty()) select << "&&";
|
|
select << "(CODCMS=='" << commessa << "')";
|
|
}
|
|
if (fase && *fase)
|
|
{
|
|
if (select.not_empty()) select << "&&";
|
|
select << "(FASCMS=='" << fase << "')";
|
|
}
|
|
if (conto && *conto)
|
|
{
|
|
if (select.not_empty()) select << "&&";
|
|
select << "(CONTO=='" << conto << "')";
|
|
}
|
|
|
|
query << "USE SALDANA";
|
|
if (select.not_empty())
|
|
query << " SELECT " << select;
|
|
if (anno > 0)
|
|
query << "\nTO ANNO=" << anno << '\n';
|
|
|
|
TISAM_recordset saldini(query);
|
|
|
|
for (int i = 0; i < saldini.items(); i++)
|
|
{
|
|
if (tipo & 1)
|
|
{
|
|
const char sez = saldini.get(SALDANA_SEZIONE).as_string()[0];
|
|
const real imp = saldini.get(SALDANA_SALDO).as_real();
|
|
saldo += TImporto(sez, imp);
|
|
}
|
|
if (tipo & 2)
|
|
{
|
|
const char sez = saldini.get(SALDANA_SEZIONEP).as_string()[0];
|
|
const real imp = saldini.get(SALDANA_SALDOP).as_real();
|
|
saldo += TImporto(sez, imp);
|
|
}
|
|
if (tipo & 4)
|
|
{
|
|
const char sez = saldini.get(SALDANA_SEZIONEV).as_string()[0];
|
|
const real imp = saldini.get(SALDANA_SALDOV).as_real();
|
|
saldo += TImporto(sez, imp);
|
|
}
|
|
}
|
|
saldo.normalize();
|
|
}
|
|
|
|
return saldo;
|
|
}
|
|
|
|
bool TAnal_balance::saldo_movimenti(const char* conto, const char* costo,
|
|
const char* commessa, const char* fase,
|
|
const TDate& dal, const TDate& al, word tipo,
|
|
TImporto& ini, TImporto& dare, TImporto& avere) const
|
|
{
|
|
TDate dataini;
|
|
if (dal.ok())
|
|
{
|
|
TEsercizi_contabili esc;
|
|
const int annoprec = esc.date2prevesc(dal);
|
|
if (annoprec > 0)
|
|
dataini = esc[annoprec].fine()+1L;
|
|
}
|
|
|
|
TString query, select;
|
|
|
|
if (dataini.ok())
|
|
select << "(ANSI(107.DATACOMP)>=" << dataini.date2ansi() << ")";
|
|
if (al.ok())
|
|
{
|
|
if (select.not_empty())
|
|
select << "&&";
|
|
select << "(ANSI(107.DATACOMP)<=" << al.date2ansi() << ")";
|
|
}
|
|
if (costo && *costo)
|
|
{
|
|
if (select.not_empty())
|
|
select << "&&";
|
|
select << "(CODCOSTO==\"" << costo << "\")";
|
|
}
|
|
if (commessa && *commessa)
|
|
{
|
|
if (select.not_empty())
|
|
select << "&&";
|
|
select << "(CODCMS==\"" << commessa << "\")";
|
|
}
|
|
if (fase && *fase)
|
|
{
|
|
if (select.not_empty())
|
|
select << "&&";
|
|
select << "(CODFASE==\"" << fase << "\")";
|
|
}
|
|
|
|
query << "USE RMOVANA KEY 2\n";
|
|
if (select.not_empty())
|
|
query << "SELECT " << select << "\n";
|
|
query << "JOIN MOVANA INTO NUMREG==NUMREG\n";
|
|
|
|
query << "FROM CODCONTO=" << conto;
|
|
if (dataini.ok())
|
|
query << " DATAREG=" << TDate(dataini-90L).string();
|
|
query << "\n";
|
|
|
|
query << "TO CODCONTO=" << conto;
|
|
if (al.ok())
|
|
query << " DATAREG=" << al.string();
|
|
query << "\n";
|
|
|
|
TISAM_recordset rmovana(query);
|
|
|
|
bool movimentato = false;
|
|
for (int i = 0; i < rmovana.items(); i++)
|
|
{
|
|
rmovana.move_to(i);
|
|
|
|
const char tipomov = rmovana.get("107.TIPOMOV").as_string()[0];
|
|
int ntipomov = 0;
|
|
if (tipomov == 'P')
|
|
ntipomov = 2; else
|
|
if (tipomov == 'V')
|
|
ntipomov = 4;
|
|
else
|
|
ntipomov = 1;
|
|
if ((ntipomov & tipo) == 0)
|
|
continue;
|
|
|
|
const TDate data = rmovana.get("107.DATACOMP").as_date();
|
|
const TImporto imp(rmovana.get(RMOVANA_SEZIONE).as_string()[0],
|
|
rmovana.get(RMOVANA_IMPORTO).as_real());
|
|
if (data >= dal)
|
|
{
|
|
if (imp.sezione() == 'D')
|
|
dare += imp;
|
|
else
|
|
avere += imp;
|
|
movimentato = true;
|
|
}
|
|
else
|
|
{
|
|
ini += imp;
|
|
}
|
|
}
|
|
ini.normalize();
|
|
|
|
return movimentato;
|
|
}
|
|
|
|
bool TAnal_balance::saldi(const char* conto, const char* costo,
|
|
const char* commessa, const char* fase,
|
|
const TDate& dal, const TDate& al, word tipo,
|
|
TImporto& ini, TImporto& dare, TImporto& avere) const
|
|
{
|
|
TEsercizi_contabili esc;
|
|
const int annoprec = esc.date2prevesc(dal);
|
|
ini = saldo_fine_anno(conto, costo, commessa, fase, annoprec, tipo);
|
|
const bool mov = saldo_movimenti(conto, costo, commessa, fase, dal, al, tipo,
|
|
ini, dare, avere);
|
|
return mov;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////
|
|
// TSaldi_cache
|
|
////////////////////////////////////////////////////////
|
|
|
|
class TSaldi_cache : private TCache
|
|
{
|
|
protected:
|
|
virtual TObject* key2obj(const char* key);
|
|
|
|
public:
|
|
const TSaldanal& saldo(const char* conto, const char* costo,
|
|
const char* commessa, const char* fase,
|
|
const TDate& dal, const TDate& al, word tipi = 0x1);
|
|
TSaldi_cache() : TCache(3883) { }
|
|
};
|
|
|
|
TObject* TSaldi_cache::key2obj(const char* key)
|
|
{
|
|
TSaldanal* s = new TSaldanal;
|
|
|
|
TToken_string tok(key);
|
|
TString80 conto = tok.get(); conto.trim();
|
|
TString80 costo = tok.get(); costo.trim();
|
|
TString80 commessa = tok.get(); commessa.trim();
|
|
TString80 fase = tok.get(); fase.trim();
|
|
const TDate dal = tok.get();
|
|
const TDate al = tok.get();
|
|
const char tipo = tok.get_char();
|
|
|
|
TAnal_balance bal;
|
|
s->_movimentato = bal.saldi(conto, costo, commessa, fase, dal, al, tipo,
|
|
s->_ini, s->_dare, s->_avere);
|
|
s->_fin = s->_ini; s->_fin += s->_dare; s->_fin += s->_avere;
|
|
s->_fin.normalize();
|
|
|
|
return s;
|
|
}
|
|
|
|
const TSaldanal& TSaldi_cache::saldo(const char* conto, const char* costo,
|
|
const char* commessa, const char* fase,
|
|
const TDate& dal, const TDate& al, word tipi)
|
|
{
|
|
TToken_string key;
|
|
key.add(conto, 0);
|
|
key.add(costo, 1);
|
|
key.add(commessa, 2);
|
|
key.add(fase, 3);
|
|
key.add(dal, 4);
|
|
key.add(al, 5);
|
|
key.add(tipi, 6);
|
|
return *(const TSaldanal*)objptr(key);
|
|
}
|
|
|
|
const TSaldanal& ca_saldo(const char* conto, const char* costo,
|
|
const char* commessa, const char* fase,
|
|
const TDate& dal, const TDate& al, word tipi)
|
|
{
|
|
static TSaldi_cache* cache = NULL;
|
|
if (cache == NULL)
|
|
cache = new TSaldi_cache;
|
|
return cache->saldo(conto, costo, commessa, fase, dal, al, tipi);
|
|
}
|