#include #include #include #include #include #include #include #include #include #include #include #include #include #include "velib.h" #include "sconti.h" #include "ve1100.h" #include "ve0100b.h" #define LISTADOC "listadoc" #define Usage "Usage: ve1 -0 {[codnum anno {D|P} dalnum alnum {D|P} [ncopie]] | [L]}" // 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; } virtual int readat(TRecnotype nrec, word lockop = _nolock); TDocisamfile(TDocumentoEsteso* doc) : TLocalisamfile(LF_DOC) { _doc = doc;} virtual ~TDocisamfile() {}; }; int TDocisamfile::readat(TRecnotype nrec, word lockop) { int err = TBaseisamfile::readat(nrec, _nolock); if (err == NOERR) if ((err=_doc->read(curr())) == 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; protected: TDocumento& doc() const { return *_doc;} public: virtual TRectype& curr() const {return (TRectype&) doc()[_row>0 ? _row : 1];} void set_row(int r) { _row = r;} TRDocisamfile(TDocumento* doc) : TLocalisamfile(LF_RIGHEDOC) { _doc = doc; _row = 1;} virtual ~TRDocisamfile() {}; }; //////////////////////////////////////////////////////////////////////////// // classe TDocumento_form customizzata dalla Form per i documenti //////////////////////////////////////////////////////////////////////////// class TDocumento_form : public TForm { static TDocumento_form* _form; TDocisamfile* _docfile; TRDocisamfile* _rdocfile; TLocalisamfile *_tab; TTable *_tip; TRelation &_firmrel; // relazione di gestione dei dati della ditta corrente TString _module; // codice del modulo di carta associato a questo al form TCond_vendita * _condv; TString_array _exclude_array_t; // array di tipi riga da escludere dalla stampa TString_array _exclude_array_a; // array di articoli da escludere dalla stampa TDocumentoEsteso * _doc; // Documento da stampare bool _valid, _cli_loaded; // flag che indica se il form e' valido | se l'oggetto cliente è già stato caricato bool _is_lista; // flag che indica se il form e' usato per la stampa della 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 // 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 // Dove e' il numero del gruppo // e' il numero di decimali per i documenti in lire // 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 void edit_picture(TForm_item & f, const int dec); void modify_pictures(); 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); public: void print_documento(); bool valid() { return _valid; } bool doc_arrange(); int ncopie() { return _doc->tipo().ncopie(); } const TString &get_module_code() { return _module; } // ritorna il codice del modulo di carta TString_array & exclude_list_t() { return _exclude_array_t; } TString_array & exclude_list_a() { return _exclude_array_a; } TDocumentoEsteso& doc() { return *_doc; } TDocumento_form(TRectype&/*TDocumentoEsteso **/ doc, TRelation& rel, const bool definitiva, const bool interattivo); TDocumento_form(const char* form, TRelation& rel); virtual ~TDocumento_form(); }; TDocumento_form* TDocumento_form::_form = NULL; TDocumento_form::TDocumento_form(TRectype&/*TDocumentoEsteso**/ doc, TRelation& rel, const bool definitiva, const bool interattivo): TForm(), _firmrel(rel), _valid(FALSE) { _form = this; //_doc = doc; _tip = new TTable("%TIP"); _tab = new TLocalisamfile(LF_TAB); TString nomeform; // TFilename profilo; TString codnum(doc.get(DOC_CODNUM)); TString numdoc(doc.get(DOC_NDOC)); // modificare ?? _tip->put("CODTAB", doc.get(DOC_TIPODOC)); // posiziona la tabella dei tipi di documento int err=_tip->read(); // legge la tabella if (err==NOERR) { // se non ci sono errori procede con la stampa nomeform= _tip->get("S5"); // legge il nome del form di stampa TFilename test(nomeform); test.ext("frm"); // profilo= _tip->get("S4"); // legge il nome del profilo di configurazione // profilo.ext("ini"); // aggiunge l'estensione al nome del file del profilo if (/* profilo.empty() || */ fexist(nomeform)) { error_box("Nome form di stampa (%s) non valido per il tipo documento %s ", (const char *) nomeform, (const char *) ((TDocumento &)doc).tipo().codice()); return; } } else { error_box("Il documento %s %s non è stato trovato nella tabella dei tipi di documento (errore %d)", (const char*) codnum,(const char*) numdoc,err); return; } _valid = TRUE; read(nomeform); _cli_loaded= FALSE; _is_lista = FALSE; _condv = new TCond_vendita(NULL); _doc = new TDocumentoEsteso(doc, _condv); // istanzia TDocumentoEsteso _condv->set_clifo(&_doc->clifor()); _docfile = new TDocisamfile(_doc); _rdocfile = new TRDocisamfile(_doc); relation()->replace(_docfile,0); relation()->replace(_rdocfile,1); if (_doc->physical_rows() > 0) relation()->update(); 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 29: p.pri_lit = t.get_int(1);p.pri_val = t.get_int(2); break; case 30: p.qta_lit = t.get_int(1);p.qta_val = t.get_int(2); break; case 31: p.imp_lit = t.get_int(1);p.imp_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), _valid(FALSE) { _is_lista = TRUE; _rdocfile=NULL; _tip = new TTable("%TIP"); _tab = new TLocalisamfile(LF_TAB); _cli_loaded= FALSE; _doc = new TDocumentoEsteso; _docfile = new TDocisamfile(_doc); relation()->replace(_docfile,0); } TDocumento_form::~TDocumento_form() { //if (_doc) delete _doc; if (_tip) delete _tip; if (_tab) delete _tab; if (_docfile) delete _docfile; if (_rdocfile) delete _rdocfile; } 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("Non e' possibile settare 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_x(),_form->offset_y()); return TRUE; } void TDocumento_form::print_documento() { // stampa tutte le righe TPrint_section& body = section('B'); TPrint_section& foot = section('F'); TPrinter& pr = printer(); const int righe = _doc->rows(); if (righe < 1) { TPrintrow r; pr.print(r); // Riga fasulla... } 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 (pr.rows_left() <= (body.height()+1)) // salto pagina pr.formfeed(); if (!print_on_body(r)) continue; const word h = body.height(); for (word j = 0; j < h; j++) pr.print(body.row(j)); } set_last_page(TRUE); // Cosi' stampera' l'ultima pagina del footer pr.formfeed(); // 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(); const word r = foot.height()-1; 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) { TString old_picture(20); old_picture = fi.picture(); TString new_picture(20); char migliaia_char = old_picture.find(',') > 0 ? ',' : '.'; 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 TString16 dec_to_add; for (int i = 0; i < dec; i++) dec_to_add << "@"; // aggiunge tanti "@" quanti sono i decimali voluti new_picture = old_picture; 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 = fi.width(); // se la picture eccede la dimensione, 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); 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 = _exclude_array_t.find(tiporiga) < 0; if (ok) { const TString & codart = riga.get(RDOC_CODART); ok = _exclude_array_a.find(codart) < 0; } 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(29);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(30);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(31);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(),','); const char * i = s.get(); if (i) { if (*i) _exclude_array_t.add(i); i = s.get(); if (i && *i) _exclude_array_a.add(i); } } } 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,{|} // dove: è un riferimento alla relazione di gestione dei dati della ditta (es. 113@->DENCOM è la denominazione del comune di residenza della ditta) // è uno delle macro seguenti: // !RAGSOC ragione sociale // !IND indirizzo (fiscale se c'è, oppure di residenza) // !NUM numero civico (fiscale se c'è, oppure di residenza) // !CAP CAP (fiscale se c'è, oppure di residenza) // !COM comune (fiscale se c'è, oppure di residenza) // !PROV provincia (fiscale se c'è, 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 è così 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); 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("REGTRIB"); const TString & vol = _firmrel[LF_UNLOC].get("VOLTRIB"); if (vol.not_empty()) valore << " Vol. " << vol; const TString & fasc = _firmrel[LF_UNLOC].get("FASCTRIB"); if (fasc.not_empty()) valore << " Fasc. " << fasc; 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; return TRUE; } } } // fine _DITTA if (code== "_CLIENTE") { // lettura dei dati del cliente // sintassi: _CLIENTE,{|} // dove: è un riferimento alla relazione di gestione dei dati del cliente // è uno delle macro seguenti: // !RAGSOC ragione sociale // !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-> accede ai campi del comune di residenza cliente // !COMN-> accede ai campi del comune di nascita del cliente TCli_for & cli_for = _doc->clifor(); TString in(s.get()); // prende la macro o il fieldref if (in[0] != '!') { cf.set(cli_for.get(in)); return TRUE; } in.ltrim(1); if (in=="INDNUM") { valore = cli_for.get(CLI_INDCF); valore << " " << 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("STATO", cli_for.get(CLI_STATONASC)); com.put("COM", cli_for.get(CLI_COMNASC)); } else { com.put("STATO", cli_for.get(CLI_STATOCF)); com.put("COM", cli_for.get(CLI_COMCF)); } if (com.read() == NOERR) cf.set(com.get(in)); return TRUE; } if (in.find("TEL") == 0) { 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") { valore = cli_for.get("PFAX"); valore << "/" << cli_for.get("FAX"); cf.set(valore); return TRUE; } if (in=="RAGSOC") { valore = cli_for.get(in); valore.strip_d_spaces(); cf.set(valore); return TRUE; } } // fine _CLIENTE if (code == "_DESCRIGA") { // Messaggio per reperire la descrizione estesa sulle righe del documento TLocalisamfile &rdoc= (cursor())->file(LF_RIGHEDOC); TString descrizione = rdoc.get("DESCR"); const bool desclunga = rdoc.get_bool("DESCLUNGA"); if (desclunga) { const TString & s = rdoc.get("DESCEST"); descrizione << s; } cf.set(descrizione); TParagraph_string p(descrizione, cf.width()); const int h = cf.height(); for (int 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(i); return TRUE; } if (code== "_RIEPILOGOIVA") { // tabella riepilogo aliquote iva e relative imposte // sintassi: _RIEPILOGOIVA,,, // dove: è 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: è 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: è 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) { byte selector = byte(atoi(s.get())); // il primo parametro e' il selettore del tipo di codice if (selector != 0) { _doc->summary_filter(selector); TString what(s.get()); // cosa deve stampare ? TString value(_doc->summary_get(what)); // Piglia il valore dalla riga selezionata sullatabellina what = s.get(); const bool next = what == "1"; // deve cambiare elemento ? if (next) _doc->summary_set_next(); cf.set(value); } } else error_box("Numero di parametri non corretto in _RIEPILOGOIVA"); return TRUE; } // fine _RIEPILOGOIVA if (code == "_TOTIMPONIBILI") { // sintassi: _TOTIMPONIBILI, // dove: 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 byte sel = atoi(s.get()); 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,, // dove e' uno dei seguenti: // DATA : stampa la data di scadenza // IMPORTO : stampa l'importo in scadenza // dove 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; } return TForm::validate(cf, s); // se il codice del messaggio non è 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 TApplication { 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 è definitiva o no TRelation *_firmrel; // puntatore alla relazione che gestisce i dati della ditta corrente TDocumento_form *_form; // puntatore al form di stampa TArray _file; protected: void open_files(int logicnum, ...); virtual bool create(); virtual bool destroy(); virtual bool menu(MENU_TAG); bool 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 virtual bool query_final_print(void); // funzione chiamata all'inizializzazione per sapere se la stampa è definitiva static bool date2num_handler(TMask_field& f, KEY key); static bool range_handler(TMask_field& f, KEY key); public: TDocumento_form& form() { return *_form; } void print_documento(); void print_selected(); TStampaDoc_application() : _key(BY_NUM_KEY) {}; virtual ~TStampaDoc_application() {}; }; void TStampaDoc_application::print_selected() { TRelation rel(LF_DOC); rel.add(LF_RIGHEDOC,"CODNUM==CODNUM|ANNO==ANNO|PROVV==PROVV|NDOC==NDOC"); TCursor cur(&rel); TLocalisamfile& doc = cur.file(); cur.setkey(_key); TRectype darec(LF_DOC),arec(LF_DOC); // Estremi filtro TString modulo_prec; const bool order_by_num = _key == BY_NUM_KEY; doc.put(DOC_CODNUM, _codnum); // compone la chiave per il record di inizio cursore doc.put(DOC_ANNO, _anno); doc.put(DOC_PROVV, _provv); doc.put(DOC_NDOC, _dalnum); if (!order_by_num) doc.put(DOC_DATADOC, _dadata); doc.setkey(_key); doc.read(); // trova il record iniziale darec = doc.curr(); doc.put(DOC_NDOC, _alnum); if (!order_by_num) doc.put(DOC_DATADOC, _adata); int err = doc.read(); // trova il record finale if (err == _iseof) doc.last(); else if (err == _iskeynotfound) doc.prev(); arec = doc.curr(); if (arec < darec) { error_box("Non vi sono documenti da stampare nell'intervallo indicato"); return; } if (!_is_lista) _definitiva= query_final_print(); // legge il flag di stampa definitiva TPrinter& pr = printer(); pr.open(); TProgind* pi = pr.printtype() != screenvis ? new TProgind(cur.items(),"Stampa documenti in corso...",FALSE,TRUE,10) : NULL; if (!_is_lista) { cur.setregion(darec, arec); const long items = cur.items(); behaviour whattodo = go; bool first_inst = TRUE; //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; _form = new TDocumento_form(cur.curr()/*documento*/, *_firmrel, _definitiva, _interattivo); // Istanzia il form if (!_form->valid()) break; // interrompe la stampa se il doc corrente non e' tra i tipi validi const TString &modulo= _form->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 (!_form->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 è cambiato dalla stampa precedente interroga la funzione per sapere che comportamento tenere if (whattodo==cancel) break; // se non si può 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 ? _form->ncopie() : _ncopie; // Numero di copie da stampare per questo documento if (ncopie == 0) ncopie = 1; for (int n=0; n < ncopie; n++) { print_documento(); _form->doc().summary_reset(); _form->doc().scadenze_reset(); } if (_definitiva && (numerazione_definitiva(_form->doc()) != NOERR)) { // se la stampa è definitiva viene lanciata la procedura di rinumerazione if (_interattivo) error_box("Non è possibile completare la procedura di numerazione definitiva dei documenti. Errore %d", doc.status()); break; } delete _form; } // Non viene fatta la delete documento perche' gia' presente nella distruzione del cursore, avendo fatto una set_curr() } else // Lista documenti { _form = new TDocumento_form(LISTADOC,*_firmrel); _form->cursor()->setregion(darec,arec); _form->print(); delete _form; } if (pi != NULL) delete pi; printer().close(); } void TStampaDoc_application::print_documento() { CHECK(_form,"Nessun form istanziato!"); TDocumento_form& f = form(); TLocalisamfile& doc = f.cursor()->file(); const bool is_vis = printer().printtype() == screenvis; if (!is_vis) { TString status("Documento: "); status << doc.get(DOC_CODNUM); status << '\\' << doc.get(DOC_NDOC); xvt_statbar_set(status); } f.print_documento(); if (!is_vis) xvt_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("Il modulo di carta è cambiato: inserisci il modulo '%s' e premi 'Sì' per continuare, 'No' per saltare il documento o 'Cancel' per interrompere la stampa", modulo); behaviour ret; switch (risp) { case K_YES: modulo_prec= modulo; // aggiorna l'inseguitore dei moduli ret= go; // la stampa può continuare break; case K_NO: ret= skip; // il documento viene saltato break; case K_ESC: ret= cancel; // la stampa viene interrotta break; } return ret; } } bool TStampaDoc_application::query_final_print() { if (_interattivo) { // se siamo in interattivo viene richiesto all'utente se la stampa è definitiva o meno if (yesno_box("E' una stampa definitiva?")) return TRUE; else return FALSE; } else return _definitiva; // altrimenti ritorna il valore letto dalla linea di comando } 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) { int rt = TRUE; TMask& m = f.mask(); if (key == K_TAB && f.focusdirty()) { const long lim_sup = atol(f.get()); const long lim_inf = m.get_long(F_DA_NDOC); if (lim_sup < lim_inf) { f.error_box("Il limite superiore deve essere maggiore del limite inferiore"); rt = FALSE; } } return rt; } void TStampaDoc_application::open_files(int logicnum, ...) { va_list marker; va_start(marker, logicnum); while (logicnum > 0) { CHECKD(_file.objptr(logicnum) == NULL, "File gia' aperto: ", logicnum); _file.add(new TLocalisamfile(logicnum), logicnum); logicnum = va_arg(marker, int); } } bool TStampaDoc_application::create() { TApplication::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, "COM=STATORES+COMRES", 1, LF_ANAG, 100+LF_COMUNI); _firmrel->add(LF_COMUNI, "COM=STATORES+COMRF", 1, LF_ANAG, 200+LF_COMUNI); open_files(LF_TABCOM, LF_TAB, LF_OCCAS, LF_CLIFO, LF_CFVEN, LF_MOVMAG, LF_RMOVMAG, LF_CONDV, 0); const int argc = TApplication::argc(); _is_lista = argc == 3 && argv(2)[0] == 'L'; on_firm_change(); if (argc>3) { // lettura dei parametri iniziali dalla linea di comando _codnum= argv(2); // il primo parametro è il codice di numerazione _anno= atoi(argv(3)); // il secondo è l'anno _provv= argv(4)[0]; // il terzo è il flag di numerazione provvisoria _dalnum= atol(argv(5)); // il quarto è il numero di documento di partenza _alnum = _dalnum; _definitiva = FALSE; _ncopie = 1; _interattivo= FALSE; if (argc > 6) { _alnum= atol(argv(6)); // il quinto è il numero di documento di fine if (argc > 7) { _definitiva= (strcmp(argv(7), "D")==0); // il sesto è se la stampa è definitiva (rinumerazione dei documenti) if (argc > 8) _ncopie = atoi(argv(8)); } else _interattivo = TRUE; } print_selected(); return FALSE; } else { if (argc == 2 || _is_lista) { // oppure lancio della maschera _interattivo= TRUE; dispatch_e_menu(BAR_ITEM(1)); } else return error_box(Usage); } return TRUE; } bool TStampaDoc_application::destroy() { delete _firmrel; // distruzione della relazione di gestione della ditta corrente return TApplication::destroy(); } void TStampaDoc_application::on_firm_change() { TLocalisamfile &firmfile= _firmrel->lfile(); firmfile.put("CODDITTA", get_firm()); _firmrel->read(); } bool TStampaDoc_application::select() { TMask m("ve1100a"); if (_is_lista) { m.hide(F_NCOPIE); m.set_caption("Lista Documenti"); } TString wdate; m.set_handler(F_DA_DATADOC, date2num_handler); m.set_handler(F_A_DATADOC, date2num_handler); m.set_handler(F_A_NDOC, range_handler); if (m.run() == K_ENTER) { _codnum= m.get(F_CODNUM); // lettura dei dati dalla maschera _anno= m.get_int(F_ANNO); _provv= m.get(F_PROVV)[0]; _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; wdate = m.get(F_DA_DATADOC); if (wdate.not_empty()) _dadata = wdate; else _dadata = botime; wdate = m.get(F_A_DATADOC); if (wdate.not_empty()) _adata = wdate; else _adata = eotime; _key = BY_NUM_KEY; if (m.get(F_DATA_O_NUM) != "N") _key = BY_DATE_KEY; return TRUE; } else return FALSE; } bool TStampaDoc_application::menu(MENU_TAG) { while (select()) print_selected(); return FALSE; } // Do all the work! int ve1100(int argc, char** argv) { TStampaDoc_application a; const bool cond = argc == 4 && argv[2][0] == 'L'; // List documenti a.run(argc, argv, cond ? "Lista documenti" : "Stampa documenti di vendita"); return (0); }