/************ TabPivot C++ Program Source Code File (.CPP) *************/ /* PROGRAM NAME: TABPIVOT */ /* ------------- */ /* Version 1.7 */ /* */ /* COPYRIGHT: */ /* ---------- */ /* (C) Copyright to the author Olivier BERTRAND 2005-2017 */ /* */ /* WHAT THIS PROGRAM DOES: */ /* ----------------------- */ /* This program are the PIVOT classes DB execution routines. */ /***********************************************************************/ /***********************************************************************/ /* Include relevant sections of the operating system header file. */ /***********************************************************************/ #include "my_global.h" #include "table.h" // MySQL table definitions #if defined(_WIN32) #if defined(__BORLANDC__) #define __MFC_COMPAT__ // To define min/max as macro #endif //#include #elif defined(UNIX) #include #include #include "osutil.h" #else #include #endif /***********************************************************************/ /* Include application header files: */ /* global.h is header containing all global declarations. */ /* plgdbsem.h is header containing the DB application declarations. */ /***********************************************************************/ #define FRM_VER 6 #include "sql_const.h" #include "field.h" #include "global.h" #include "plgdbsem.h" #include "xtable.h" #include "tabext.h" #include "tabcol.h" #include "colblk.h" #include "tabmysql.h" #include "csort.h" #include "tabutil.h" #include "tabpivot.h" #include "valblk.h" #include "ha_connect.h" /***********************************************************************/ /* Make the Pivot table column list. */ /***********************************************************************/ PQRYRES PivotColumns(PGLOBAL g, const char *tab, const char *src, const char *picol, const char *fncol, const char *skcol, const char *host, const char *db, const char *user, const char *pwd, int port) { PIVAID pvd(tab, src, picol, fncol, skcol, host, db, user, pwd, port); return pvd.MakePivotColumns(g); } // end of PivotColumns /* --------------- Implementation of the PIVAID classe --------------- */ /***********************************************************************/ /* PIVAID constructor. */ /***********************************************************************/ PIVAID::PIVAID(const char *tab, const char *src, const char *picol, const char *fncol, const char *skcol, const char *host, const char *db, const char *user, const char *pwd, int port) : CSORT(false) { Host = (char*)host; User = (char*)user; Pwd = (char*)pwd; Qryp = NULL; Database = (char*)db; Tabname = (char*)tab; Tabsrc = (char*)src; Picol = (char*)picol; Fncol = (char*)fncol; Skcol = (char*)skcol; Rblkp = NULL; Port = (port) ? port : GetDefaultPort(); } // end of PIVAID constructor /***********************************************************************/ /* Skip columns that are in the skipped column list. */ /***********************************************************************/ bool PIVAID::SkipColumn(PCOLRES crp, char *skc) { if (skc) for (char *p = skc; *p; p += (strlen(p) + 1)) if (!stricmp(crp->Name, p)) return true; return false; } // end of SkipColumn /***********************************************************************/ /* Make the Pivot table column list. */ /***********************************************************************/ PQRYRES PIVAID::MakePivotColumns(PGLOBAL g) { char *p, *query, *colname, *skc, buf[64]; int ndif, nblin, w = 0; bool b = false; PVAL valp; PQRYRES qrp; PCOLRES *pcrp, crp, fncrp = NULL; try { // Are there columns to skip? if (Skcol) { uint n = strlen(Skcol); skc = (char*)PlugSubAlloc(g, NULL, n + 2); snprintf(skc, n + 2, "%s", Skcol); skc[n + 1] = 0; // Replace ; by nulls in skc for (p = strchr(skc, ';'); p; p = strchr(p, ';')) *p++ = 0; } else skc = NULL; if (!Tabsrc && Tabname) { // Locate the query query = (char*)PlugSubAlloc(g, NULL, strlen(Tabname) + 26); snprintf(query, strlen(Tabname) + 26, "SELECT * FROM `%s` LIMIT 1", Tabname); } else if (!Tabsrc) { snprintf(g->Message, sizeof(g->Message), MSG(SRC_TABLE_UNDEF)); goto err; } else query = (char*)Tabsrc; // Open a MySQL connection for this table if (!Myc.Open(g, Host, Database, User, Pwd, Port)) { b = true; // Returned values must be in their original character set if (Myc.ExecSQL(g, "SET character_set_results=NULL", &w) == RC_FX) goto err; else Myc.FreeResult(); } else goto err; // Send the source command to MySQL if (Myc.ExecSQL(g, query, &w) == RC_FX) goto err; // We must have a storage query to get pivot column values if (!(Qryp = Myc.GetResult(g, true))) goto err; if (!Fncol) { for (crp = Qryp->Colresp; crp; crp = crp->Next) if ((!Picol || stricmp(Picol, crp->Name)) && !SkipColumn(crp, skc)) Fncol = crp->Name; if (!Fncol) { snprintf(g->Message, sizeof(g->Message), MSG(NO_DEF_FNCCOL)); goto err; } // endif Fncol } // endif Fncol if (!Picol) { // Find default Picol as the last one not equal to Fncol for (crp = Qryp->Colresp; crp; crp = crp->Next) if (stricmp(Fncol, crp->Name) && !SkipColumn(crp, skc)) Picol = crp->Name; if (!Picol) { snprintf(g->Message, sizeof(g->Message), MSG(NO_DEF_PIVOTCOL)); goto err; } // endif Picol } // endif picol // Prepare the column list for (pcrp = &Qryp->Colresp; (crp = *pcrp); ) if (SkipColumn(crp, skc)) { // Ignore this column *pcrp = crp->Next; } else if (!stricmp(Picol, crp->Name)) { if (crp->Nulls) { snprintf(g->Message, sizeof(g->Message), "Pivot column %s cannot be nullable", Picol); goto err; } // endif Nulls Rblkp = crp->Kdata; *pcrp = crp->Next; } else if (!stricmp(Fncol, crp->Name)) { fncrp = crp; *pcrp = crp->Next; } else pcrp = &crp->Next; if (!Rblkp) { snprintf(g->Message, sizeof(g->Message), MSG(NO_DEF_PIVOTCOL)); goto err; } else if (!fncrp) { snprintf(g->Message, sizeof(g->Message), MSG(NO_DEF_FNCCOL)); goto err; } // endif if (Tabsrc) { Myc.Close(); b = false; // Before calling sort, initialize all nblin = Qryp->Nblin; Index.Size = nblin * sizeof(int); Index.Sub = TRUE; // Should be small enough if (!PlgDBalloc(g, NULL, Index)) goto err; Offset.Size = (nblin + 1) * sizeof(int); Offset.Sub = TRUE; // Should be small enough if (!PlgDBalloc(g, NULL, Offset)) goto err; ndif = Qsort(g, nblin); if (ndif < 0) // error goto err; } else { // The query was limited, we must get pivot column values // Returned values must be in their original character set // if (Myc.ExecSQL(g, "SET character_set_results=NULL", &w) == RC_FX) // goto err; query = (char*)PlugSubAlloc(g, NULL, 0); sprintf(query, "SELECT DISTINCT `%s` FROM `%s`", Picol, Tabname); PlugSubAlloc(g, NULL, strlen(query) + 1); Myc.FreeResult(); // Send the source command to MySQL if (Myc.ExecSQL(g, query, &w) == RC_FX) goto err; // We must have a storage query to get pivot column values if (!(qrp = Myc.GetResult(g, true))) goto err; Myc.Close(); b = false; // Get the column list crp = qrp->Colresp; Rblkp = crp->Kdata; ndif = qrp->Nblin; } // endif Tabsrc // Allocate the Value used to retieve column names if (!(valp = AllocateValue(g, Rblkp->GetType(), Rblkp->GetVlen(), Rblkp->GetPrec()))) goto err; // Now make the functional columns for (int i = 0; i < ndif; i++) { if (i) { crp = (PCOLRES)PlugSubAlloc(g, NULL, sizeof(COLRES)); memcpy(crp, fncrp, sizeof(COLRES)); } else crp = fncrp; // Get the value that will be the generated column name if (Tabsrc) valp->SetValue_pvblk(Rblkp, Pex[Pof[i]]); else valp->SetValue_pvblk(Rblkp, i); colname = valp->GetCharString(buf); crp->Name = PlugDup(g, colname); crp->Flag = 1; // Add this column *pcrp = crp; crp->Next = NULL; pcrp = &crp->Next; } // endfor i // We added ndif columns and removed 2 (picol and fncol) Qryp->Nbcol += (ndif - 2); return Qryp; } catch (int n) { if (trace(1)) htrc("Exception %d: %s\n", n, g->Message); } catch (const char *msg) { snprintf(g->Message, sizeof(g->Message), "%s", msg); } // end catch err: if (b) Myc.Close(); return NULL; } // end of MakePivotColumns /***********************************************************************/ /* PIVAID: Compare routine for sorting pivot column values. */ /***********************************************************************/ int PIVAID::Qcompare(int *i1, int *i2) { // TODO: the actual comparison between pivot column result values. return Rblkp->CompVal(*i1, *i2); } // end of Qcompare /* --------------- Implementation of the PIVOT classes --------------- */ /***********************************************************************/ /* PIVOTDEF constructor. */ /***********************************************************************/ PIVOTDEF::PIVOTDEF(void) { Host = User = Pwd = DB = NULL; Tabname = Tabsrc = Picol = Fncol = Function = NULL; GBdone = Accept = false; Port = 0; } // end of PIVOTDEF constructor /***********************************************************************/ /* DefineAM: define specific AM block values from PIVOT table. */ /***********************************************************************/ bool PIVOTDEF::DefineAM(PGLOBAL g, LPCSTR am, int poff) { char *p1, *p2; PHC hc __attribute__((unused))= ((MYCAT*)Cat)->GetHandler(); if (PRXDEF::DefineAM(g, am, poff)) return TRUE; Tabname = (char*)Tablep->GetName(); DB = (char*)Tablep->GetSchema(); Tabsrc = (char*)Tablep->GetSrc(); Host = GetStringCatInfo(g, "Host", "localhost"); User = GetStringCatInfo(g, "User", "*"); Pwd = GetStringCatInfo(g, "Password", NULL); Picol = GetStringCatInfo(g, "PivotCol", NULL); Fncol = GetStringCatInfo(g, "FncCol", NULL); // If fncol is like avg(colname), separate Fncol and Function if (Fncol && (p1 = strchr(Fncol, '(')) && (p2 = strchr(p1, ')')) && (*Fncol != '"') && (!*(p2+1))) { *p1++ = '\0'; *p2 = '\0'; Function = Fncol; Fncol = p1; } else Function = GetStringCatInfo(g, "Function", "SUM"); GBdone = GetBoolCatInfo("Groupby", false); Accept = GetBoolCatInfo("Accept", false); Port = GetIntCatInfo("Port", 3306); Desc = (Tabsrc) ? Tabsrc : Tabname; return FALSE; } // end of DefineAM /***********************************************************************/ /* GetTable: makes a new TDB of the proper type. */ /***********************************************************************/ PTDB PIVOTDEF::GetTable(PGLOBAL g, MODE) { return new(g) TDBPIVOT(this); } // end of GetTable /* ------------------------------------------------------------------- */ /***********************************************************************/ /* Implementation of the TDBPIVOT class. */ /***********************************************************************/ TDBPIVOT::TDBPIVOT(PPIVOTDEF tdp) : TDBPRX(tdp) { Host = tdp->Host; Database = tdp->DB; User = tdp->User; Pwd = tdp->Pwd; Port = tdp->Port; Tabname = tdp->Tabname; // Name of source table Tabsrc = tdp->Tabsrc; // SQL description of source table Picol = tdp->Picol; // Pivot column name Fncol = tdp->Fncol; // Function column name Function = tdp->Function; // Aggregate function name Xcolp = NULL; // To the FNCCOL column //Xresp = NULL; // To the pivot result column //Rblkp = NULL; // The value block of the pivot column Fcolp = NULL; // To the function column Dcolp = NULL; // To the dump column GBdone = tdp->GBdone; Accept = tdp->Accept; Mult = -1; // Estimated table size N = 0; // The current table index M = 0; // The occurence rank FileStatus = 0; // Logical End-of-File RowFlag = 0; // 0: Ok, 1: Same, 2: Skip } // end of TDBPIVOT constructor /***********************************************************************/ /* Allocate source column description block. */ /***********************************************************************/ PCOL TDBPIVOT::MakeCol(PGLOBAL g, PCOLDEF cdp, PCOL cprec, int n) { PCOL colp; if (cdp->GetOffset()) { colp = new(g) FNCCOL(cdp, this, cprec, n); if (cdp->GetOffset() > 1) Dcolp = colp; } else colp = new(g) SRCCOL(cdp, this, cprec, n); return colp; } // end of MakeCol /***********************************************************************/ /* Find default fonction and pivot columns. */ /***********************************************************************/ bool TDBPIVOT::FindDefaultColumns(PGLOBAL g) { PCOLDEF cdp; PTABDEF defp = Tdbp->GetDef(); if (!Fncol) { for (cdp = defp->GetCols(); cdp; cdp = cdp->GetNext()) if (!Picol || stricmp(Picol, cdp->GetName())) Fncol = cdp->GetName(); if (!Fncol) { snprintf(g->Message, sizeof(g->Message), MSG(NO_DEF_FNCCOL)); return true; } // endif Fncol } // endif Fncol if (!Picol) { // Find default Picol as the last one not equal to Fncol for (cdp = defp->GetCols(); cdp; cdp = cdp->GetNext()) if (stricmp(Fncol, cdp->GetName())) Picol = cdp->GetName(); if (!Picol) { snprintf(g->Message, sizeof(g->Message), MSG(NO_DEF_PIVOTCOL)); return true; } // endif Picol } // endif Picol return false; } // end of FindDefaultColumns /***********************************************************************/ /* Prepare the source table Query. */ /***********************************************************************/ bool TDBPIVOT::GetSourceTable(PGLOBAL g) { if (Tdbp) return false; // Already done if (!Tabsrc && Tabname) { // Get the table description block of this table if (!(Tdbp = GetSubTable(g, ((PPIVOTDEF)To_Def)->Tablep, true))) return true; if (!GBdone) { char *colist; PCOLDEF cdp; if (FindDefaultColumns(g)) return true; // Locate the suballocated colist (size is not known yet) *(colist = (char*)PlugSubAlloc(g, NULL, 0)) = 0; // Make the column list for (cdp = To_Def->GetCols(); cdp; cdp = cdp->GetNext()) if (!cdp->GetOffset()) strcat(strcat(colist, cdp->GetName()), ", "); // Add the Pivot column at the end of the list strcat(colist, Picol); // Now we know how much was suballocated PlugSubAlloc(g, NULL, strlen(colist) + 1); // Locate the source string (size is not known yet) Tabsrc = (char*)PlugSubAlloc(g, NULL, 0); // Start making the definition strcat(strcat(strcpy(Tabsrc, "SELECT "), colist), ", "); // Make it suitable for Pivot by doing the group by strcat(strcat(Tabsrc, Function), "("); strcat(strcat(strcat(Tabsrc, Fncol), ") "), Fncol); strcat(strcat(Tabsrc, " FROM "), Tabname); strcat(strcat(Tabsrc, " GROUP BY "), colist); if (Tdbp->IsView()) // Until MariaDB bug is fixed strcat(strcat(Tabsrc, " ORDER BY "), colist); // Now we know how much was suballocated PlugSubAlloc(g, NULL, strlen(Tabsrc) + 1); } // endif !GBdone } else if (!Tabsrc) { snprintf(g->Message, sizeof(g->Message), MSG(SRC_TABLE_UNDEF)); return true; } // endif if (Tabsrc) { // Get the new table description block of this source table PTABLE tablep = new(g) XTAB("whatever", Tabsrc); tablep->SetSchema(Database); if (!(Tdbp = GetSubTable(g, tablep, true))) return true; } // endif Tabsrc return false; } // end of GetSourceTable /***********************************************************************/ /* Make the required pivot columns. */ /***********************************************************************/ bool TDBPIVOT::MakePivotColumns(PGLOBAL g) { if (!Tdbp->IsView()) { // This was not done yet if GBdone is true if (FindDefaultColumns(g)) return true; // Now it is time to allocate the pivot and function columns if (!(Fcolp = Tdbp->ColDB(g, Fncol, 0))) { // Function column not found in table snprintf(g->Message, sizeof(g->Message), MSG(COL_ISNOT_TABLE), Fncol, Tabname); return true; } else if (Fcolp->InitValue(g)) return true; if (!(Xcolp = Tdbp->ColDB(g, Picol, 0))) { // Pivot column not found in table snprintf(g->Message, sizeof(g->Message), MSG(COL_ISNOT_TABLE), Picol, Tabname); return true; } else if (Xcolp->InitValue(g)) return true; // Check and initialize the subtable columns for (PCOL cp = Columns; cp; cp = cp->GetNext()) if (cp->GetAmType() == TYPE_AM_SRC) { if (((PSRCCOL)cp)->Init(g, NULL)) return TRUE; } else if (cp->GetAmType() == TYPE_AM_FNC) if (((PFNCCOL)cp)->InitColumn(g)) return TRUE; } // endif isview return false; } // end of MakePivotColumns /***********************************************************************/ /* Make the required pivot columns for an object view. */ /***********************************************************************/ bool TDBPIVOT::MakeViewColumns(PGLOBAL g) { if (Tdbp->IsView()) { // Tdbp is a view ColDB cannot be used PCOL colp, cp; PTDBMY tdbp; if (Tdbp->GetAmType() != TYPE_AM_MYSQL) { snprintf(g->Message, sizeof(g->Message),"View is not MySQL"); return true; } else tdbp = (PTDBMY)Tdbp; if (!Fncol && !(Fncol = tdbp->FindFieldColumn(Picol))) { snprintf(g->Message, sizeof(g->Message), MSG(NO_DEF_FNCCOL)); return true; } // endif Fncol if (!Picol && !(Picol = tdbp->FindFieldColumn(Fncol))) { snprintf(g->Message, sizeof(g->Message), MSG(NO_DEF_PIVOTCOL)); return true; } // endif Picol // Now it is time to allocate the pivot and function columns if (!(Fcolp = tdbp->MakeFieldColumn(g, Fncol))) return true; if (!(Xcolp = tdbp->MakeFieldColumn(g, Picol))) return true; // Check and initialize the subtable columns for (cp = Columns; cp; cp = cp->GetNext()) if (cp->GetAmType() == TYPE_AM_SRC) { if ((colp = tdbp->MakeFieldColumn(g, cp->GetName()))) { ((PSRCCOL)cp)->Colp = colp; ((PSRCCOL)cp)->To_Val = colp->GetValue(); cp->AddStatus(BUF_READ); // All is done here } else return true; } else if (cp->GetAmType() == TYPE_AM_FNC) if (((PFNCCOL)cp)->InitColumn(g)) return TRUE; } // endif isview return false; } // end of MakeViewColumns /***********************************************************************/ /* PIVOT GetMaxSize: returns the maximum number of rows in the table. */ /***********************************************************************/ int TDBPIVOT::GetMaxSize(PGLOBAL g __attribute__((unused))) { #if 0 if (MaxSize < 0) MaxSize = MakePivotColumns(g); return MaxSize; #endif // 0 return 10; } // end of GetMaxSize /***********************************************************************/ /* In this sample, ROWID will be the (virtual) row number, */ /* while ROWNUM will be the occurence rank in the multiple column. */ /***********************************************************************/ int TDBPIVOT::RowNumber(PGLOBAL, bool b) { return (b) ? M : N; } // end of RowNumber /***********************************************************************/ /* PIVOT Access Method opening routine. */ /***********************************************************************/ bool TDBPIVOT::OpenDB(PGLOBAL g) { if (Use == USE_OPEN) { /*******************************************************************/ /* Table already open, just replace it at its beginning. */ /*******************************************************************/ N = M = 0; RowFlag = 0; FileStatus = 0; return FALSE; } // endif use if (Mode != MODE_READ) { /*******************************************************************/ /* Currently PIVOT tables cannot be modified. */ /*******************************************************************/ snprintf(g->Message, sizeof(g->Message), MSG(TABLE_READ_ONLY), "PIVOT"); return TRUE; } // endif Mode if (To_Key_Col || To_Kindex) { /*******************************************************************/ /* Direct access of PIVOT tables is not implemented yet. */ /*******************************************************************/ snprintf(g->Message, sizeof(g->Message), MSG(NO_PIV_DIR_ACC)); return TRUE; } // endif To_Key_Col /*********************************************************************/ /* Do it here if not done yet (should not be the case). */ /*********************************************************************/ if (GetSourceTable(g)) return TRUE; // For tables, columns must be allocated before opening if (MakePivotColumns(g)) return TRUE; /*********************************************************************/ /* Physically open the object table. */ /*********************************************************************/ if (Tdbp->OpenDB(g)) return TRUE; Use = USE_OPEN; // Do it now in case we are recursively called /*********************************************************************/ /* Make all required pivot columns for object views. */ /*********************************************************************/ return MakeViewColumns(g); } // end of OpenDB /***********************************************************************/ /* Data Base read routine for PIVOT access method. */ /***********************************************************************/ int TDBPIVOT::ReadDB(PGLOBAL g) { int rc = RC_OK; bool newrow = FALSE; PCOL colp; if (FileStatus == 2) return RC_EF; if (FileStatus) for (colp = Columns; colp; colp = colp->GetNext()) if (colp->GetAmType() == TYPE_AM_SRC) ((PSRCCOL)colp)->SetColumn(); // New row, reset all function column values for (colp = Columns; colp; colp = colp->GetNext()) if (colp->GetAmType() == TYPE_AM_FNC) colp->GetValue()->Reset(); /*********************************************************************/ /* Now start the multi reading process. */ /*********************************************************************/ do { if (RowFlag != 1) { if ((rc = Tdbp->ReadDB(g)) != RC_OK) { if (FileStatus && rc == RC_EF) { // A prepared row remains to be sent FileStatus = 2; rc = RC_OK; } // endif FileStatus break; } // endif rc for (colp = Tdbp->GetColumns(); colp; colp = colp->GetNext()) colp->ReadColumn(g); for (colp = Columns; colp; colp = colp->GetNext()) { if (colp->GetAmType() == TYPE_AM_SRC) { if (FileStatus) { if (((PSRCCOL)colp)->CompareLast()) { newrow = (RowFlag) ? TRUE : FALSE; break; } // endif CompareLast } else ((PSRCCOL)colp)->SetColumn(); } } FileStatus = 1; } // endif RowFlag if (newrow) { RowFlag = 1; break; } else RowFlag = 2; // Look for the column having this header for (colp = Columns; colp; colp = colp->GetNext()) if (colp->GetAmType() == TYPE_AM_FNC) { if (((PFNCCOL)colp)->CompareColumn()) break; } // endif AmType if (!colp && !(colp = Dcolp)) { if (!Accept) { snprintf(g->Message, sizeof(g->Message), MSG(NO_MATCH_COL)); return RC_FX; } else continue; } // endif colp // Set the value of the matching column from the fonction value colp->GetValue()->SetValue_pval(Fcolp->GetValue()); } while (RowFlag == 2); N++; return rc; } // end of ReadDB /***********************************************************************/ /* WriteDB: Data Base write routine for PIVOT access methods. */ /***********************************************************************/ int TDBPIVOT::WriteDB(PGLOBAL g) { snprintf(g->Message, sizeof(g->Message), MSG(TABLE_READ_ONLY), "PIVOT"); return RC_FX; } // end of WriteDB /***********************************************************************/ /* Data Base delete line routine for PIVOT access methods. */ /***********************************************************************/ int TDBPIVOT::DeleteDB(PGLOBAL g, int) { snprintf(g->Message, sizeof(g->Message), MSG(NO_TABLE_DEL), "PIVOT"); return RC_FX; } // end of DeleteDB /***********************************************************************/ /* Data Base close routine for PIVOT access method. */ /***********************************************************************/ void TDBPIVOT::CloseDB(PGLOBAL g) { if (Tdbp) Tdbp->CloseDB(g); } // end of CloseDB // ------------------------ FNCCOL functions ---------------------------- /***********************************************************************/ /* FNCCOL public constructor. */ /***********************************************************************/ FNCCOL::FNCCOL(PCOLDEF cdp, PTDB tdbp, PCOL cprec, int i) : COLBLK(cdp, tdbp, i) { if (cprec) { Next = cprec->GetNext(); cprec->SetNext(this); } else { Next = tdbp->GetColumns(); tdbp->SetColumns(this); } // endif cprec Value = NULL; // We'll get a new one later Hval = NULL; // The unconverted header value Xcolp = NULL; } // end of FNCCOL constructor /***********************************************************************/ /* FNCCOL initialization function. */ /***********************************************************************/ bool FNCCOL::InitColumn(PGLOBAL g) { // Must have its own value block if (InitValue(g)) return TRUE; // Make a value from the column name Hval = AllocateValue(g, Name, TYPE_STRING); Hval->SetPrec(1); // Case insensitive Xcolp = ((PTDBPIVOT)To_Tdb)->Xcolp; AddStatus(BUF_READ); // All is done here return FALSE; } // end of InitColumn /***********************************************************************/ /* CompareColumn: Compare column value with source column value. */ /***********************************************************************/ bool FNCCOL::CompareColumn(void) { // Compare the unconverted values return Hval->IsEqual(Xcolp->GetValue(), false); } // end of CompareColumn // ------------------------ SRCCOL functions ---------------------------- /***********************************************************************/ /* SRCCOL public constructor. */ /***********************************************************************/ SRCCOL::SRCCOL(PCOLDEF cdp, PTDB tdbp, PCOL cprec, int n) : PRXCOL(cdp, tdbp, cprec, n) { } // end of SRCCOL constructor /***********************************************************************/ /* Initialize the column as pointing to the source column. */ /***********************************************************************/ bool SRCCOL::Init(PGLOBAL g, PTDB tp) { if (PRXCOL::Init(g, tp)) return true; AddStatus(BUF_READ); // All is done here return false; } // end of SRCCOL constructor /***********************************************************************/ /* SetColumn: have the column value set from the source column. */ /***********************************************************************/ void SRCCOL::SetColumn(void) { Value->SetValue_pval(To_Val); } // end of SetColumn /***********************************************************************/ /* SetColumn: Compare column value with source column value. */ /***********************************************************************/ bool SRCCOL::CompareLast(void) { // Compare the unconverted values return !Value->IsEqual(To_Val, true); } // end of CompareColumn /* --------------------- End of TabPivot/TabQrs ---------------------- */