2145 lines
66 KiB
C++
2145 lines
66 KiB
C++
|
//////////////////////////////////////////////////////////////
|
|||
|
// Stampa documenti
|
|||
|
//////////////////////////////////////////////////////////////
|
|||
|
|
|||
|
#include <applicat.h>
|
|||
|
#include <config.h>
|
|||
|
|
|||
|
#include <defmask.h>
|
|||
|
#include <execp.h>
|
|||
|
#include <modaut.h>
|
|||
|
#include <postman.h>
|
|||
|
#include <printer.h>
|
|||
|
#include <progind.h>
|
|||
|
#include <sheet.h>
|
|||
|
#include <toolfld.h>
|
|||
|
|
|||
|
#include "velib05.h"
|
|||
|
#include "sconti.h"
|
|||
|
#include "ve1100.h"
|
|||
|
|
|||
|
#include <comuni.h>
|
|||
|
#include <nditte.h>
|
|||
|
|
|||
|
#define LISTADOC "listadoc"
|
|||
|
#define FAKETOTFLD 9999
|
|||
|
|
|||
|
// Queste classi (TDocisamfile e TRDocisamfile) servono nel costruttore di TDocumento_form
|
|||
|
// in modo da sostituire i file della relazione, ovvero LF_DOC e LF_RIGHEDOC.
|
|||
|
// Facendo in questo modo ogni get() del record, viene reindirizzata alla get() dei
|
|||
|
// TVariable_recfield, in modo da utilizzare nel form le istruzioni FIELD anche per i campi
|
|||
|
// virtuali.
|
|||
|
|
|||
|
class TDocisamfile : public TLocalisamfile
|
|||
|
{
|
|||
|
TDocumentoEsteso* _doc;
|
|||
|
|
|||
|
public:
|
|||
|
virtual TRectype& curr() const { return (TRectype&) *_doc; }
|
|||
|
TDocisamfile(TDocumentoEsteso* doc) : TLocalisamfile(LF_DOC), _doc(doc)
|
|||
|
{ CHECK(_doc, "Invalid document"); }
|
|||
|
virtual int readat(TRecnotype nrec, word lockop);
|
|||
|
virtual ~TDocisamfile() {};
|
|||
|
};
|
|||
|
|
|||
|
int TDocisamfile::readat(TRecnotype nrec, word lockop)
|
|||
|
{
|
|||
|
int err = TBaseisamfile::readat(nrec, _nolock);
|
|||
|
if (err == NOERR)
|
|||
|
{
|
|||
|
err = _doc->read(curr());
|
|||
|
if (err == NOERR)
|
|||
|
{
|
|||
|
_doc->summary_reset(true); // forza il ricalcolo perche' trattasi di documento diverso
|
|||
|
_doc->summary_filter(1);
|
|||
|
}
|
|||
|
}
|
|||
|
return err;
|
|||
|
}
|
|||
|
|
|||
|
class TRDocisamfile : public TLocalisamfile
|
|||
|
{
|
|||
|
TDocumento *_doc;
|
|||
|
int _row;
|
|||
|
bool _normal_next;
|
|||
|
|
|||
|
protected:
|
|||
|
TDocumento& doc() const { return *_doc;}
|
|||
|
|
|||
|
public:
|
|||
|
void set_normal_next(bool b = true) { _normal_next = b; }
|
|||
|
|
|||
|
virtual int next(word lockop = _nolock) { return _normal_next ? TLocalisamfile::next(lockop) : NOERR; }
|
|||
|
virtual TRectype& curr() const ;
|
|||
|
void set_row(int r) { _row = r;}
|
|||
|
TRDocisamfile(TDocumento* doc) : TLocalisamfile(LF_RIGHEDOC), _doc(doc), _row(1), _normal_next(false) {}
|
|||
|
|
|||
|
virtual ~TRDocisamfile() {};
|
|||
|
};
|
|||
|
|
|||
|
TRectype& TRDocisamfile::curr() const
|
|||
|
{
|
|||
|
TRectype& rr = ((_row > 0 && _row <= _doc->rows()) ? (TRectype&) doc()[_row] : TLocalisamfile::curr());
|
|||
|
return rr;
|
|||
|
}
|
|||
|
////////////////////////////////////////////////////////////////////////////
|
|||
|
// classe TDocumento_form customizzata dalla Form per i documenti
|
|||
|
////////////////////////////////////////////////////////////////////////////
|
|||
|
|
|||
|
class TDocumento_form : public TForm
|
|||
|
{
|
|||
|
static TDocumento_form* _form;
|
|||
|
TRelation &_firmrel; // relazione di gestione dei dati della ditta corrente
|
|||
|
TString _module; // codice del modulo di carta associato a questo al form
|
|||
|
|
|||
|
TString_array _exclude_array; // array di tipi riga e articoli da escludere dalla stampa
|
|||
|
TAssoc_array _doc_totals; // Assocarray per codice numerazione contenente i totali nel caso di stampa lista documenti
|
|||
|
TString_array _group_decimals; // Array di TToken_string per ogni gruppo definito in GENERAL.
|
|||
|
// Il primo elelemento della token_string conterra' il numero del gruppo
|
|||
|
// il secondo il n.ro di decimali per importi in lire ed il terzo il n.ro
|
|||
|
// di decimali per gli importi in valuta
|
|||
|
|
|||
|
TSorted_cursor * _sorted_cur; // Valido solo per i form di lista documenti
|
|||
|
TDocisamfile* _docfile;
|
|||
|
TRDocisamfile* _rdocfile;
|
|||
|
TDocumentoEsteso * _doc; // Documento da stampare
|
|||
|
bool _valid, _cli_loaded; // flag che indica se il form e' valido | se l'oggetto cliente <20> gi<67> stato caricato
|
|||
|
|
|||
|
// I gruppi sono cosi' predefiniti:
|
|||
|
// PRI_DECIMALS corrisponde al gruppo 29
|
|||
|
// QTA_DECIMALS corrisponde al gruppo 30
|
|||
|
// IMP_DECIMALS corrisponde al gruppo 31
|
|||
|
// Altri gruppi definiti dall'utente saranno cosi' sintatticamente impostati:
|
|||
|
// NEW_GROUP <n> <lit_dec> <val_dec>
|
|||
|
// Dove <n> e' il numero del gruppo
|
|||
|
// <lit_dec> e' il numero di decimali per i documenti in lire
|
|||
|
// <val_dec> e' il numero di decimali per i documenti in valuta
|
|||
|
// ATTENZIONE: e' importante che i nomi dei gruppi utilizzati per modificare le pictures non siano usati per
|
|||
|
// per altri messaggi. Inoltre un TForm_item che appartiene ad un gruppo di modifica picture
|
|||
|
// non puo' appartenere ad un altro gruppo dello stesso tipo, ad esempio i gruppi 29 e 30 contemporaneamente.
|
|||
|
// Puo' pero' appartenere anche ad altri gruppi che non siano utilizzati per lo scopo qui definito
|
|||
|
protected:
|
|||
|
virtual void extended_parse_general(TScanner &); // gestione dei parametri estesi nella sezione general
|
|||
|
virtual bool validate(TForm_item &, TToken_string &); // gestione dei messaggi estesi nei campi
|
|||
|
bool print_on_body(int r); // Trascrive la riga 'r' del documento sul body. Ritorna true se va stampata, false se va saltata
|
|||
|
|
|||
|
void print_header(TPrinter& p); // Stampa la testata
|
|||
|
void print_footer(TPrinter& p); // Stampa la pedata
|
|||
|
|
|||
|
static void doc_header_handler(TPrinter& p);
|
|||
|
static void doc_footer_handler(TPrinter& p);
|
|||
|
void output_values(const TRectype & rec, const char * output, TForm_item & cf);
|
|||
|
|
|||
|
public:
|
|||
|
void edit_picture(TForm_item & f, const int dec);
|
|||
|
void modify_pictures();
|
|||
|
virtual TCursor* cursor() const { return _sorted_cur ? _sorted_cur : TForm::cursor(); }
|
|||
|
void hide_sections();
|
|||
|
bool is_faketotfld();
|
|||
|
void print_documento();
|
|||
|
bool valid() const { return _valid; }
|
|||
|
bool doc_arrange();
|
|||
|
int ncopie() const { return _doc->tipo().ncopie(); }
|
|||
|
const TString &get_module_code() const { return _module; } // ritorna il codice del modulo di carta
|
|||
|
TString_array& exclude_list() { return _exclude_array; }
|
|||
|
|
|||
|
TDocumentoEsteso& doc() { return *_doc; }
|
|||
|
void set_doc_ext(TRectype* doc);
|
|||
|
|
|||
|
TDocumento_form(TRectype& doc, TRelation& rel, bool definitiva, bool interattivo, bool aggiuntivo);
|
|||
|
TDocumento_form(const char* form, TRelation& rel);
|
|||
|
virtual ~TDocumento_form();
|
|||
|
};
|
|||
|
|
|||
|
TDocumento_form* TDocumento_form::_form = NULL;
|
|||
|
|
|||
|
void TDocumento_form::set_doc_ext(TRectype* doc)
|
|||
|
{
|
|||
|
CHECK(_doc == NULL, "Doppio documeto esteso");
|
|||
|
|
|||
|
if (doc != NULL)
|
|||
|
_doc = new TDocumentoEsteso(*doc);
|
|||
|
else
|
|||
|
_doc = new TDocumentoEsteso;
|
|||
|
|
|||
|
_docfile = new TDocisamfile(_doc);
|
|||
|
_rdocfile = new TRDocisamfile(_doc);
|
|||
|
_rdocfile->set_normal_next();
|
|||
|
relation()->replace(_docfile,0);
|
|||
|
relation()->replace(_rdocfile,1);
|
|||
|
|
|||
|
if (_doc->physical_rows() > 0)
|
|||
|
relation()->update();
|
|||
|
}
|
|||
|
|
|||
|
TDocumento_form::TDocumento_form(TRectype& doc, TRelation& rel, bool definitiva, bool interattivo, bool aggiuntivo)
|
|||
|
: _firmrel(rel), _sorted_cur(NULL), _docfile(NULL), _rdocfile(NULL), _doc(NULL), _valid(false)
|
|||
|
{
|
|||
|
_form = this;
|
|||
|
|
|||
|
const TString4 tipodoc(doc.get(DOC_TIPODOC));
|
|||
|
bool found = false;
|
|||
|
TFilename nomeform;
|
|||
|
const TTipo_documento& tipo = cached_tipodoc(tipodoc);
|
|||
|
|
|||
|
if (tipo.empty())
|
|||
|
{
|
|||
|
error_box(FR("Tipo di documento non valido: '%s'"), (const char*)tipodoc);
|
|||
|
return;
|
|||
|
}
|
|||
|
if (tipo.printable())
|
|||
|
{ // se non ci sono errori procede con la stampa
|
|||
|
if (aggiuntivo)
|
|||
|
tipo.additional_print_profile(nomeform, 1);
|
|||
|
else
|
|||
|
tipo.main_print_profile(nomeform, 1); // legge il nome del form di stampa
|
|||
|
nomeform.trim();
|
|||
|
|
|||
|
TFilename test(nomeform); test.ext("frm");
|
|||
|
if (!test.custom_path())
|
|||
|
{
|
|||
|
error_box(FR("Nome form di stampa '%s' non valido per il tipo documento '%s'"),
|
|||
|
(const char*)nomeform, (const char*)tipodoc);
|
|||
|
return;
|
|||
|
}
|
|||
|
}
|
|||
|
else
|
|||
|
return;
|
|||
|
|
|||
|
_valid = true;
|
|||
|
read(nomeform);
|
|||
|
_cli_loaded= false;
|
|||
|
set_doc_ext(&doc); // istanzia TDocumentoEsteso
|
|||
|
modify_pictures();
|
|||
|
|
|||
|
dec_parm p;
|
|||
|
const int items = _group_decimals.items();
|
|||
|
|
|||
|
for (int i = 0; i< items; i++)
|
|||
|
{
|
|||
|
TToken_string& t = _group_decimals.row(i);
|
|||
|
int gruppo = t.get_int(0);
|
|||
|
switch (gruppo)
|
|||
|
{
|
|||
|
case GROUP_QTA : p.qta_lit = t.get_int(1);p.qta_val = t.get_int(2);
|
|||
|
break;
|
|||
|
// add other groups here
|
|||
|
default:
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
_doc->set_decimals(p);
|
|||
|
|
|||
|
// Inizializza lo sfondo delle pagine normali
|
|||
|
set_background(3, true);
|
|||
|
|
|||
|
TPrinter& pr = printer();
|
|||
|
|
|||
|
pr.setheaderhandler(doc_header_handler);
|
|||
|
TPrint_section& head = section('H');
|
|||
|
pr.headerlen(head.height());
|
|||
|
|
|||
|
pr.setfooterhandler(doc_footer_handler);
|
|||
|
const TPrint_section& foot = section('F');
|
|||
|
pr.footerlen(foot.height());
|
|||
|
}
|
|||
|
|
|||
|
// costruttore per stampa lista documenti (uso convenzionale dei forms)
|
|||
|
TDocumento_form::TDocumento_form(const char* form, TRelation& rel)
|
|||
|
: TForm(form), _firmrel(rel), _docfile(NULL), _rdocfile(NULL), _doc(NULL), _valid(false)
|
|||
|
{
|
|||
|
_cli_loaded= false;
|
|||
|
_sorted_cur = new TSorted_cursor(relation(), DOC_PROVV"|"DOC_ANNO"|"DOC_CODNUM"|"DOC_DATADOC"|"DOC_NDOC);
|
|||
|
}
|
|||
|
|
|||
|
TDocumento_form::~TDocumento_form()
|
|||
|
{
|
|||
|
// Membri di cui NON va fatta la delete:
|
|||
|
// _docfile : perche' viene fatta dal distruttore di TRelation
|
|||
|
// _rdocfile : perche' viene fatta dal distruttore di TRelation
|
|||
|
if (_doc) delete _doc;
|
|||
|
if (_sorted_cur) delete _sorted_cur;
|
|||
|
}
|
|||
|
|
|||
|
void TDocumento_form::hide_sections()
|
|||
|
{
|
|||
|
// Scorre tutte le sezioni e nasconde gli items
|
|||
|
const char s[3] = { 'B', 'G', 'H' };
|
|||
|
for (int sn = 0; sn < 3 ; sn++)
|
|||
|
{
|
|||
|
const char sc = s[sn];
|
|||
|
for (pagetype pt = odd_page; pt <= last_page; pt = pagetype(pt+1))
|
|||
|
{
|
|||
|
TPrint_section* sec = exist(sc, pt);
|
|||
|
if (sec == NULL)
|
|||
|
continue;
|
|||
|
TForm_item* f;
|
|||
|
for(word i = 0; i < sec->fields(); i++)
|
|||
|
{
|
|||
|
f = &(sec->field(i));
|
|||
|
f->hide();
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
bool TDocumento_form::is_faketotfld()
|
|||
|
{
|
|||
|
TPrint_section* fl = exist('F',last_page);
|
|||
|
if (fl != NULL)
|
|||
|
{
|
|||
|
TForm_item* f;
|
|||
|
for(word i = 0; i < fl->fields(); i++)
|
|||
|
{
|
|||
|
f = &(fl->field(i));
|
|||
|
if (f->id() == FAKETOTFLD)
|
|||
|
return true;
|
|||
|
}
|
|||
|
}
|
|||
|
return false;
|
|||
|
}
|
|||
|
|
|||
|
bool TDocumento_form::doc_arrange()
|
|||
|
{
|
|||
|
TPrinter& pr = printer();
|
|||
|
|
|||
|
if (char_to_pos() != '\0' || (ipx()+ipy()+fpx()) != 0)
|
|||
|
{
|
|||
|
if (offset_x() != 0 || offset_y() != 0)
|
|||
|
{
|
|||
|
error_box(FR("Non <20> possibile impostare contemporaneamente gli offset"
|
|||
|
" e i parametri di posizionamento del modulo %s."), (const char*)name());
|
|||
|
return false;
|
|||
|
}
|
|||
|
else
|
|||
|
if (pr.printtype() == winprinter)
|
|||
|
_form->arrange_form();
|
|||
|
}
|
|||
|
else
|
|||
|
pr.set_offset(_form->offset_y(), _form->offset_x());
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
void TDocumento_form::print_documento()
|
|||
|
{
|
|||
|
TPrinter& pr = printer();
|
|||
|
|
|||
|
// stampa tutte le righe
|
|||
|
TPrint_section& body = section('B');
|
|||
|
// TPrint_section& foot = section('F'); // qui verificare
|
|||
|
TPrint_section* sect = exist('B', last_page);
|
|||
|
TString last_section;
|
|||
|
|
|||
|
const int righe = _doc->rows();
|
|||
|
bool one_row_printed = false;
|
|||
|
|
|||
|
const bool update_relation = relation()->has_children(1);
|
|||
|
|
|||
|
set_last_page(false); // E' importante settare questo flag, per evitare "Falli di Piede" eheh :-)
|
|||
|
for (int r=1; r<=righe; r++)
|
|||
|
{
|
|||
|
_rdocfile->set_row(r);
|
|||
|
|
|||
|
if (update_relation)
|
|||
|
relation()->update(1);
|
|||
|
|
|||
|
if (!print_on_body(r))
|
|||
|
continue;
|
|||
|
if (sect)
|
|||
|
{
|
|||
|
sect->update();
|
|||
|
const TString& curr_section = sect->field(0).get();
|
|||
|
if (r == 1 || curr_section != last_section)
|
|||
|
{
|
|||
|
last_section = curr_section;
|
|||
|
const word h = sect->height();
|
|||
|
for (word j = 0; j < h; j++)
|
|||
|
pr.print(sect->row(j));
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
const word h = body.height();
|
|||
|
for (word j = 0; j < h; j++)
|
|||
|
pr.print(body.row(j));
|
|||
|
|
|||
|
if (!one_row_printed)
|
|||
|
one_row_printed = true;
|
|||
|
|
|||
|
if ((*_doc)[r].tipo().formfeed())
|
|||
|
pr.formfeed();
|
|||
|
}
|
|||
|
|
|||
|
if (!one_row_printed)
|
|||
|
{
|
|||
|
// Riga fasulla... per stampare l'intestazione obbligatoriamente,
|
|||
|
// anche in caso che non vi siano righe nel documento o che siano tutte escluse
|
|||
|
TPrintrow r;
|
|||
|
pr.print(r);
|
|||
|
}
|
|||
|
|
|||
|
if (_doc->tipo().add_conai() && _doc->clifor().vendite().get_bool("CONAIASS"))
|
|||
|
{
|
|||
|
TRiga_documento last_row(_doc);
|
|||
|
|
|||
|
last_row = _rdocfile->curr();
|
|||
|
_rdocfile->zero();
|
|||
|
const TString& desc = ini_get_string(CONFIG_DITTA, "ve", "DESCCONAIASS");
|
|||
|
if (desc.blank())
|
|||
|
_rdocfile->put(RDOC_DESCR, TR("Contributo CONAI assolto"));
|
|||
|
else
|
|||
|
_rdocfile->put(RDOC_DESCR, desc);
|
|||
|
body.update();
|
|||
|
const word h = body.height();
|
|||
|
if (pr.rows_left() <= h+1) // salto pagina
|
|||
|
pr.formfeed();
|
|||
|
|
|||
|
for (word j = 0; j < h; j++)
|
|||
|
pr.print(body.row(j));
|
|||
|
|
|||
|
_rdocfile->curr() = last_row;
|
|||
|
}
|
|||
|
|
|||
|
TPrint_section* last_foot = exist('F', last_page, false);
|
|||
|
if (last_foot != NULL)
|
|||
|
{
|
|||
|
const word lfh = last_foot->height();
|
|||
|
const word left = pr.rows_left() + pr.footersize();
|
|||
|
if (lfh > left) // Se l'ultimo footer e' troppo grande ...
|
|||
|
pr.formfeed(); // Stampa il footer normale
|
|||
|
pr.footerlen(lfh); // Fondamentale!
|
|||
|
}
|
|||
|
set_last_page(true);
|
|||
|
pr.formfeed(); // Stampa l'ultimo footer
|
|||
|
|
|||
|
// Rimette ad 1 il numero della pagina
|
|||
|
pr.setcurrentpage(1);
|
|||
|
}
|
|||
|
|
|||
|
void TDocumento_form::print_header(TPrinter& pr)
|
|||
|
{
|
|||
|
TPrint_section& head = section('H');
|
|||
|
head.update();
|
|||
|
|
|||
|
const word r = head.height()-1;
|
|||
|
for (word j = 0; j <= r; j++)
|
|||
|
pr.setheaderline(j, head.row(j));
|
|||
|
}
|
|||
|
|
|||
|
void TDocumento_form::print_footer(TPrinter& pr)
|
|||
|
{
|
|||
|
const bool p = _form->page(pr)>0;
|
|||
|
TPrint_section& foot = section('F',p ? odd_page : last_page);
|
|||
|
|
|||
|
foot.update();
|
|||
|
word r = foot.height();
|
|||
|
|
|||
|
for (word j = 0; j < r; j++)
|
|||
|
pr.setfooterline(j, foot.row(j));
|
|||
|
}
|
|||
|
|
|||
|
void TDocumento_form::doc_header_handler(TPrinter& pr)
|
|||
|
{
|
|||
|
pr.resetheader();
|
|||
|
_form->print_header(pr);
|
|||
|
}
|
|||
|
|
|||
|
void TDocumento_form::doc_footer_handler(TPrinter& pr)
|
|||
|
{
|
|||
|
pr.resetfooter();
|
|||
|
_form->print_footer(pr);
|
|||
|
}
|
|||
|
|
|||
|
void TDocumento_form::edit_picture(TForm_item & fi, const int dec)
|
|||
|
{
|
|||
|
const TString old_picture = fi.picture();
|
|||
|
TString new_picture(20);
|
|||
|
const int comma_pos = old_picture.find(',');
|
|||
|
const int stop_pos = old_picture.find('.');
|
|||
|
char migliaia_char = '.';
|
|||
|
|
|||
|
if (comma_pos > 0 && stop_pos > 0)
|
|||
|
{
|
|||
|
if (stop_pos > comma_pos)
|
|||
|
migliaia_char = ',';
|
|||
|
}
|
|||
|
|
|||
|
if (old_picture.empty()) // picture di default
|
|||
|
{
|
|||
|
new_picture = "."; // in lire
|
|||
|
if (dec > 0) new_picture << dec; // in valuta
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
if (dec == 0) return; // 0 non cambia la picture
|
|||
|
|
|||
|
new_picture = old_picture;
|
|||
|
|
|||
|
// Resetta la picture: toglie eventuali decimali gia' presenti...
|
|||
|
const int pos = new_picture.rfind(migliaia_char == '.' ? ',' : '.');
|
|||
|
if (pos >= 0)
|
|||
|
new_picture.cut(pos);
|
|||
|
|
|||
|
// Se si decide di metterne... altrimenti ndec -1 lascia la picture senza decimali
|
|||
|
if (dec > 0)
|
|||
|
{
|
|||
|
TString16 dec_to_add;
|
|||
|
for (int i = 0; i < dec; i++)
|
|||
|
dec_to_add << "@"; // aggiunge tanti "@" quanti sono i decimali voluti
|
|||
|
if (migliaia_char == ',')
|
|||
|
new_picture << "."; // se ha trovato la virgola come separatore di migliaia significa che deve aggiungere il punto decimale
|
|||
|
else
|
|||
|
new_picture << ","; // altrimenti aggiunge la solita virgola
|
|||
|
new_picture << dec_to_add; // infine aggiunge i decimali richiesti
|
|||
|
}
|
|||
|
}
|
|||
|
const int w = old_picture.len(); // se la picture eccede la dimensione della picture, toglie i caratteri piu' a sx
|
|||
|
int exceed = w - new_picture.len();
|
|||
|
if (exceed<0 && w>0)
|
|||
|
{
|
|||
|
exceed=::abs(exceed);
|
|||
|
new_picture = new_picture.mid(exceed,new_picture.len()-exceed);
|
|||
|
if (new_picture[0] == migliaia_char)
|
|||
|
new_picture.ltrim(1);
|
|||
|
}
|
|||
|
fi.set_picture(new_picture); // setta la nuova picture
|
|||
|
}
|
|||
|
|
|||
|
void TDocumento_form::modify_pictures()
|
|||
|
{
|
|||
|
const bool valuta = _doc->in_valuta();
|
|||
|
const char sechar[4] = { 'B', 'F', 'G', 'H' };
|
|||
|
for (int sn = 0; sn < 4 ; sn++)
|
|||
|
{
|
|||
|
const char sc = sechar[sn];
|
|||
|
for (pagetype pt = odd_page; pt <= last_page; pt = pagetype(pt+1))
|
|||
|
{
|
|||
|
TPrint_section* sec = exist(sc, pt);
|
|||
|
if (sec != NULL)
|
|||
|
for (word i = 0; i < sec->fields() ; i++)
|
|||
|
{
|
|||
|
TForm_item& fi = sec->field(i);
|
|||
|
if (fi.in_group(GROUP_PRICES))
|
|||
|
edit_picture(fi, _doc->decimals(true));
|
|||
|
else
|
|||
|
if (fi.in_group(GROUP_IMPORTI))
|
|||
|
edit_picture(fi, _doc->decimals());
|
|||
|
else
|
|||
|
{
|
|||
|
const int items = _group_decimals.items(); // numero di gruppi definiti
|
|||
|
for (int j = 0; j < items; j++)
|
|||
|
{
|
|||
|
TToken_string& r = _group_decimals.row(j);
|
|||
|
const int group = r.get_int(0);
|
|||
|
if (fi.in_group(group)) // trova se appartiene al gruppo, modifica la picture
|
|||
|
{
|
|||
|
edit_picture(fi,valuta ? r.get_int(2) : r.get_int(1));
|
|||
|
break; // considera solo il primo gruppo trovato
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
bool TDocumento_form::print_on_body(int r)
|
|||
|
{
|
|||
|
TPrint_section& body = section('B');
|
|||
|
TRiga_documento& riga = doc()[r];
|
|||
|
const TString & tiporiga = riga.get(RDOC_TIPORIGA);
|
|||
|
bool ok = true;
|
|||
|
|
|||
|
FOR_EACH_ARRAY_ROW(_exclude_array, i, row)
|
|||
|
{
|
|||
|
if (row->starts_with(tiporiga))
|
|||
|
{
|
|||
|
const int pos = row->find(',');
|
|||
|
|
|||
|
if (pos <= 0)
|
|||
|
ok = false;
|
|||
|
else
|
|||
|
{
|
|||
|
const TString & codart = riga.get(RDOC_CODART);
|
|||
|
const TString art_to_exclude(row->mid(pos+1));
|
|||
|
|
|||
|
ok = art_to_exclude.not_empty() && (art_to_exclude.find(codart) < 0);
|
|||
|
}
|
|||
|
if (!ok)
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (ok)
|
|||
|
body.update(); // Crea la vera riga di stampa, eventuali allineamenti avverranno nella validate(), come al solito.
|
|||
|
|
|||
|
return ok;
|
|||
|
}
|
|||
|
|
|||
|
void TDocumento_form::extended_parse_general(TScanner &scanner)
|
|||
|
{
|
|||
|
// se viene riconosciuto il token per l'impostazione del modulo legge il codice...
|
|||
|
if (scanner.key() == "MO")
|
|||
|
_module= scanner.string();
|
|||
|
|
|||
|
// Legge i decimali necessari per gli arrotondamenti (il primo per gli importi in lire, l'altro per quelli in valuta)
|
|||
|
// if (scanner.key() == "PR")
|
|||
|
// {
|
|||
|
// TToken_string t;
|
|||
|
// t.add(GROUP_PRICES);t.add(scanner.integer());t.add(scanner.integer());
|
|||
|
// _group_decimals.add(t);
|
|||
|
// }
|
|||
|
|
|||
|
// Stessa cosa per le quantita'
|
|||
|
if (scanner.key() == "QT")
|
|||
|
{
|
|||
|
TToken_string t;
|
|||
|
t.add(GROUP_QTA);t.add(scanner.integer());t.add(scanner.integer());
|
|||
|
_group_decimals.add(t);
|
|||
|
}
|
|||
|
|
|||
|
// Stessa cosa per gli importi in genere
|
|||
|
// if (scanner.key() == "IM")
|
|||
|
// {
|
|||
|
// TToken_string t;
|
|||
|
// t.add(GROUP_IMPORTI);t.add(scanner.integer());t.add(scanner.integer());
|
|||
|
// _group_decimals.add(t);
|
|||
|
// }
|
|||
|
|
|||
|
if (scanner.key() == "NE")
|
|||
|
{
|
|||
|
TToken_string t;
|
|||
|
t.add(scanner.integer());t.add(scanner.integer());t.add(scanner.integer());
|
|||
|
_group_decimals.add(t);
|
|||
|
}
|
|||
|
|
|||
|
// Esclude certi tipi riga e codici articolo
|
|||
|
if (scanner.key() == "EX")
|
|||
|
{
|
|||
|
TToken_string s(scanner.string(),',');
|
|||
|
_exclude_array.add(s);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void TDocumento_form::output_values(const TRectype & rec, const char * output, TForm_item & cf)
|
|||
|
{
|
|||
|
TToken_string out(output, '!');
|
|||
|
TString curr;
|
|||
|
for (const char * str = out.get(0); str; str = out.get())
|
|||
|
{ // scansione sugli elementi dell'output
|
|||
|
curr = str;
|
|||
|
int poseq = curr.find('='); // divide la stringa corrente in lvalue e rvalue
|
|||
|
if (poseq < 0)
|
|||
|
cf.set(rec.get(curr));
|
|||
|
else
|
|||
|
{
|
|||
|
int posrv = poseq+1;
|
|||
|
if (poseq >= 0 && curr[posrv] == '=')
|
|||
|
posrv++;
|
|||
|
TString16 fld(curr.left(poseq)); // preleva il nome del campo del form alla sinistra dell'uguale
|
|||
|
const TString dat(rec.get(curr.mid(posrv))); // preleva il nome del campo del file alla destra dell'uguale e lo legge dal record
|
|||
|
if (fld[0] == '#') fld.ltrim(1);
|
|||
|
|
|||
|
if (fld.right(1) == "@")
|
|||
|
{ // se c'<27> la a-commerciale <20> un gruppo
|
|||
|
char sec = cf.section().section_type();
|
|||
|
pagetype pt = cf.section().page_type();
|
|||
|
int group = atoi(fld);
|
|||
|
|
|||
|
if (fld.find("->") >= 0)
|
|||
|
{ // se nel gruppo c'<27> la freccia si riferisce ad un'altra sezione
|
|||
|
sec = fld[0];
|
|||
|
pt= (fld[1] != '-') ? char2page(fld[1]) : even_page;
|
|||
|
}
|
|||
|
|
|||
|
TPrint_section &fs = section(sec, pt);
|
|||
|
word itms = fs.fields();
|
|||
|
|
|||
|
for (word j=0; j<itms; j++)
|
|||
|
{
|
|||
|
TForm_item & fi = fs.field(j);
|
|||
|
|
|||
|
if (fi.in_group(group))
|
|||
|
fi.set(dat);
|
|||
|
}
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
TForm_item & fi= cf.find_field(fld);
|
|||
|
fi.set(dat);
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
bool TDocumento_form::validate(TForm_item &cf, TToken_string &s)
|
|||
|
{
|
|||
|
const TString code(s.get(0)); // prende il primo parametro, il codice del messaggio
|
|||
|
TString valore;
|
|||
|
|
|||
|
if (code== "_DITTA")
|
|||
|
{
|
|||
|
// lettura dei dati della ditta
|
|||
|
// sintassi: _DITTA,{<campo relazione>|<macro>}
|
|||
|
// dove: <campo relazione> <20> un riferimento alla relazione di gestione dei dati della ditta (es. 113@->DENCOM <20> la denominazione del comune di residenza della ditta)
|
|||
|
// <macro> <20> uno delle macro seguenti:
|
|||
|
// !RAGSOC ragione sociale
|
|||
|
// !IND indirizzo (fiscale se c'<27>, oppure di residenza)
|
|||
|
// !NUM numero civico (fiscale se c'<27>, oppure di residenza)
|
|||
|
// !CAP CAP (fiscale se c'<27>, oppure di residenza)
|
|||
|
// !COM comune (fiscale se c'<27>, oppure di residenza)
|
|||
|
// !PROV provincia (fiscale se c'<27>, oppure di residenza)
|
|||
|
// !IVA partita iva
|
|||
|
// !CF codice fiscale
|
|||
|
// !TEL numero di telefono (con prefisso)
|
|||
|
// !FAX numero di fax (con prefisso)
|
|||
|
// !REGSOC numero di registrazione presso il Tribunale
|
|||
|
// !CCIAA numero di registrazione presso la camera di commercio
|
|||
|
// nota: la relazione della ditta <20> cos<6F> strutturata:
|
|||
|
// %NDITTE (9) Dati ditte
|
|||
|
// + %ANAGR (6) Anagrafica generale (indirizzo, ecc.)
|
|||
|
// + %COMUNI (113@) Comune di residenza
|
|||
|
// + %COMUNI (213@) Comune di residenza fiscale
|
|||
|
TString in(s.get());
|
|||
|
|
|||
|
if (in[0]!='!')
|
|||
|
{
|
|||
|
cf.set(_firmrel.curr().get(in));
|
|||
|
return true;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
in.ltrim(1);
|
|||
|
const bool _fisc= _firmrel.lfile(6).get("INDRF").not_empty();
|
|||
|
if (in=="RAGSOC")
|
|||
|
{
|
|||
|
cf.set(_firmrel.lfile().get("RAGSOC"));
|
|||
|
return true;
|
|||
|
}
|
|||
|
if (in=="IND")
|
|||
|
{
|
|||
|
if (_fisc)
|
|||
|
cf.set(_firmrel.lfile(6).get("INDRF"));
|
|||
|
else
|
|||
|
cf.set(_firmrel.lfile(6).get("INDRES"));
|
|||
|
return true;
|
|||
|
}
|
|||
|
if (in=="NUM")
|
|||
|
{
|
|||
|
if (_fisc)
|
|||
|
cf.set(_firmrel.lfile(6).get("CIVRF"));
|
|||
|
else
|
|||
|
cf.set(_firmrel.lfile(6).get("CIVRES"));
|
|||
|
return true;
|
|||
|
}
|
|||
|
if (in=="CAP")
|
|||
|
{
|
|||
|
if (_fisc)
|
|||
|
cf.set(_firmrel.lfile(6).get("CAPRF"));
|
|||
|
else
|
|||
|
cf.set(_firmrel.lfile(6).get("CAPRES"));
|
|||
|
return true;
|
|||
|
}
|
|||
|
if (in=="COM")
|
|||
|
{
|
|||
|
if (_fisc)
|
|||
|
cf.set(_firmrel.lfile(-213).get("DENCOM"));
|
|||
|
else
|
|||
|
cf.set(_firmrel.lfile(-113).get("DENCOM"));
|
|||
|
return true;
|
|||
|
}
|
|||
|
if (in=="PROV")
|
|||
|
{
|
|||
|
if (_fisc)
|
|||
|
cf.set(_firmrel.lfile(-213).get("PROVCOM"));
|
|||
|
else
|
|||
|
cf.set(_firmrel.lfile(-113).get("PROVCOM"));
|
|||
|
return true;
|
|||
|
}
|
|||
|
if (in=="IVA")
|
|||
|
{
|
|||
|
cf.set(_firmrel.lfile(6).get("PAIV"));
|
|||
|
return true;
|
|||
|
}
|
|||
|
if (in=="CF")
|
|||
|
{
|
|||
|
cf.set(_firmrel.lfile(6).get("COFI"));
|
|||
|
return true;
|
|||
|
}
|
|||
|
if (in=="TEL")
|
|||
|
{
|
|||
|
valore = _firmrel.lfile().get("PTEL");
|
|||
|
valore << "/" << _firmrel.lfile().get("TEL");
|
|||
|
cf.set(valore);
|
|||
|
return true;
|
|||
|
}
|
|||
|
if (in=="FAX")
|
|||
|
{
|
|||
|
valore = _firmrel.lfile().get("PFAX");
|
|||
|
valore << "/" << _firmrel.lfile().get("FAX");
|
|||
|
cf.set(valore);
|
|||
|
return true;
|
|||
|
}
|
|||
|
if (in=="REGSOC")
|
|||
|
{
|
|||
|
valore = _firmrel[LF_UNLOC].get("REGIMP");
|
|||
|
valore.insert(" ", 2); valore.insert(" ", 6);
|
|||
|
valore.insert(" ", 11); valore.insert(" ", 21);
|
|||
|
valore.insert("Reg.Imp. ", 0);
|
|||
|
cf.set(valore);
|
|||
|
return true;
|
|||
|
}
|
|||
|
if (in=="CCIAA")
|
|||
|
{
|
|||
|
valore = _firmrel[LF_UNLOC].get("NUMCCIAA");
|
|||
|
const TString & data = _firmrel[LF_UNLOC].get("DATAICCIAA");
|
|||
|
if (data.not_empty())
|
|||
|
valore << " del " << data;
|
|||
|
cf.set(valore);
|
|||
|
return true;
|
|||
|
}
|
|||
|
}
|
|||
|
} // fine _DITTA
|
|||
|
|
|||
|
if (code== "_CLIENTE")
|
|||
|
{
|
|||
|
// lettura dei dati del cliente
|
|||
|
// sintassi: _CLIENTE,{<campo relazione>|<macro>}
|
|||
|
// dove: <campo relazione> <20> un riferimento alla relazione di gestione dei dati del cliente
|
|||
|
// <macro> <20> uno delle macro seguenti:
|
|||
|
// !RAGSOC ragione sociale
|
|||
|
// !CAP Codice Avviamento Postale (viene implementato un messaggio perche' sugli occasionali ha un nome campo diverso!!)
|
|||
|
// !IND indirizzo
|
|||
|
// !NUM numero civico
|
|||
|
// !INDNUM indirizzo + numero civico
|
|||
|
// !COM comune
|
|||
|
// !PROV provincia
|
|||
|
// !TEL primo numero di telefono (con prefisso)
|
|||
|
// !TEL2 secondo numero di telefono (con prefisso)
|
|||
|
// !TEL3 terzo numero di telefono (con prefisso)
|
|||
|
// !FAX numero di fax (con prefisso)
|
|||
|
// !COM-><FIELD> accede ai campi del comune di residenza cliente
|
|||
|
// !COMN-><FIELD> accede ai campi del comune di nascita del cliente
|
|||
|
TCli_for & cli_for = _doc->clifor();
|
|||
|
TOccasionale & cli_occ = _doc->occas();
|
|||
|
const bool occasionale = cli_for.occasionale();
|
|||
|
TString in(s.get()); // prende la macro o il fieldref
|
|||
|
if (in[0] != '!')
|
|||
|
{
|
|||
|
// Controlla l'esistenza dei campi...
|
|||
|
if (occasionale && cli_occ.exist(in))
|
|||
|
valore = cli_occ.get(in);
|
|||
|
|
|||
|
if (!occasionale && cli_for.exist(in))
|
|||
|
valore = cli_for.get(in);
|
|||
|
|
|||
|
cf.set(valore);
|
|||
|
return true;
|
|||
|
}
|
|||
|
in.ltrim(1);
|
|||
|
if (in=="INDNUM")
|
|||
|
{
|
|||
|
valore = occasionale ? cli_occ.get(OCC_INDIR) : cli_for.get(CLI_INDCF);
|
|||
|
valore << " " ;
|
|||
|
valore << (occasionale ? cli_occ.get(OCC_CIV) : cli_for.get(CLI_CIVCF));
|
|||
|
cf.set(valore);
|
|||
|
return true;
|
|||
|
}
|
|||
|
if (in.find("COM") == 0)
|
|||
|
{
|
|||
|
const bool nascita = in[3] == 'N';
|
|||
|
const int p = in.find("->");
|
|||
|
if (p > 0)
|
|||
|
in.ltrim(p + 2);
|
|||
|
TLocalisamfile com(LF_COMUNI);
|
|||
|
if (nascita)
|
|||
|
{
|
|||
|
com.put(COM_STATO, occasionale ? cli_occ.get(OCC_STATONASC) : cli_for.get(CLI_STATONASC));
|
|||
|
com.put(COM_COM, occasionale ? cli_occ.get(OCC_COMNASC) : cli_for.get(CLI_COMNASC));
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
com.put(COM_STATO, occasionale ? cli_occ.get(OCC_STATO): cli_for.get(CLI_STATOCF));
|
|||
|
com.put(COM_COM, occasionale ? cli_occ.get(OCC_COM): cli_for.get(CLI_COMCF));
|
|||
|
}
|
|||
|
if (com.read() == NOERR)
|
|||
|
cf.set(com.get(in));
|
|||
|
return true;
|
|||
|
}
|
|||
|
if (in.find("CAP") == 0)
|
|||
|
{
|
|||
|
valore = occasionale ? cli_occ.get(OCC_CAP) : cli_for.get(CLI_CAPCF);
|
|||
|
cf.set(valore);
|
|||
|
return true;
|
|||
|
}
|
|||
|
if (in.find("TEL") == 0)
|
|||
|
{
|
|||
|
if (!occasionale)
|
|||
|
{
|
|||
|
if (in.len() == 3)
|
|||
|
in << "1";
|
|||
|
const TString num(cli_for.get(in));
|
|||
|
in.insert("P");
|
|||
|
valore = cli_for.get(in);
|
|||
|
valore << "/" << num;
|
|||
|
}
|
|||
|
cf.set(valore);
|
|||
|
return true;
|
|||
|
}
|
|||
|
if (in=="FAX")
|
|||
|
{
|
|||
|
if (!occasionale)
|
|||
|
{
|
|||
|
valore = cli_for.get("PFAX");
|
|||
|
valore << "/" << cli_for.get("FAX");
|
|||
|
}
|
|||
|
cf.set(valore);
|
|||
|
return true;
|
|||
|
}
|
|||
|
if (in=="RAGSOC")
|
|||
|
{
|
|||
|
valore = occasionale ? cli_occ.get(in) : cli_for.get(in);
|
|||
|
valore.strip_double_spaces();
|
|||
|
cf.set(valore);
|
|||
|
return true;
|
|||
|
}
|
|||
|
} // fine _CLIENTE
|
|||
|
|
|||
|
if (code == "_DESCRIGA")
|
|||
|
{
|
|||
|
// Messaggio per reperire la descrizione estesa sulle righe del documento
|
|||
|
const TRectype &rdoc= cursor()->curr(LF_RIGHEDOC);
|
|||
|
TString descrizione = rdoc.get(RDOC_DESCR);
|
|||
|
const bool desclunga = rdoc.get_bool(RDOC_DESCLUNGA);
|
|||
|
if (desclunga)
|
|||
|
{
|
|||
|
const TString& dest = rdoc.get(RDOC_DESCEST);
|
|||
|
if (!dest.blank())
|
|||
|
descrizione << dest;
|
|||
|
}
|
|||
|
int nfields = s.items();
|
|||
|
for (int j = 1; j < nfields; j++)
|
|||
|
{
|
|||
|
const TString & fld = s.get(j);
|
|||
|
TForm_item & f = cf.find_field(fld);
|
|||
|
const TString & val = f.get();
|
|||
|
if (!val.blank())
|
|||
|
descrizione << ' ' << val;
|
|||
|
}
|
|||
|
cf.set(descrizione);
|
|||
|
TParagraph_string p(descrizione, cf.width());
|
|||
|
const int h = cf.height();
|
|||
|
int i;
|
|||
|
|
|||
|
for (i = 0; p.get() != NULL && i < h; i++);
|
|||
|
// cf.put_paragraph(descrizione);
|
|||
|
// Setta l'altezza effettiva del body, per evitare sprechi di righe
|
|||
|
cf.section().set_height(p.empty() ? 1 : i);
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
if (code== "_RIEPILOGOIVA")
|
|||
|
{
|
|||
|
// tabella riepilogo aliquote iva e relative imposte
|
|||
|
// sintassi: _RIEPILOGOIVA,<selettore>,<macro>,<cambio codice>
|
|||
|
// dove: <selettore> <20> uno dei seguenti:
|
|||
|
// 1 = codici IVA a regime normale
|
|||
|
// 2 = codici IVA da ventilare
|
|||
|
// 4 = codici IVA esenti
|
|||
|
// 8 = codici IVA non imponibili
|
|||
|
// 16 = codici IVA non soggetti
|
|||
|
// oppure la combinazione di uno o piu' di essi:
|
|||
|
// 12 = 4+8, 19 = 1+2+16, 29 = 1+4+8+16 ecc...
|
|||
|
// dove: <macro> <20> uno dei seguenti:
|
|||
|
// COD colonna dei codici
|
|||
|
// IMP colonna degli imponibili
|
|||
|
// IVA colonna delle imposte
|
|||
|
// ALI colonna delle aliquote
|
|||
|
// DES colonna delle descrizioni (stampata solo se il regime IVA non e' normale)
|
|||
|
// dove: <cambio codice> <20> uno dei seguenti:
|
|||
|
// 0 indica di non leggere il successivo codice IVA nella tabella riepilogativa
|
|||
|
// 1 indica di leggere il successivo codice IVA nella tabella riepilogativa
|
|||
|
|
|||
|
if (s.items() == 4)
|
|||
|
{
|
|||
|
const byte selector = byte(s.get_int()); // il primo parametro e' il selettore del tipo di codice
|
|||
|
if (selector != 0)
|
|||
|
{
|
|||
|
_doc->summary_filter(selector);
|
|||
|
|
|||
|
const TString4 what = s.get(); // cosa deve stampare ?
|
|||
|
TString80 value = _doc->summary_get(what); // Piglia il valore dalla riga selezionata sulla tabellina
|
|||
|
if ((what == "IMP" || what == "IVA") && real::is_null(value))
|
|||
|
{
|
|||
|
cf.hide();
|
|||
|
cf.set("");
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
cf.show();
|
|||
|
cf.set(value);
|
|||
|
}
|
|||
|
|
|||
|
if (s.get_int() == 1) // deve cambiare elemento ?
|
|||
|
_doc->summary_set_next();
|
|||
|
}
|
|||
|
}
|
|||
|
else
|
|||
|
error_box("Numero di parametri non corretto in _RIEPILOGOIVA");
|
|||
|
return true;
|
|||
|
} // fine _RIEPILOGOIVA
|
|||
|
|
|||
|
if (code == "_TOTIMPONIBILI")
|
|||
|
{
|
|||
|
// sintassi: _TOTIMPONIBILI,<selettore>
|
|||
|
// dove: <selettore> funge da filtro per la somma degli imponibili
|
|||
|
// se selettore vale 0 restituisce il tot. imponibili con le spese
|
|||
|
// vedi _RIEPILOGOIVA per la spiegazione dei filtri selettivi
|
|||
|
const byte sel = (byte)s.get_int();
|
|||
|
const real x = sel == 0 ? _doc->imponibile(true) : _doc->tot_imponibili(sel);
|
|||
|
cf.set(x.string());
|
|||
|
return true;
|
|||
|
|
|||
|
} // fine _TOTIMPONIBILI
|
|||
|
|
|||
|
if (code== "_SCADENZE")
|
|||
|
{
|
|||
|
// messaggio per stampare le scadenze
|
|||
|
// sintassi: _SCADENZE,<macro>,<cambio codice>
|
|||
|
// dove <macro> e' uno dei seguenti:
|
|||
|
// DATA : stampa la data di scadenza
|
|||
|
// IMPORTO : stampa l'importo in scadenza
|
|||
|
// dove <cambio codice> vale 0 o 1 se indica di rendere corrente la prossima scadenza
|
|||
|
if (s.items() == 3)
|
|||
|
{
|
|||
|
TString what(s.get());
|
|||
|
TString value(_doc->scadenze_get(what));
|
|||
|
|
|||
|
what = s.get();
|
|||
|
const bool next = what == "1";
|
|||
|
if (next) _doc->scadenze_set_next();
|
|||
|
cf.set(value);
|
|||
|
}
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
if (code == "_EDITPICTURE")
|
|||
|
{
|
|||
|
const int flds2set = s.items() -1;
|
|||
|
// TString16 val(cursor()->file(LF_DOC).get(DOC_CODVAL));
|
|||
|
// const bool valuta = val.not_empty() && val != "LIT";
|
|||
|
const int ndec = _doc->decimals();
|
|||
|
for (int i = 1; i<=flds2set; i++)
|
|||
|
{
|
|||
|
const short fld = s.get_int(i);
|
|||
|
edit_picture(cf.section().find_field(fld), ndec);
|
|||
|
}
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
if (code == "_SEPARATOR") // Riempitore
|
|||
|
{
|
|||
|
TString sep;
|
|||
|
sep.fill('-',s.get_int(1));
|
|||
|
cf.set(sep);
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
if (code == "_WEEK" || code == "_YEAR")
|
|||
|
{
|
|||
|
const TString16 which(s.get());
|
|||
|
TString16 data;
|
|||
|
if (which == RDOC_DATACONS)
|
|||
|
{
|
|||
|
const TRectype& rdoc= cursor()->curr(LF_RIGHEDOC);
|
|||
|
data = rdoc.get(which);
|
|||
|
}
|
|||
|
if (data.empty())
|
|||
|
data = _doc->get(which);
|
|||
|
TDate d(data);
|
|||
|
const char * c = s.get();
|
|||
|
bool complete = c != NULL && *c != '\0';
|
|||
|
int week;
|
|||
|
int year;
|
|||
|
d.get_week_year(week, year, complete);
|
|||
|
if (code == "_WEEK")
|
|||
|
cf.set(data.format("%d", week));
|
|||
|
else
|
|||
|
cf.set(data.format("%d", year));
|
|||
|
return true;
|
|||
|
}
|
|||
|
if (code== "_PARENTDOC")
|
|||
|
{
|
|||
|
const TRectype* rdoc = &cursor()->curr(LF_RIGHEDOC);
|
|||
|
|
|||
|
// Se il campo corrente non appartiene al body allora cerco la prima riga documento buona!
|
|||
|
if (cf.section().section_type() != 'B')
|
|||
|
{
|
|||
|
const TDocumento& doc = (const TDocumento&)cursor()->curr();
|
|||
|
const TRiga_documento* first_merc = NULL;
|
|||
|
const TRiga_documento* first_desc = NULL;
|
|||
|
for (int r = 1; r <= doc.physical_rows(); r++)
|
|||
|
{
|
|||
|
const TRiga_documento& row = doc[r];
|
|||
|
if (row.get(RDOC_DANDOC).not_empty())
|
|||
|
{
|
|||
|
if (row.is_descrizione())
|
|||
|
{
|
|||
|
if (first_desc == NULL)
|
|||
|
first_desc = &row; // Non e' una riga buona, ma nemmeno da buttare!
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
first_merc = &row;
|
|||
|
break; // Ho trovato la riga buona!
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
if (first_merc != NULL)
|
|||
|
rdoc = first_merc;
|
|||
|
else
|
|||
|
{
|
|||
|
if (first_desc != NULL)
|
|||
|
rdoc = first_desc;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
int level = s.get_int(1);
|
|||
|
for (; rdoc != NULL && level > 0; level--)
|
|||
|
rdoc = ((const TRiga_documento*)rdoc)->find_original_rdoc();
|
|||
|
|
|||
|
if (rdoc != NULL && rdoc->get(RDOC_PROVV).not_empty())
|
|||
|
{
|
|||
|
const char provv = rdoc->get_char(RDOC_PROVV);
|
|||
|
const int anno = rdoc->get_int(RDOC_ANNO);
|
|||
|
const TString4 codnum = rdoc->get(RDOC_CODNUM);
|
|||
|
const long ndoc = rdoc->get_long(RDOC_NDOC);
|
|||
|
|
|||
|
if (s.get(3) != NULL) // "FULL"
|
|||
|
{
|
|||
|
TDocumento doc(provv, anno, codnum, ndoc);
|
|||
|
output_values(doc, s.get(2), cf);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
TToken_string key;
|
|||
|
key.add(provv);
|
|||
|
key.add(anno);
|
|||
|
key.add(codnum);
|
|||
|
key.add(ndoc);
|
|||
|
|
|||
|
const TRectype& doc = cache().get(LF_DOC, key);
|
|||
|
output_values(doc, s.get(2), cf);
|
|||
|
}
|
|||
|
}
|
|||
|
return true;
|
|||
|
}
|
|||
|
if (code== "_PARENTROW")
|
|||
|
{
|
|||
|
const TRectype * rdoc = & cursor()->file(LF_RIGHEDOC).curr();
|
|||
|
int level = s.get_int(1);
|
|||
|
|
|||
|
for (; rdoc != NULL && level > 0; level--)
|
|||
|
rdoc = ((const TRiga_documento *) rdoc)->find_original_rdoc();
|
|||
|
|
|||
|
if (rdoc != NULL && rdoc->get(RDOC_PROVV).not_empty())
|
|||
|
{
|
|||
|
if (s.get(3) != NULL) // "FULL"
|
|||
|
{
|
|||
|
const char provv = rdoc->get_char(RDOC_PROVV);
|
|||
|
const int anno = rdoc->get_int(RDOC_ANNO);
|
|||
|
const TString4 codnum = rdoc->get(RDOC_CODNUM);
|
|||
|
const long ndoc = rdoc->get_long(RDOC_NDOC);
|
|||
|
TDocumento doc(provv, anno, codnum, ndoc);
|
|||
|
output_values(doc[rdoc->get_int(RDOC_NRIGA)], s.get(2), cf);
|
|||
|
}
|
|||
|
else
|
|||
|
output_values(*rdoc, s.get(2), cf);
|
|||
|
}
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
if (code == "_LISTADOC") // Messaggio per riepilogo lista documenti
|
|||
|
{
|
|||
|
const TString16 what(s.get(1));
|
|||
|
TString16 which(s.get(2));
|
|||
|
if (what == "STORE")
|
|||
|
{
|
|||
|
if (which[0] == '#')
|
|||
|
which.ltrim(1); // Toglie il #
|
|||
|
const TString4 codnum(cf.section().find_field(atoi(which)).get());
|
|||
|
const real r(cf.get());
|
|||
|
|
|||
|
real* v = (real*)_doc_totals.objptr(codnum);
|
|||
|
if (v == NULL)
|
|||
|
{
|
|||
|
v = new real(ZERO);
|
|||
|
_doc_totals.add(codnum, v);
|
|||
|
}
|
|||
|
if (_doc->is_nota_credito())
|
|||
|
*v -= r;
|
|||
|
else
|
|||
|
*v += r;
|
|||
|
} else
|
|||
|
if (what == "ADDTOT")
|
|||
|
{
|
|||
|
real r = cf.get();
|
|||
|
if (!r.is_zero())
|
|||
|
{
|
|||
|
TForm_item& tot = cf.find_field(which);
|
|||
|
real v = tot.get();
|
|||
|
if (_doc->is_nota_credito())
|
|||
|
r = -r;
|
|||
|
v += r;
|
|||
|
tot.set(v.string());
|
|||
|
}
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
TString_array k;
|
|||
|
_doc_totals.get_keys(k);
|
|||
|
const int index = atoi(which)-1;
|
|||
|
if (index < k.items())
|
|||
|
{
|
|||
|
const TString& codnum = k.row(index);
|
|||
|
if (what == "CODICE")
|
|||
|
cf.set(codnum);
|
|||
|
else
|
|||
|
if (what == "TOTALE")
|
|||
|
{
|
|||
|
real& r = (real&)_doc_totals[codnum];
|
|||
|
cf.set(r.string());
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
return TForm::validate(cf, s); // se il codice del messaggio non <20> identificato viene passato alla funzione standard
|
|||
|
}
|
|||
|
|
|||
|
//////////////////////////////////////////////////////////////////////////////////////////////
|
|||
|
// classe TStampaDoc_application customizzata dalla TApplication per l'applicazione principale
|
|||
|
//////////////////////////////////////////////////////////////////////////////////////////////
|
|||
|
|
|||
|
|
|||
|
enum behaviour
|
|||
|
{
|
|||
|
skip,
|
|||
|
go,
|
|||
|
cancel
|
|||
|
};
|
|||
|
|
|||
|
// Chiavi di ordinamento LF_DOC:
|
|||
|
// Chiave 1: ordinamento per Provvisorio + Anno + Codice numerazione + Numero documento
|
|||
|
// Chiave 3: ordinamento per Data documento + Provvisorio + Anno + Codice numerazione + Numero documento
|
|||
|
|
|||
|
#define BY_NUM_KEY 1
|
|||
|
#define BY_DATE_KEY 3
|
|||
|
|
|||
|
class TStampaDoc_application: public TSkeleton_application
|
|||
|
{
|
|||
|
TString _codnum; // codice di numerazione
|
|||
|
char _provv; // stampa documenti provvisiori o definitivi (D o P)
|
|||
|
int _anno; // anno della documentazione
|
|||
|
int _key; // chiave per scorrere i documenti (1 o 3, vedi sopra)
|
|||
|
int _ncopie; // numero di copie per ogni documento
|
|||
|
long _dalnum, _alnum; // estremi di numerazione dei documenti
|
|||
|
TDate _dadata, _adata; // estremi di data dei documenti
|
|||
|
bool _interattivo; // flag che indica se il prog. funziona in interattivo o in batch
|
|||
|
bool _is_lista; // flga che indica se e' stata selezionata la lista documenti
|
|||
|
bool _definitiva; // flag che indica se la stampa <20> definitiva o no
|
|||
|
TRelation *_firmrel; // puntatore alla relazione che gestisce i dati della ditta corrente
|
|||
|
TMask * _selection_mask;
|
|||
|
|
|||
|
TRelation* _clifo_rel;
|
|||
|
TCursor* _clifo_cur;
|
|||
|
TCursor_sheet* _clifo_sheet; // Array sheet per la selezione cli/fo
|
|||
|
TAssoc_array _clifo_sel; // Assoc array con solo i cli/fo selezionati. Facilita il filtro...
|
|||
|
|
|||
|
enum TOutput_mode {out_preview, out_print, out_mail, out_signed_mail, out_pdf, out_signed_pdf};
|
|||
|
|
|||
|
protected:
|
|||
|
virtual bool create();
|
|||
|
virtual bool destroy();
|
|||
|
virtual void main_loop();
|
|||
|
KEY select(void);
|
|||
|
virtual void on_firm_change(void);
|
|||
|
virtual behaviour on_module_change(const TString &, TString &); // funzione chiamata ad ogni cambio modulo durante la stampa
|
|||
|
void set_filter(TDocumento_form& frm);
|
|||
|
static bool codnum_handler(TMask_field& f, KEY key);
|
|||
|
static bool date2num_handler(TMask_field& f, KEY key);
|
|||
|
static bool range_handler(TMask_field& f, KEY key);
|
|||
|
static bool tipocf_handler (TMask_field& f, KEY k);
|
|||
|
static bool fr_cod_handler (TMask_field& f, KEY k);
|
|||
|
static bool to_cod_handler (TMask_field& f, KEY k);
|
|||
|
static bool select_button (TMask_field& f, KEY k);
|
|||
|
static bool reset_button (TMask_field& f, KEY k);
|
|||
|
static bool tipodoc_handler (TMask_field& f, KEY k);
|
|||
|
static bool filter_clifo(const TRelation* r);
|
|||
|
long select_cod_range(long from, long to);
|
|||
|
void reset_choices(TMask& m);
|
|||
|
void set_choice_limits(TMask& m);
|
|||
|
void build_clifo_list(const char c='C');
|
|||
|
int numerazione_definitiva(TDocumento& doc) const;
|
|||
|
TOutput_mode key2mode(KEY k) const;
|
|||
|
|
|||
|
public:
|
|||
|
void print_documento(TDocumento_form& frm);
|
|||
|
void print_selected(KEY k);
|
|||
|
TStampaDoc_application() : _key(BY_NUM_KEY) {};
|
|||
|
virtual ~TStampaDoc_application() {};
|
|||
|
};
|
|||
|
|
|||
|
inline TStampaDoc_application& app() { return (TStampaDoc_application&) main_app(); }
|
|||
|
|
|||
|
int TStampaDoc_application::numerazione_definitiva(TDocumento& doc) const
|
|||
|
{
|
|||
|
int err = NOERR;
|
|||
|
if (doc.get_char(DOC_PROVV) == 'D') // Se e' una numerazione definitiva
|
|||
|
{
|
|||
|
if (doc.stampabile()) // Controlla se non e' gia' nello stato si stampato in definitiva
|
|||
|
{
|
|||
|
const char sfs = doc.tipo().stato_finale_stampa();
|
|||
|
doc.stato(sfs); // Se e' gia' in definitiva aggiorna solo lo stato
|
|||
|
err = doc.rewrite();
|
|||
|
|
|||
|
// Invia la transazione di cambio stato se necessario
|
|||
|
if (::can_dispatch_transaction(doc))
|
|||
|
{
|
|||
|
TFilename tmpname; tmpname.temp();
|
|||
|
{ // Parentesi strategiche
|
|||
|
TConfig ini(tmpname, "Transaction");
|
|||
|
ini.set("Action", "MODIFY");
|
|||
|
ini.set("Firm", prefix().get_codditta());
|
|||
|
ini.set("Mode", "A");
|
|||
|
TString8 paradoc; paradoc.format("%d", LF_DOC);
|
|||
|
ini.set_paragraph(paradoc);
|
|||
|
ini.set(DOC_PROVV, doc.get(DOC_PROVV));
|
|||
|
ini.set(DOC_ANNO, doc.get(DOC_ANNO));
|
|||
|
ini.set(DOC_CODNUM, doc.get(DOC_CODNUM));
|
|||
|
ini.set(DOC_NDOC, doc.get(DOC_NDOC));
|
|||
|
ini.set(DOC_STATO, doc.stato());
|
|||
|
}
|
|||
|
::dispatch_transaction(doc, tmpname);
|
|||
|
::remove(tmpname);
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
else // Se e' una numerazione provvisoria
|
|||
|
{
|
|||
|
// Scrive il nuovo documento con lo stato, numero e flag di definitiva
|
|||
|
TDocumento bak_doc;
|
|||
|
bak_doc = doc; // Setta il flag di nuovo documento
|
|||
|
bak_doc.put(DOC_STATO,doc.tipo().stato_finale_stampa());
|
|||
|
bak_doc.put(DOC_PROVV,"D");
|
|||
|
bak_doc.put(DOC_NDOC,-1L);
|
|||
|
const int pr = bak_doc.physical_rows();
|
|||
|
for (int i=1;i<=pr;i++)
|
|||
|
bak_doc[i].put(DOC_PROVV,"D");
|
|||
|
err = bak_doc.write(); // Esegue automagicamente rinumerazione di testata e righe nel caso di reinsert
|
|||
|
if (err == NOERR) // Cancella il vecchio documento
|
|||
|
doc.remove();
|
|||
|
}
|
|||
|
return err;
|
|||
|
}
|
|||
|
|
|||
|
void TStampaDoc_application::print_selected(KEY k)
|
|||
|
{
|
|||
|
TRelation rel(LF_DOC);
|
|||
|
rel.add(LF_RIGHEDOC,"CODNUM==CODNUM|ANNO==ANNO|PROVV==PROVV|NDOC==NDOC");
|
|||
|
TCursor cur(&rel);
|
|||
|
cur.setkey(_key);
|
|||
|
TLocalisamfile& doc = cur.file();
|
|||
|
TRectype darec(LF_DOC),arec(LF_DOC); // Estremi filtro
|
|||
|
TString modulo_prec;
|
|||
|
const bool order_by_num = _key == BY_NUM_KEY;
|
|||
|
|
|||
|
if (!_is_lista)
|
|||
|
{
|
|||
|
doc.put(DOC_CODNUM, _codnum); // compone la chiave per il record di inizio cursore
|
|||
|
doc.put(DOC_ANNO, _anno);
|
|||
|
doc.put(DOC_PROVV, _provv);
|
|||
|
arec = doc.curr();
|
|||
|
if (order_by_num)
|
|||
|
doc.put(DOC_NDOC, _dalnum);
|
|||
|
else
|
|||
|
doc.put(DOC_DATADOC, _dadata);
|
|||
|
//doc.setkey(_key);
|
|||
|
doc.read(); // trova il record iniziale
|
|||
|
darec = doc.curr();
|
|||
|
doc.curr() = arec;
|
|||
|
if (order_by_num)
|
|||
|
{
|
|||
|
doc.put(DOC_NDOC, _alnum);
|
|||
|
if (doc.read() == _iskeynotfound) // trova il record finale
|
|||
|
doc.prev();
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
doc.put(DOC_DATADOC, _adata); // trova il record finale
|
|||
|
doc.put(DOC_NDOC, 999999L);
|
|||
|
int err = doc.read(_isgteq);
|
|||
|
if (err == NOERR)
|
|||
|
err = doc.prev();
|
|||
|
}
|
|||
|
|
|||
|
arec = doc.curr();
|
|||
|
|
|||
|
if (arec < darec)
|
|||
|
{
|
|||
|
error_box("Non vi sono documenti da stampare nell'intervallo indicato");
|
|||
|
return;
|
|||
|
}
|
|||
|
if (argc() < 6)
|
|||
|
_definitiva = _selection_mask->get(F_TIPOST) == "D";
|
|||
|
}
|
|||
|
|
|||
|
TPrinter& pr = printer();
|
|||
|
|
|||
|
if (_interattivo)
|
|||
|
{
|
|||
|
if (k == 'A')
|
|||
|
pr.set_printtype(screenvis);
|
|||
|
else
|
|||
|
pr.set_printtype(winprinter);
|
|||
|
}
|
|||
|
|
|||
|
pr.open();
|
|||
|
TProgress_monitor* pi = pr.printtype() != screenvis ? new TProgress_monitor(cur.items(), TR("Stampa documenti in corso..."),false) : NULL;
|
|||
|
if (!_is_lista)
|
|||
|
{
|
|||
|
cur.setregion(darec, arec);
|
|||
|
if (!order_by_num)
|
|||
|
{
|
|||
|
TString80 filter;
|
|||
|
filter.format("(CODNUM==\"%s\")&&(PROVV==\"%c\")", (const char*)_codnum, _provv);
|
|||
|
cur.setfilter(filter);
|
|||
|
}
|
|||
|
const long items = cur.items();
|
|||
|
behaviour whattodo = go;
|
|||
|
bool first_inst = true;
|
|||
|
real totdocumenti = ZERO;
|
|||
|
//TDocumentoEsteso *documento = new TDocumentoEsteso;
|
|||
|
//cur.file().set_curr(documento);
|
|||
|
for (long i = 0; i < items; i++)
|
|||
|
{
|
|||
|
cur = i; // Posiziona il documento
|
|||
|
if (_definitiva && !((TDocumento &) cur.curr()).stampabile())
|
|||
|
continue;
|
|||
|
|
|||
|
// Istanzia il form principale
|
|||
|
TDocumento_form* mainform = new TDocumento_form(cur.curr(), *_firmrel, _definitiva, _interattivo, false);
|
|||
|
if (!mainform->valid())
|
|||
|
break; // interrompe la stampa se il doc corrente non e' tra i tipi validi
|
|||
|
|
|||
|
const TString &modulo = mainform->get_module_code(); // legge dal form il codice del modulo di carta per la stampa
|
|||
|
if (modulo_prec.empty())
|
|||
|
modulo_prec = modulo; // se siamo al primo passaggio la variabile di modulo precedente viene riempita
|
|||
|
else
|
|||
|
first_inst = false;
|
|||
|
|
|||
|
const bool module_changed = modulo != modulo_prec;
|
|||
|
|
|||
|
if (first_inst || module_changed)
|
|||
|
if (!mainform->doc_arrange()) // Setta l'offset o posiziona manualmente
|
|||
|
break; // Se vi sono errori interrompe la stampa
|
|||
|
if (module_changed)
|
|||
|
whattodo = on_module_change(modulo, modulo_prec); // se il modulo <20> cambiato dalla stampa precedente interroga la funzione per sapere che comportamento tenere
|
|||
|
if (whattodo==cancel)
|
|||
|
break; // se non si pu<70> procedere la stampa viene interrotta
|
|||
|
if (whattodo==skip)
|
|||
|
continue; // Salta il documento corrente
|
|||
|
// altrimenti prosegue
|
|||
|
|
|||
|
// Carica il numero di copie da stampare per questo form
|
|||
|
int ncopie = _ncopie <= 0 ? mainform->ncopie() : _ncopie; // Numero di copie da stampare per questo documento
|
|||
|
if (ncopie <= 0) ncopie = 1;
|
|||
|
|
|||
|
TDocumentoEsteso& extdoc = mainform->doc();
|
|||
|
for (int n=0; n < ncopie; n++)
|
|||
|
{
|
|||
|
print_documento(*mainform);
|
|||
|
extdoc.summary_reset();
|
|||
|
extdoc.scadenze_reset();
|
|||
|
|
|||
|
const int ncopie2 = extdoc.tipo().get_int("I2");
|
|||
|
// Stampa eventuali documenti allegati
|
|||
|
TFilename formagg;
|
|||
|
// Se esiste un tipo documento da accodare
|
|||
|
if (ncopie2 > 0 && extdoc.tipo().additional_print_profile(formagg, 1))
|
|||
|
{
|
|||
|
TDocumento_form* secform = new TDocumento_form(cur.curr(), *_firmrel, _definitiva, _interattivo, true);
|
|||
|
if (secform->valid())
|
|||
|
{
|
|||
|
for (int i = 0; i < ncopie2; i++)
|
|||
|
{
|
|||
|
print_documento(*secform);
|
|||
|
extdoc.summary_reset();
|
|||
|
extdoc.scadenze_reset();
|
|||
|
}
|
|||
|
}
|
|||
|
delete secform;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// se la stampa <20> definitiva viene lanciata la procedura di rinumerazione
|
|||
|
if (_definitiva)
|
|||
|
{
|
|||
|
if (numerazione_definitiva(mainform->doc()) != NOERR)
|
|||
|
{
|
|||
|
error_box(FR("Non <20> possibile completare la procedura di numerazione definitiva dei documenti. Errore %d"), doc.status());
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// Totalizza gli importi per eventuale stampa su FAKETOTFLD
|
|||
|
const real tot = extdoc.totale_doc();
|
|||
|
if (extdoc.is_nota_credito())
|
|||
|
totdocumenti -= tot;
|
|||
|
else
|
|||
|
totdocumenti += tot;
|
|||
|
|
|||
|
if (i == items - 1 && mainform->is_faketotfld())
|
|||
|
{
|
|||
|
mainform->hide_sections();
|
|||
|
TForm_item& fk = mainform->find_field('F', last_page, FAKETOTFLD);
|
|||
|
fk.show();
|
|||
|
fk.set(totdocumenti.string());
|
|||
|
print_documento(*mainform);
|
|||
|
}
|
|||
|
|
|||
|
delete mainform;
|
|||
|
}
|
|||
|
}
|
|||
|
else // Lista documenti
|
|||
|
{
|
|||
|
TDocumento_form* mainform = new TDocumento_form(LISTADOC,*_firmrel);
|
|||
|
TCursor& cur = *mainform->cursor();
|
|||
|
|
|||
|
cur.setkey(_key);
|
|||
|
darec.put(DOC_DATADOC, _dadata);
|
|||
|
darec.put(DOC_PROVV, _provv);
|
|||
|
darec.put(DOC_ANNO, _anno);
|
|||
|
arec = darec;
|
|||
|
arec.put(DOC_DATADOC, _adata);
|
|||
|
cur.setregion(darec,arec);
|
|||
|
|
|||
|
const bool dettaglio = _selection_mask->get_bool(F_DETTAGLIO);
|
|||
|
mainform->find_field('B', odd_page, "H_RIGHE").enable(dettaglio); // Visualizza i dettagli righe se richiesto
|
|||
|
mainform->find_field('B', odd_page, "RIGHE").enable(dettaglio);
|
|||
|
|
|||
|
set_filter(*mainform);
|
|||
|
const TRecnotype items = cur.items();
|
|||
|
if (items > 0)
|
|||
|
{
|
|||
|
cur.freeze();
|
|||
|
mainform->set_doc_ext(NULL); // Setta il documento esteso DOPO aver fatto il filtro
|
|||
|
mainform->print();
|
|||
|
}
|
|||
|
delete mainform;
|
|||
|
}
|
|||
|
|
|||
|
if (pi != NULL) delete pi;
|
|||
|
printer().close();
|
|||
|
}
|
|||
|
|
|||
|
void TStampaDoc_application::print_documento(TDocumento_form& f)
|
|||
|
{
|
|||
|
const TRectype& doc = f.cursor()->curr();
|
|||
|
const bool is_vis = printer().printtype() == screenvis;
|
|||
|
if (!is_vis)
|
|||
|
{
|
|||
|
TString80 status(TR("Documento: "));
|
|||
|
status << doc.get(DOC_CODNUM);
|
|||
|
status << '\\' << doc.get(DOC_NDOC);
|
|||
|
xvtil_statbar_set(status);
|
|||
|
}
|
|||
|
f.print_documento();
|
|||
|
|
|||
|
if (!is_vis)
|
|||
|
xvtil_statbar_set(NULL);
|
|||
|
}
|
|||
|
|
|||
|
behaviour TStampaDoc_application::on_module_change(const TString &modulo, TString &modulo_prec)
|
|||
|
{
|
|||
|
if (!_interattivo)
|
|||
|
return skip; // se siamo in interattivo il documento viene saltato...
|
|||
|
else
|
|||
|
{ // ...altrimenti viene chiesto all'utente il da farsi
|
|||
|
int risp= yesnocancel_box(FR("Il modulo di carta <20> cambiato: inserire il modulo '%s' e premere 'Si' per continuare,"
|
|||
|
"'No' per saltare il documento o 'Annulla' per interrompere la stampa"), (const char*) modulo);
|
|||
|
behaviour ret;
|
|||
|
switch (risp)
|
|||
|
{
|
|||
|
case K_YES:
|
|||
|
modulo_prec= modulo; // aggiorna l'inseguitore dei moduli
|
|||
|
ret= go; // la stampa pu<70> continuare
|
|||
|
break;
|
|||
|
case K_NO:
|
|||
|
ret= skip; // il documento viene saltato
|
|||
|
break;
|
|||
|
case K_ESC:
|
|||
|
default:
|
|||
|
ret= cancel; // la stampa viene interrotta
|
|||
|
break;
|
|||
|
}
|
|||
|
return ret;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void TStampaDoc_application::set_filter(TDocumento_form& frm)
|
|||
|
{
|
|||
|
TCursor* cur = frm.cursor();
|
|||
|
TString filtro,e1,e2,sw;
|
|||
|
|
|||
|
// Compone la lista dei clienti/forntitori selezionati
|
|||
|
_clifo_sel.destroy();
|
|||
|
TString16 key;
|
|||
|
const long items = _clifo_sheet->items();
|
|||
|
for (long i = 0L; i<items; i++)
|
|||
|
if (_clifo_sheet->checked(i))
|
|||
|
{
|
|||
|
key.format("%06ld", _clifo_sheet->row(i).get_long(1)); // Formatta il codice
|
|||
|
_clifo_sel.add(key, NULL);
|
|||
|
}
|
|||
|
// NB: se _clifo_sel non contiene nulla, non viene effettuato alcun filtro su CLI/FO (non setta la funzione!!)
|
|||
|
filtro.format("TIPOCF==\"%c\"", _selection_mask->get(F_TIPOCF)[0]);
|
|||
|
|
|||
|
const int selval = _selection_mask->get_int(F_SELVAL);
|
|||
|
const TString16 val = _selection_mask->get(F_VALUTA);
|
|||
|
const int ndec = TCurrency::get_firm_dec();
|
|||
|
|
|||
|
frm.edit_picture(frm.find_field('F',last_page, 6), ndec); // pictures per totali finali
|
|||
|
frm.edit_picture(frm.find_field('F',last_page, 8), ndec);
|
|||
|
frm.edit_picture(frm.find_field('F',last_page, 10), ndec);
|
|||
|
frm.edit_picture(frm.find_field('F',last_page, 12), ndec);
|
|||
|
frm.edit_picture(frm.find_field('F',last_page, 16), ndec);
|
|||
|
frm.edit_picture(frm.find_field('F',last_page, 17), ndec);
|
|||
|
if (selval == 1) //In Lire
|
|||
|
{
|
|||
|
const TString16 firm_val(TCurrency::get_firm_val());
|
|||
|
const bool not_empty = firm_val.not_empty();
|
|||
|
|
|||
|
filtro << "&&";
|
|||
|
if (not_empty)
|
|||
|
filtro << "((CODVAL==\"" << firm_val << "\")||";
|
|||
|
filtro << "(CODVAL==\"\")";
|
|||
|
if (not_empty)
|
|||
|
filtro << ")";
|
|||
|
}
|
|||
|
else
|
|||
|
if (selval == 2) // nella valuta specificata
|
|||
|
filtro << "&&(CODVAL==\"" << val << "\")";
|
|||
|
else
|
|||
|
frm.find_field('B',odd_page,35).set("1"); // Cosi' effettua i totali generali in lire ?????
|
|||
|
|
|||
|
// Compone l'espressione filtro...
|
|||
|
// prende tutte le righe dello spreasheet che non sono totalmente vuote:
|
|||
|
// (CODNUM=="xxx"&&(STATO=="x"||STATO=="y"||STATO=="z"...)) ---> questo per una singola riga...
|
|||
|
// se vi sono piu' righe, lo si ripete aggiungendo prima un bellissimo "||"
|
|||
|
bool put_parentheses = false;
|
|||
|
TSheet_field& sf = (TSheet_field&)_selection_mask->field(F_SHEETNUMS);
|
|||
|
const int rows = sf.items();
|
|||
|
for (int j=0; j<rows; j++)
|
|||
|
{
|
|||
|
TToken_string& riga = sf.row(j);
|
|||
|
sw = riga.get(0); sw.trim();
|
|||
|
if (riga.empty_items() || sw.empty())
|
|||
|
break; // Interrompe alla prima riga vuota...
|
|||
|
|
|||
|
e1.format("((CODNUM==\"%s\")", (const char*)sw);
|
|||
|
|
|||
|
TString4 td = riga.get(1);
|
|||
|
if (td.full())
|
|||
|
e1 << "&&(TIPODOC==\"" << td.trim() <<"\")";
|
|||
|
|
|||
|
const long da_ndoc = riga.get_long(2);
|
|||
|
const long a_ndoc = riga.get_long(3);
|
|||
|
if (da_ndoc > 0 || a_ndoc > 0)
|
|||
|
e1 << "&&(BETWEEN(NDOC," << da_ndoc << "," << a_ndoc << "))";
|
|||
|
|
|||
|
e2 = "";
|
|||
|
for (int k=4; k<=7; k++) // Famme vede' li stati generali... Aho' A BURINO! Che e' la Rivoluzione Francese?
|
|||
|
{
|
|||
|
const char c = riga.get_char(k);
|
|||
|
if (c <= ' ')
|
|||
|
break; // Interrompe al primo blank...
|
|||
|
e2.format("(STATO==\"%c\")",c);
|
|||
|
// Se k vale 3 o piu' significa ke gli stati prec erano != "" no!?
|
|||
|
if (k == 4)
|
|||
|
e1 << "&&(";
|
|||
|
else
|
|||
|
e1 << "||";
|
|||
|
e1 << e2;
|
|||
|
}
|
|||
|
if (e2.not_empty())
|
|||
|
e1 << ")";
|
|||
|
e1 << ")"; // Piazza la parentesi finale
|
|||
|
|
|||
|
put_parentheses = true;
|
|||
|
if (j == 0)
|
|||
|
filtro << "&&(";
|
|||
|
else
|
|||
|
filtro << "||";
|
|||
|
|
|||
|
filtro << e1;
|
|||
|
}
|
|||
|
|
|||
|
if (put_parentheses)
|
|||
|
filtro << ")"; // Parentesi finale
|
|||
|
|
|||
|
cur->setfilter(filtro);
|
|||
|
cur->set_filterfunction(_clifo_sheet->checked() > 0 ? filter_clifo : NULL);
|
|||
|
}
|
|||
|
|
|||
|
////////////////////////////////////////////////////////////////
|
|||
|
// Handlers della maschera
|
|||
|
////////////////////////////////////////////////////////////////
|
|||
|
long TStampaDoc_application::select_cod_range(long from, long to)
|
|||
|
{
|
|||
|
TWait_cursor hourglass;
|
|||
|
|
|||
|
if (to <= 0l) to = 999999L;
|
|||
|
|
|||
|
for (int i = 0; i < _clifo_sheet->items(); i++)
|
|||
|
{
|
|||
|
TToken_string& c = _clifo_sheet->row(i);
|
|||
|
|
|||
|
const long cod = c.get_long(1);
|
|||
|
if (cod >= from && cod <= to)
|
|||
|
_clifo_sheet->check(i);
|
|||
|
else
|
|||
|
_clifo_sheet->uncheck(i);
|
|||
|
}
|
|||
|
|
|||
|
return _clifo_sheet->checked();
|
|||
|
}
|
|||
|
|
|||
|
void TStampaDoc_application::build_clifo_list(const char c)
|
|||
|
{
|
|||
|
// Semplice ed efficace
|
|||
|
TRectype rec(LF_CLIFO);
|
|||
|
rec.put(CLI_TIPOCF, c);
|
|||
|
_clifo_cur->setregion(rec, rec);
|
|||
|
}
|
|||
|
|
|||
|
bool TStampaDoc_application::tipocf_handler(TMask_field& f, KEY key)
|
|||
|
{
|
|||
|
if (f.to_check(key) && key == K_TAB)
|
|||
|
{
|
|||
|
TWait_cursor hourglass;
|
|||
|
app().reset_choices(f.mask());
|
|||
|
app().build_clifo_list(f.get()[0]);
|
|||
|
}
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
bool TStampaDoc_application::fr_cod_handler(TMask_field& f, KEY key)
|
|||
|
{
|
|||
|
TMask& m = f.mask();
|
|||
|
if (key == K_F9)
|
|||
|
{
|
|||
|
TMask& m = f.mask();
|
|||
|
TCursor_sheet* sh = app()._clifo_sheet;
|
|||
|
|
|||
|
sh->disable_check();
|
|||
|
sh->disable(DLG_USER);
|
|||
|
if (sh->run() == K_ENTER)
|
|||
|
{
|
|||
|
app().select_cod_range(sh->row(sh->selected()).get_long(1), m.get_long(F_CODTO));
|
|||
|
app().set_choice_limits(m);
|
|||
|
}
|
|||
|
sh->enable(DLG_USER);
|
|||
|
}
|
|||
|
else if (key == K_TAB && f.focusdirty())
|
|||
|
{
|
|||
|
const long l = app().select_cod_range(m.get_long(F_CODFR), m.get_long(F_CODTO));
|
|||
|
app().set_choice_limits(m);
|
|||
|
m.set(F_SELECTED, l);
|
|||
|
}
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
bool TStampaDoc_application::to_cod_handler(TMask_field& f, KEY key)
|
|||
|
{
|
|||
|
TMask& m = f.mask();
|
|||
|
if (key == K_F9)
|
|||
|
{
|
|||
|
TCursor_sheet* sh = app()._clifo_sheet;
|
|||
|
TMask& m = f.mask();
|
|||
|
|
|||
|
sh->disable_check();
|
|||
|
sh->disable(DLG_USER);
|
|||
|
if (sh->run() == K_ENTER)
|
|||
|
{
|
|||
|
app().select_cod_range(m.get_long(F_CODFR),sh->row(sh->selected()).get_long(1));
|
|||
|
app().set_choice_limits(m);
|
|||
|
}
|
|||
|
sh->enable(DLG_USER);
|
|||
|
}
|
|||
|
if (key == K_TAB && f.focusdirty())
|
|||
|
{
|
|||
|
const long l = app().select_cod_range(m.get_long(F_CODFR),
|
|||
|
m.get_long(F_CODTO));
|
|||
|
app().set_choice_limits(m);
|
|||
|
m.set(F_SELECTED, l);
|
|||
|
}
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
bool TStampaDoc_application::select_button(TMask_field& f, KEY key)
|
|||
|
{
|
|||
|
if (key == K_SPACE)
|
|||
|
{
|
|||
|
app()._clifo_sheet->enable_check();
|
|||
|
if (app()._clifo_sheet->run() == K_ENTER)
|
|||
|
app().set_choice_limits(f.mask());
|
|||
|
}
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
void TStampaDoc_application::reset_choices(TMask& m)
|
|||
|
{
|
|||
|
m.reset(F_SELECTED);
|
|||
|
m.reset(F_CODFR);
|
|||
|
m.reset(F_CODTO);
|
|||
|
_clifo_sheet->check(-1, false);
|
|||
|
}
|
|||
|
|
|||
|
bool TStampaDoc_application::reset_button(TMask_field& f, KEY key)
|
|||
|
{
|
|||
|
if (key == K_SPACE)
|
|||
|
app().reset_choices(f.mask());
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
void TStampaDoc_application::set_choice_limits(TMask& m)
|
|||
|
{
|
|||
|
TWait_cursor hourglass;
|
|||
|
long first = -1l, last = -1l;
|
|||
|
for (int i = 0; i < _clifo_sheet->items(); i++)
|
|||
|
{
|
|||
|
if (_clifo_sheet->checked(i))
|
|||
|
{
|
|||
|
const long cf = _clifo_sheet->row(i).get_long(1);
|
|||
|
if (first == -1l)
|
|||
|
first = cf;
|
|||
|
if (last < cf)
|
|||
|
last = cf;
|
|||
|
}
|
|||
|
}
|
|||
|
if (first != -1)
|
|||
|
m.set(F_CODFR, first);
|
|||
|
if (last != -1)
|
|||
|
m.set(F_CODTO, last);
|
|||
|
m.set(F_SELECTED, _clifo_sheet->checked());
|
|||
|
}
|
|||
|
|
|||
|
bool TStampaDoc_application::date2num_handler(TMask_field& f, KEY key)
|
|||
|
{
|
|||
|
TMask& m = f.mask();
|
|||
|
if (key == K_TAB && f.focusdirty())
|
|||
|
{
|
|||
|
short dlg = f.dlg();
|
|||
|
TLocalisamfile doc(LF_DOC);
|
|||
|
doc.setkey(3);
|
|||
|
TString codnum1(m.get(F_CODNUM)),codnum2;
|
|||
|
TString anno1(m.get(F_ANNO)),anno2;
|
|||
|
TString provv1(m.get(F_PROVV)),provv2;
|
|||
|
TDate data1(m.get_date(dlg)),data2;
|
|||
|
long numdoc;
|
|||
|
doc.zero();
|
|||
|
doc.put("CODNUM", codnum1);
|
|||
|
doc.put("ANNO", anno1);
|
|||
|
doc.put("PROVV", provv1);
|
|||
|
doc.put("DATADOC", data1);
|
|||
|
if (doc.read(_isgteq) == NOERR)
|
|||
|
{
|
|||
|
codnum2 = doc.get("CODNUM");
|
|||
|
anno2 = doc.get("ANNO");
|
|||
|
provv2 = doc.get("PROVV");
|
|||
|
data2 = doc.get_date("DATADOC");
|
|||
|
if (codnum1 == codnum2 && anno1 == anno2 && provv1 == provv2 && data1 == data2)
|
|||
|
{
|
|||
|
numdoc = doc.get_long("NDOC");
|
|||
|
m.set(dlg == F_DA_DATADOC ? F_DA_NDOC : F_A_NDOC, numdoc);
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
bool TStampaDoc_application::range_handler(TMask_field& f, KEY key)
|
|||
|
{
|
|||
|
bool rt = true;
|
|||
|
|
|||
|
if (key == K_TAB && f.focusdirty())
|
|||
|
{
|
|||
|
const long lim_sup = atol(f.get());
|
|||
|
const long lim_inf = f.mask().get_long(F_DA_NDOC);
|
|||
|
if (lim_sup < lim_inf)
|
|||
|
rt = f.error_box(TR("Il limite superiore deve essere maggiore del limite inferiore"));
|
|||
|
}
|
|||
|
return rt;
|
|||
|
}
|
|||
|
|
|||
|
bool TStampaDoc_application::codnum_handler(TMask_field& f, KEY key)
|
|||
|
{
|
|||
|
bool ok = true;
|
|||
|
if (key == K_TAB && !f.empty() && app().has_module(RSAUT))
|
|||
|
{
|
|||
|
const TCodice_numerazione& cn = cached_numerazione(f.get());
|
|||
|
TFilename n;
|
|||
|
for (int t = cn.ntipi_doc()-1; t >= 0; t--)
|
|||
|
{
|
|||
|
const TTipo_documento& td = cached_tipodoc(cn.tipo_doc(t));
|
|||
|
if (td.main_print_profile(n, 2)) // Se esiste il .rep allora forse sto usando il programma errato
|
|||
|
{
|
|||
|
ok = yesno_box(FR("Il tipo documento %s richiede la stampa avanzata del report %s:\nSi desidera continuare ugualmente?"),
|
|||
|
(const char*)td.codice(), (const char*)n.name());
|
|||
|
if (!ok)
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
return ok;
|
|||
|
}
|
|||
|
|
|||
|
bool TStampaDoc_application::tipodoc_handler(TMask_field& f, KEY key)
|
|||
|
{
|
|||
|
TMask& m = f.mask();
|
|||
|
switch (key)
|
|||
|
{
|
|||
|
case K_F9: //caso del bottone di selezione
|
|||
|
{
|
|||
|
TArray_sheet as(-1,-1,70,20,TR("Tipi documento"),HR("Codice|Descrizione@50")); //costruisce uno sheet di selezione dei tipi doc
|
|||
|
|
|||
|
const TCodice_numerazione& cn = cached_numerazione(m.get(101));
|
|||
|
for (int t = cn.ntipi_doc()-1; t >= 0; t--)
|
|||
|
{
|
|||
|
const TString4 tipodoc = cn.tipo_doc(t);
|
|||
|
TToken_string row; //classica token_string con codice e descrizione del tipodoc: questa viene scelta nella tabella
|
|||
|
row.add(tipodoc); //dei tipi docs %TIP
|
|||
|
row.add(cache().get("%TIP", tipodoc, "S0"));
|
|||
|
as.add(row); //..e viene aggiunta allo sheet di selezione
|
|||
|
}
|
|||
|
if (as.run() != K_ESC)
|
|||
|
{
|
|||
|
TToken_string& riga = as.row(-1); //setta sul campo a maschera il codice della riga selezionata di as
|
|||
|
f.set(riga.get(0));
|
|||
|
}
|
|||
|
}
|
|||
|
break;
|
|||
|
case K_ENTER: //caso del bottone invio; + o - come sopra; deve xo' verificare che l'eventuale codice di tipodoc digitato
|
|||
|
if (!f.empty()) //sia valido (esista nel campo S2 della tabella NUM)
|
|||
|
{
|
|||
|
const TCodice_numerazione& cn = cached_numerazione(m.get(101));
|
|||
|
for (int t = cn.ntipi_doc()-1; t >= 0; t--)
|
|||
|
{
|
|||
|
const TString& tipodoc = cn.tipo_doc(t);
|
|||
|
if (tipodoc == f.get())
|
|||
|
return true;
|
|||
|
}
|
|||
|
return f.error_box(FR("Tipo documento non valido per la numerazione %s"), (const char*)cn.codice());
|
|||
|
}
|
|||
|
break;
|
|||
|
|
|||
|
default:
|
|||
|
break;
|
|||
|
}
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
////////////////////////////////////////////////////////////////
|
|||
|
// Filtro per cli/fo sul cursore della lista documenti
|
|||
|
////////////////////////////////////////////////////////////////
|
|||
|
bool TStampaDoc_application::filter_clifo(const TRelation* r)
|
|||
|
{
|
|||
|
const long codcf = r->curr().get_long(CLI_CODCF);
|
|||
|
TString8 key; key.format("%06ld", codcf);
|
|||
|
return app()._clifo_sel.is_key(key);
|
|||
|
}
|
|||
|
|
|||
|
////////////////////////////////////////////////////////////////
|
|||
|
// Funzioni rimanenti
|
|||
|
////////////////////////////////////////////////////////////////
|
|||
|
|
|||
|
TStampaDoc_application::TOutput_mode TStampaDoc_application::key2mode(KEY k) const
|
|||
|
{
|
|||
|
TOutput_mode mode = out_print;
|
|||
|
|
|||
|
if (k >= 'a' && !has_module(FDAUT))
|
|||
|
k -= ' '; // toupper dei poveri
|
|||
|
|
|||
|
switch (k)
|
|||
|
{
|
|||
|
case 'A': mode = out_preview; break;
|
|||
|
case 'E': mode = out_mail; break;
|
|||
|
case 'e': mode = out_signed_mail; break;
|
|||
|
case 'P': mode = out_pdf; break;
|
|||
|
case 'p': mode = out_signed_pdf; break;
|
|||
|
case 'S':
|
|||
|
default : mode = out_print; break;
|
|||
|
}
|
|||
|
return mode;
|
|||
|
}
|
|||
|
|
|||
|
bool TStampaDoc_application::create()
|
|||
|
{
|
|||
|
_firmrel= new TRelation(LF_NDITTE); // istanziamento e impostazione della relazione di gestione della ditta corrente
|
|||
|
_firmrel->add(LF_ANAG, "TIPOA=TIPOA|CODANAGR=CODANAGR");
|
|||
|
_firmrel->add(LF_UNLOC, "CODDITTA=CODDITTA"); // si posiziona sulla prima unita' locale della ditta
|
|||
|
_firmrel->add(LF_COMUNI, "STATO=STATORES|COM=COMRES", 1, LF_ANAG, 100+LF_COMUNI);
|
|||
|
_firmrel->add(LF_COMUNI, "STATO=STATORES|COM=COMRF", 1, LF_ANAG, 200+LF_COMUNI);
|
|||
|
open_files(LF_TABCOM, LF_TAB, LF_OCCAS, LF_CLIFO, LF_INDSP, LF_CFVEN, LF_MOVMAG, LF_RMOVMAG, LF_CONDV, LF_ANAMAG , LF_SVRIEP, LF_AGENTI, LF_PERCPROV, LF_CAUSALI, 0);
|
|||
|
const int argc = TApplication::argc();
|
|||
|
|
|||
|
_is_lista = argc == 3 && argv(2)[0] == 'L';
|
|||
|
on_firm_change();
|
|||
|
_selection_mask = new TMask(_is_lista ? "ve1100b" : "ve1100a");
|
|||
|
|
|||
|
if (!_is_lista)
|
|||
|
{
|
|||
|
_clifo_sheet = NULL;
|
|||
|
_selection_mask->set_handler(F_CODNUM, codnum_handler);
|
|||
|
_selection_mask->set_handler(F_DA_DATADOC, date2num_handler);
|
|||
|
_selection_mask->set_handler(F_A_DATADOC, date2num_handler);
|
|||
|
_selection_mask->set_handler(F_A_NDOC, range_handler);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
_clifo_rel = new TRelation(LF_CLIFO);
|
|||
|
_clifo_cur = new TCursor(_clifo_rel);
|
|||
|
_clifo_sheet = new TCursor_sheet(_clifo_cur, " |CODCF|RAGSOC", TR("Selezione Clienti/Fornitori"),
|
|||
|
HR("@1|Codice@6R|Descrizione@50"), 0, 1);
|
|||
|
build_clifo_list(); // Costruisce l'array sheet dei clienti (si parte!!)
|
|||
|
_selection_mask->set_handler(F_TIPOCF, tipocf_handler);
|
|||
|
_selection_mask->set_handler(F_CODFR, fr_cod_handler);
|
|||
|
_selection_mask->set_handler(F_CODTO, to_cod_handler);
|
|||
|
_selection_mask->set_handler(BUT_SEL, select_button);
|
|||
|
_selection_mask->set_handler(BUT_ANN, reset_button);
|
|||
|
_selection_mask->sfield(F_SHEETNUMS).sheet_mask().set_handler(102, tipodoc_handler);
|
|||
|
|
|||
|
}
|
|||
|
TButton_tool& ap = (TButton_tool&)_selection_mask->field(DLG_PREVIEW);
|
|||
|
ap.set_exit_key('A');
|
|||
|
|
|||
|
if (argc>3)
|
|||
|
{ // lettura dei parametri iniziali dalla linea di comando
|
|||
|
_codnum= argv(2); // il primo parametro <20> il codice di numerazione
|
|||
|
_anno= atoi(argv(3)); // il secondo <20> l'anno
|
|||
|
_provv= argv(4)[0]; // il terzo <20> il flag di numerazione provvisoria
|
|||
|
_dalnum= atol(argv(5)); // il quarto <20> il numero di documento di partenza
|
|||
|
_alnum = _dalnum;
|
|||
|
_definitiva = false;
|
|||
|
_ncopie = 1;
|
|||
|
_interattivo= false;
|
|||
|
if (argc > 6)
|
|||
|
{
|
|||
|
TOutput_mode o = key2mode(argv(6)[0]); // il quinto <20> la modalita' di stampa o pdf (NON gestito qui)
|
|||
|
if (argc > 7)
|
|||
|
{
|
|||
|
_definitiva= *argv(7) == 'D'; // il sesto <20> se la stampa <20> definitiva (rinumerazione dei documenti)
|
|||
|
if (argc > 8)
|
|||
|
_ncopie = atoi(argv(8));
|
|||
|
}
|
|||
|
else
|
|||
|
_interattivo = true;
|
|||
|
}
|
|||
|
|
|||
|
print_selected(K_ENTER);
|
|||
|
return false;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
if (argc == 2 || _is_lista)
|
|||
|
{ // oppure lancio della maschera
|
|||
|
_interattivo= true;
|
|||
|
TSkeleton_application::create();
|
|||
|
}
|
|||
|
else
|
|||
|
return error_box("Usage: ve1 -0 {[codnum anno {D|P} dalnum alnum {D|P} [ncopie]] | [L]}");
|
|||
|
}
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
bool TStampaDoc_application::destroy()
|
|||
|
{
|
|||
|
delete _firmrel; // distruzione della relazione di gestione della ditta corrente
|
|||
|
delete _selection_mask;
|
|||
|
if (_clifo_sheet != NULL)
|
|||
|
{
|
|||
|
delete _clifo_sheet;
|
|||
|
delete _clifo_cur;
|
|||
|
delete _clifo_rel;
|
|||
|
}
|
|||
|
return TApplication::destroy();
|
|||
|
}
|
|||
|
|
|||
|
void TStampaDoc_application::on_firm_change()
|
|||
|
{
|
|||
|
TLocalisamfile &firmfile= _firmrel->lfile();
|
|||
|
firmfile.put(NDT_CODDITTA, get_firm());
|
|||
|
_firmrel->read();
|
|||
|
}
|
|||
|
|
|||
|
KEY TStampaDoc_application::select()
|
|||
|
{
|
|||
|
TMask& m = *_selection_mask;
|
|||
|
|
|||
|
#if 0
|
|||
|
m.reset();
|
|||
|
if (_is_lista)
|
|||
|
reset_choices(m);
|
|||
|
#endif
|
|||
|
|
|||
|
const KEY k = m.run();
|
|||
|
if (k != K_QUIT)
|
|||
|
{
|
|||
|
if (!_is_lista)
|
|||
|
{
|
|||
|
_codnum= m.get(F_CODNUM); // lettura dei dati dalla maschera
|
|||
|
_dalnum= m.get_long(F_DA_NDOC);
|
|||
|
_alnum= m.get_long(F_A_NDOC);
|
|||
|
_ncopie = m.get_int(F_NCOPIE);
|
|||
|
if (_alnum <= 0)
|
|||
|
_alnum = 9999999L;
|
|||
|
/* Per ora disabilitamo questa feature, che forse deve essere studiata meglio
|
|||
|
TString16 config; config << "NUM" << _codnum;
|
|||
|
printer().read_configuration(config);
|
|||
|
*/
|
|||
|
}
|
|||
|
|
|||
|
_anno= m.get_int(F_ANNO);
|
|||
|
_provv= m.get(F_PROVV)[0];
|
|||
|
_dadata = m.get_date(F_DA_DATADOC);
|
|||
|
if (!_dadata.ok())
|
|||
|
_dadata = TDate(1,1,_anno);
|
|||
|
_adata = m.get_date(F_A_DATADOC);
|
|||
|
if (!_adata.ok())
|
|||
|
_adata = TDate(31,12,_anno);
|
|||
|
if (_is_lista || m.get(F_DATA_O_NUM) == "D")
|
|||
|
_key = BY_DATE_KEY;
|
|||
|
else
|
|||
|
_key = BY_NUM_KEY;
|
|||
|
}
|
|||
|
return k;
|
|||
|
}
|
|||
|
|
|||
|
void TStampaDoc_application::main_loop()
|
|||
|
{
|
|||
|
KEY k = K_ENTER;
|
|||
|
while ((k = select()) != K_QUIT)
|
|||
|
print_selected(k);
|
|||
|
}
|
|||
|
|
|||
|
// Do all the work!
|
|||
|
int ve1100(int argc, char* argv[])
|
|||
|
{
|
|||
|
TStampaDoc_application a;
|
|||
|
const bool riep = argc >= 4 && argv[2][0] == 'L'; // Lista documenti
|
|||
|
a.run(argc, argv, riep ? TR("Lista documenti") : TR("Stampa documenti"));
|
|||
|
return 0;
|
|||
|
}
|