SpreadSheet.cpp 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290
  1. // Class to read and write to Excel and text delimited spreadsheet
  2. //
  3. // Created by Yap Chun Wei
  4. // December 2001
  5. //
  6. // Version 1.1
  7. // Updates: Fix bug in ReadRow() which prevent reading of single column spreadsheet
  8. // Modified by jingzhou xu
  9. #include "stdafx.h"
  10. #include "SpreadSheet.h"
  11. // Open spreadsheet for reading and writing
  12. CSpreadSheet::CSpreadSheet(CString File, CString SheetOrSeparator, bool Backup) :
  13. m_Database(NULL), m_rSheet(NULL), m_sFile(File),
  14. m_dTotalRows(0), m_dTotalColumns(0), m_dCurrentRow(1),
  15. m_bAppend(false), m_bBackup(Backup), m_bTransaction(false)
  16. {
  17. // Detect whether file is an Excel spreadsheet or a text delimited file
  18. m_stempString = m_sFile.Right(4);
  19. m_stempString.MakeLower();
  20. if (m_stempString == ".xls") // File is an Excel spreadsheet
  21. {
  22. m_bExcel = true;
  23. m_sSheetName = SheetOrSeparator;
  24. m_sSeparator = ",;.?";
  25. }
  26. else // File is a text delimited file
  27. {
  28. m_bExcel = false;
  29. m_sSeparator = SheetOrSeparator;
  30. }
  31. if (m_bExcel) // If file is an Excel spreadsheet
  32. {
  33. m_Database = new CDatabase;
  34. GetExcelDriver();
  35. m_sDsn.Format("DRIVER={%s};DSN='';FIRSTROWHASNAMES=1;READONLY=FALSE;CREATE_DB=\"%s\";DBQ=%s", m_sExcelDriver, m_sFile, m_sFile);
  36. if (Open())
  37. {
  38. if (m_bBackup)
  39. {
  40. if ((m_bBackup) && (m_bAppend))
  41. {
  42. CString tempSheetName = m_sSheetName;
  43. m_sSheetName = "CSpreadSheetBackup";
  44. m_bAppend = false;
  45. if (!Commit())
  46. {
  47. m_bBackup = false;
  48. }
  49. m_bAppend = true;
  50. m_sSheetName = tempSheetName;
  51. m_dCurrentRow = 1;
  52. }
  53. }
  54. }
  55. }
  56. else // if file is a text delimited file
  57. {
  58. if (Open())
  59. {
  60. if ((m_bBackup) && (m_bAppend))
  61. {
  62. m_stempString = m_sFile;
  63. m_stempSql.Format("%s.bak", m_sFile);
  64. m_sFile = m_stempSql;
  65. if (!Commit())
  66. {
  67. m_bBackup = false;
  68. }
  69. m_sFile = m_stempString;
  70. }
  71. }
  72. }
  73. }
  74. // Perform some cleanup functions
  75. CSpreadSheet::~CSpreadSheet()
  76. {
  77. if (m_Database != NULL)
  78. {
  79. m_Database->Close();
  80. delete m_Database;
  81. }
  82. }
  83. // Add header row to spreadsheet
  84. bool CSpreadSheet::AddHeaders(CStringArray &FieldNames, bool replace)
  85. {
  86. if (m_bAppend) // Append to old Sheet
  87. {
  88. if (replace) // Replacing header row rather than adding new columns
  89. {
  90. if (!AddRow(FieldNames, 1, true))
  91. {
  92. return false;
  93. }
  94. else
  95. {
  96. return true;
  97. }
  98. }
  99. if (ReadRow(m_atempArray, 1)) // Add new columns
  100. {
  101. if (m_bExcel)
  102. {
  103. // Check for duplicate header row field
  104. for (int i = 0; i < FieldNames.GetSize(); i++)
  105. {
  106. for (int j = 0; j < m_atempArray.GetSize(); j++)
  107. {
  108. if (FieldNames.GetAt(i) == m_atempArray.GetAt(j))
  109. {
  110. m_sLastError.Format("Duplicate header row field:%s\n", FieldNames.GetAt(i));
  111. return false;
  112. }
  113. }
  114. }
  115. }
  116. m_atempArray.Append(FieldNames);
  117. if (!AddRow(m_atempArray, 1, true))
  118. {
  119. m_sLastError = "Problems with adding headers\n";
  120. return false;
  121. }
  122. // Update largest number of columns if necessary
  123. if (m_atempArray.GetSize() > m_dTotalColumns)
  124. {
  125. m_dTotalColumns = m_atempArray.GetSize();
  126. }
  127. return true;
  128. }
  129. return false;
  130. }
  131. else // New Sheet
  132. {
  133. m_dTotalColumns = FieldNames.GetSize();
  134. if (!AddRow(FieldNames, 1, true))
  135. {
  136. return false;
  137. }
  138. else
  139. {
  140. m_dTotalRows = 1;
  141. return true;
  142. }
  143. }
  144. }
  145. // Clear text delimited file content
  146. bool CSpreadSheet::DeleteSheet()
  147. {
  148. if (m_bExcel)
  149. {
  150. if (DeleteSheet(m_sSheetName))
  151. {
  152. return true;
  153. }
  154. else
  155. {
  156. m_sLastError = "Error deleting sheet\n";
  157. return false;
  158. }
  159. }
  160. else
  161. {
  162. m_aRows.RemoveAll();
  163. m_aFieldNames.RemoveAll();
  164. m_dTotalColumns = 0;
  165. m_dTotalRows = 0;
  166. if (!m_bTransaction)
  167. {
  168. Commit();
  169. }
  170. m_bAppend = false; // Set flag to new sheet
  171. return true;
  172. }
  173. }
  174. // Clear entire Excel spreadsheet content. The sheet itself is not deleted
  175. bool CSpreadSheet::DeleteSheet(CString SheetName)
  176. {
  177. if (m_bExcel) // If file is an Excel spreadsheet
  178. {
  179. // Delete sheet
  180. m_Database->OpenEx(m_sDsn, CDatabase::noOdbcDialog);
  181. SheetName = "[" + SheetName + "$A1:IV65536]";
  182. m_stempSql.Format ("DROP TABLE %s", SheetName);
  183. try
  184. {
  185. m_Database->ExecuteSQL(m_stempSql);
  186. m_Database->Close();
  187. m_aRows.RemoveAll();
  188. m_aFieldNames.RemoveAll();
  189. m_dTotalColumns = 0;
  190. m_dTotalRows = 0;
  191. }
  192. catch(CDBException *e)
  193. {
  194. m_sLastError = e->m_strError;
  195. m_Database->Close();
  196. return false;
  197. }
  198. return true;
  199. }
  200. else // if file is a text delimited file
  201. {
  202. return DeleteSheet();
  203. }
  204. }
  205. // Insert or replace a row into spreadsheet.
  206. // Default is add new row.
  207. bool CSpreadSheet::AddRow(CStringArray &RowValues, long row, bool replace)
  208. {
  209. long tempRow;
  210. if (row == 1)
  211. {
  212. if (m_bExcel)
  213. {
  214. // Check for duplicate header row field for Excel spreadsheet
  215. for (int i = 0; i < RowValues.GetSize(); i++)
  216. {
  217. for (int j = 0; j < RowValues.GetSize(); j++)
  218. {
  219. if ((i != j) && (RowValues.GetAt(i) == RowValues.GetAt(j)))
  220. {
  221. m_sLastError.Format("Duplicate header row field:%s\n", RowValues.GetAt(i));
  222. return false;
  223. }
  224. }
  225. }
  226. // Check for reduced header row columns
  227. if (RowValues.GetSize() < m_dTotalColumns)
  228. {
  229. m_sLastError = "Number of columns in new header row cannot be less than the number of columns in previous header row";
  230. return false;
  231. }
  232. m_dTotalColumns = RowValues.GetSize();
  233. }
  234. // Update header row
  235. m_aFieldNames.RemoveAll();
  236. m_aFieldNames.Copy(RowValues);
  237. }
  238. else
  239. {
  240. if (m_bExcel)
  241. {
  242. if (m_dTotalColumns == 0)
  243. {
  244. m_sLastError = "No header row. Add header row first\n";
  245. return false;
  246. }
  247. }
  248. }
  249. if (m_bExcel) // For Excel spreadsheet
  250. {
  251. if (RowValues.GetSize() > m_aFieldNames.GetSize())
  252. {
  253. m_sLastError = "Number of columns to be added cannot be greater than the number of fields\n";
  254. return false;
  255. }
  256. }
  257. else // For text delimited spreadsheet
  258. {
  259. // Update largest number of columns if necessary
  260. if (RowValues.GetSize() > m_dTotalColumns)
  261. {
  262. m_dTotalColumns = RowValues.GetSize();
  263. }
  264. }
  265. // Convert row values
  266. m_stempString.Empty();
  267. for (int i = 0; i < RowValues.GetSize(); i++)
  268. {
  269. if (i != RowValues.GetSize()-1) // Not last column
  270. {
  271. m_stempSql.Format("\"%s\"%s", RowValues.GetAt(i), m_sSeparator);
  272. m_stempString += m_stempSql;
  273. }
  274. else // Last column
  275. {
  276. m_stempSql.Format("\"%s\"", RowValues.GetAt(i));
  277. m_stempString += m_stempSql;
  278. }
  279. }
  280. if (row)
  281. {
  282. if (row <= m_dTotalRows) // Not adding new rows
  283. {
  284. if (replace) // Replacing row
  285. {
  286. m_aRows.SetAt(row-1, m_stempString);
  287. }
  288. else // Inserting row
  289. {
  290. m_aRows.InsertAt(row-1, m_stempString);
  291. m_dTotalRows++;
  292. }
  293. if (!m_bTransaction)
  294. {
  295. Commit();
  296. }
  297. return true;
  298. }
  299. else // Adding new rows
  300. {
  301. // Insert null rows until specified row
  302. m_dCurrentRow = m_dTotalRows;
  303. m_stempSql.Empty();
  304. CString nullString;
  305. for (int i = 1; i <= m_dTotalColumns; i++)
  306. {
  307. if (i != m_dTotalColumns)
  308. {
  309. if (m_bExcel)
  310. {
  311. nullString.Format("\" \"%s", m_sSeparator);
  312. }
  313. else
  314. {
  315. nullString.Format("\"\"%s", m_sSeparator);
  316. }
  317. m_stempSql += nullString;
  318. }
  319. else
  320. {
  321. if (m_bExcel)
  322. {
  323. m_stempSql += "\" \"";
  324. }
  325. else
  326. {
  327. m_stempSql += "\"\"";
  328. }
  329. }
  330. }
  331. for (int j = m_dTotalRows + 1; j < row; j++)
  332. {
  333. m_dCurrentRow++;
  334. m_aRows.Add(m_stempSql);
  335. }
  336. }
  337. }
  338. else
  339. {
  340. tempRow = m_dCurrentRow;
  341. m_dCurrentRow = m_dTotalRows;
  342. }
  343. // Insert new row
  344. m_dCurrentRow++;
  345. m_aRows.Add(m_stempString);
  346. if (row > m_dTotalRows)
  347. {
  348. m_dTotalRows = row;
  349. }
  350. else if (!row)
  351. {
  352. m_dTotalRows = m_dCurrentRow;
  353. m_dCurrentRow = tempRow;
  354. }
  355. if (!m_bTransaction)
  356. {
  357. Commit();
  358. }
  359. return true;
  360. }
  361. // Replace or add a cell into Excel spreadsheet using header row or column alphabet.
  362. // Default is add cell into new row.
  363. // Set Auto to false if want to force column to be used as header name
  364. bool CSpreadSheet::AddCell(CString CellValue, CString column, long row, bool Auto)
  365. {
  366. short columnIndex = CalculateColumnNumber(column, Auto);
  367. if (columnIndex == 0)
  368. {
  369. return false;
  370. }
  371. if (AddCell(CellValue, columnIndex, row))
  372. {
  373. return true;
  374. }
  375. return false;
  376. }
  377. // Replace or add a cell into spreadsheet using column number
  378. // Default is add cell into new row.
  379. bool CSpreadSheet::AddCell(CString CellValue, short column, long row)
  380. {
  381. if (column == 0)
  382. {
  383. m_sLastError = "Column cannot be zero\n";
  384. return false;
  385. }
  386. long tempRow;
  387. if (m_bExcel) // For Excel spreadsheet
  388. {
  389. if (column > m_aFieldNames.GetSize() + 1)
  390. {
  391. m_sLastError = "Cell column to be added cannot be greater than the number of fields\n";
  392. return false;
  393. }
  394. }
  395. else // For text delimited spreadsheet
  396. {
  397. // Update largest number of columns if necessary
  398. if (column > m_dTotalColumns)
  399. {
  400. m_dTotalColumns = column;
  401. }
  402. }
  403. if (row)
  404. {
  405. if (row <= m_dTotalRows)
  406. {
  407. ReadRow(m_atempArray, row);
  408. // Change desired row
  409. m_atempArray.SetAtGrow(column-1, CellValue);
  410. if (row == 1)
  411. {
  412. if (m_bExcel) // Check for duplicate header row field
  413. {
  414. for (int i = 0; i < m_atempArray.GetSize(); i++)
  415. {
  416. for (int j = 0; j < m_atempArray.GetSize(); j++)
  417. {
  418. if ((i != j) && (m_atempArray.GetAt(i) == m_atempArray.GetAt(j)))
  419. {
  420. m_sLastError.Format("Duplicate header row field:%s\n", m_atempArray.GetAt(i));
  421. return false;
  422. }
  423. }
  424. }
  425. }
  426. // Update header row
  427. m_aFieldNames.RemoveAll();
  428. m_aFieldNames.Copy(m_atempArray);
  429. }
  430. if (!AddRow(m_atempArray, row, true))
  431. {
  432. return false;
  433. }
  434. if (!m_bTransaction)
  435. {
  436. Commit();
  437. }
  438. return true;
  439. }
  440. else
  441. {
  442. // Insert null rows until specified row
  443. m_dCurrentRow = m_dTotalRows;
  444. m_stempSql.Empty();
  445. CString nullString;
  446. for (int i = 1; i <= m_dTotalColumns; i++)
  447. {
  448. if (i != m_dTotalColumns)
  449. {
  450. if (m_bExcel)
  451. {
  452. nullString.Format("\" \"%s", m_sSeparator);
  453. }
  454. else
  455. {
  456. nullString.Format("\"\"%s", m_sSeparator);
  457. }
  458. m_stempSql += nullString;
  459. }
  460. else
  461. {
  462. if (m_bExcel)
  463. {
  464. m_stempSql += "\" \"";
  465. }
  466. else
  467. {
  468. m_stempSql += "\"\"";
  469. }
  470. }
  471. }
  472. for (int j = m_dTotalRows + 1; j < row; j++)
  473. {
  474. m_dCurrentRow++;
  475. m_aRows.Add(m_stempSql);
  476. }
  477. }
  478. }
  479. else
  480. {
  481. tempRow = m_dCurrentRow;
  482. m_dCurrentRow = m_dTotalRows;
  483. }
  484. // Insert cell
  485. m_dCurrentRow++;
  486. m_stempString.Empty();
  487. for (int j = 1; j <= m_dTotalColumns; j++)
  488. {
  489. if (j != m_dTotalColumns) // Not last column
  490. {
  491. if (j != column)
  492. {
  493. if (m_bExcel)
  494. {
  495. m_stempSql.Format("\" \"%s", m_sSeparator);
  496. }
  497. else
  498. {
  499. m_stempSql.Format("\"\"%s", m_sSeparator);
  500. }
  501. m_stempString += m_stempSql;
  502. }
  503. else
  504. {
  505. m_stempSql.Format("\"%s\"%s", CellValue, m_sSeparator);
  506. m_stempString += m_stempSql;
  507. }
  508. }
  509. else // Last column
  510. {
  511. if (j != column)
  512. {
  513. if (m_bExcel)
  514. {
  515. m_stempString += "\" \"";
  516. }
  517. else
  518. {
  519. m_stempString += "\"\"";
  520. }
  521. }
  522. else
  523. {
  524. m_stempSql.Format("\"%s\"", CellValue);
  525. m_stempString += m_stempSql;
  526. }
  527. }
  528. }
  529. m_aRows.Add(m_stempString);
  530. if (row > m_dTotalRows)
  531. {
  532. m_dTotalRows = row;
  533. }
  534. else if (!row)
  535. {
  536. m_dTotalRows = m_dCurrentRow;
  537. m_dCurrentRow = tempRow;
  538. }
  539. if (!m_bTransaction)
  540. {
  541. Commit();
  542. }
  543. return true;
  544. }
  545. // Search and replace rows in Excel spreadsheet
  546. bool CSpreadSheet::ReplaceRows(CStringArray &NewRowValues, CStringArray &OldRowValues)
  547. {
  548. if (m_bExcel) // If file is an Excel spreadsheet
  549. {
  550. m_Database->OpenEx(m_sDsn, CDatabase::noOdbcDialog);
  551. m_stempSql.Format("UPDATE [%s] SET ", m_sSheetName);
  552. for (int i = 0; i < NewRowValues.GetSize(); i++)
  553. {
  554. m_stempString.Format("[%s]='%s', ", m_aFieldNames.GetAt(i), NewRowValues.GetAt(i));
  555. m_stempSql = m_stempSql + m_stempString;
  556. }
  557. m_stempSql.Delete(m_stempSql.GetLength()-2, 2);
  558. m_stempSql = m_stempSql + " WHERE (";
  559. for (int j = 0; j < OldRowValues.GetSize()-1; j++)
  560. {
  561. m_stempString.Format("[%s]='%s' AND ", m_aFieldNames.GetAt(j), OldRowValues.GetAt(j));
  562. m_stempSql = m_stempSql + m_stempString;
  563. }
  564. m_stempSql.Delete(m_stempSql.GetLength()-4, 5);
  565. m_stempSql += ")";
  566. try
  567. {
  568. m_Database->ExecuteSQL(m_stempSql);
  569. m_Database->Close();
  570. Open();
  571. return true;
  572. }
  573. catch(CDBException *e)
  574. {
  575. m_sLastError = e->m_strError;
  576. m_Database->Close();
  577. return false;
  578. }
  579. }
  580. else // if file is a text delimited file
  581. {
  582. m_sLastError = "Function not available for text delimited file\n";
  583. return false;
  584. }
  585. }
  586. // Read a row from spreadsheet.
  587. // Default is read the next row
  588. bool CSpreadSheet::ReadRow(CStringArray &RowValues, long row)
  589. {
  590. // Check if row entered is more than number of rows in sheet
  591. if (row <= m_aRows.GetSize())
  592. {
  593. if (row != 0)
  594. {
  595. m_dCurrentRow = row;
  596. }
  597. else if (m_dCurrentRow > m_aRows.GetSize())
  598. {
  599. return false;
  600. }
  601. // Read the desired row
  602. RowValues.RemoveAll();
  603. m_stempString = m_aRows.GetAt(m_dCurrentRow-1);
  604. m_dCurrentRow++;
  605. // Search for separator to split row
  606. int separatorPosition;
  607. m_stempSql.Format("\"%s\"", m_sSeparator);
  608. separatorPosition = m_stempString.Find(m_stempSql); // If separator is "?"
  609. if (separatorPosition != -1)
  610. {
  611. // Save columns
  612. int nCount = 0;
  613. int stringStartingPosition = 0;
  614. while (separatorPosition != -1)
  615. {
  616. nCount = separatorPosition - stringStartingPosition;
  617. RowValues.Add(m_stempString.Mid(stringStartingPosition, nCount));
  618. stringStartingPosition = separatorPosition + m_stempSql.GetLength();
  619. separatorPosition = m_stempString.Find(m_stempSql, stringStartingPosition);
  620. }
  621. nCount = m_stempString.GetLength() - stringStartingPosition;
  622. RowValues.Add(m_stempString.Mid(stringStartingPosition, nCount));
  623. // Remove quotes from first column
  624. m_stempString = RowValues.GetAt(0);
  625. m_stempString.Delete(0, 1);
  626. RowValues.SetAt(0, m_stempString);
  627. // Remove quotes from last column
  628. m_stempString = RowValues.GetAt(RowValues.GetSize()-1);
  629. m_stempString.Delete(m_stempString.GetLength()-1, 1);
  630. RowValues.SetAt(RowValues.GetSize()-1, m_stempString);
  631. return true;
  632. }
  633. else
  634. {
  635. // Save columns
  636. separatorPosition = m_stempString.Find(m_sSeparator); // if separator is ?
  637. if (separatorPosition != -1)
  638. {
  639. int nCount = 0;
  640. int stringStartingPosition = 0;
  641. while (separatorPosition != -1)
  642. {
  643. nCount = separatorPosition - stringStartingPosition;
  644. RowValues.Add(m_stempString.Mid(stringStartingPosition, nCount));
  645. stringStartingPosition = separatorPosition + m_sSeparator.GetLength();
  646. separatorPosition = m_stempString.Find(m_sSeparator, stringStartingPosition);
  647. }
  648. nCount = m_stempString.GetLength() - stringStartingPosition;
  649. RowValues.Add(m_stempString.Mid(stringStartingPosition, nCount));
  650. return true;
  651. }
  652. else // Treat spreadsheet as having one column
  653. {
  654. // Remove opening and ending quotes if any
  655. int quoteBegPos = m_stempString.Find('\"');
  656. int quoteEndPos = m_stempString.ReverseFind('\"');
  657. if ((quoteBegPos == 0) && (quoteEndPos == m_stempString.GetLength()-1))
  658. {
  659. m_stempString.Delete(0, 1);
  660. m_stempString.Delete(m_stempString.GetLength()-1, 1);
  661. }
  662. RowValues.Add(m_stempString);
  663. }
  664. }
  665. }
  666. m_sLastError = "Desired row is greater than total number of rows in spreadsheet\n";
  667. return false;
  668. }
  669. // Read a column from Excel spreadsheet using header row or column alphabet.
  670. // Set Auto to false if want to force column to be used as header name
  671. bool CSpreadSheet::ReadColumn(CStringArray &ColumnValues, CString column, bool Auto)
  672. {
  673. short columnIndex = CalculateColumnNumber(column, Auto);
  674. if (columnIndex == 0)
  675. {
  676. return false;
  677. }
  678. if (ReadColumn(ColumnValues, columnIndex))
  679. {
  680. return true;
  681. }
  682. return false;
  683. }
  684. // Read a column from spreadsheet using column number
  685. bool CSpreadSheet::ReadColumn(CStringArray &ColumnValues, short column)
  686. {
  687. if (column == 0)
  688. {
  689. m_sLastError = "Column cannot be zero\n";
  690. return false;
  691. }
  692. int tempRow = m_dCurrentRow;
  693. m_dCurrentRow = 1;
  694. ColumnValues.RemoveAll();
  695. for (int i = 1; i <= m_aRows.GetSize(); i++)
  696. {
  697. // Read each row
  698. if (ReadRow(m_atempArray, i))
  699. {
  700. // Get value of cell in desired column
  701. if (column <= m_atempArray.GetSize())
  702. {
  703. ColumnValues.Add(m_atempArray.GetAt(column-1));
  704. }
  705. else
  706. {
  707. ColumnValues.Add("");
  708. }
  709. }
  710. else
  711. {
  712. m_dCurrentRow = tempRow;
  713. m_sLastError = "Error reading row\n";
  714. return false;
  715. }
  716. }
  717. m_dCurrentRow = tempRow;
  718. return true;
  719. }
  720. // Read a cell from Excel spreadsheet using header row or column alphabet.
  721. // Default is read the next cell in next row.
  722. // Set Auto to false if want to force column to be used as header name
  723. bool CSpreadSheet::ReadCell (CString &CellValue, CString column, long row, bool Auto)
  724. {
  725. short columnIndex = CalculateColumnNumber(column, Auto);
  726. if (columnIndex == 0)
  727. {
  728. return false;
  729. }
  730. if (ReadCell(CellValue, columnIndex, row))
  731. {
  732. return true;
  733. }
  734. return false;
  735. }
  736. // Read a cell from spreadsheet using column number.
  737. // Default is read the next cell in next row.
  738. bool CSpreadSheet::ReadCell (CString &CellValue, short column, long row)
  739. {
  740. if (column == 0)
  741. {
  742. m_sLastError = "Column cannot be zero\n";
  743. return false;
  744. }
  745. int tempRow = m_dCurrentRow;
  746. if (row)
  747. {
  748. m_dCurrentRow = row;
  749. }
  750. if (ReadRow(m_atempArray, m_dCurrentRow))
  751. {
  752. // Get value of cell in desired column
  753. if (column <= m_atempArray.GetSize())
  754. {
  755. CellValue = m_atempArray.GetAt(column-1);
  756. }
  757. else
  758. {
  759. CellValue.Empty();
  760. m_dCurrentRow = tempRow;
  761. return false;
  762. }
  763. m_dCurrentRow = tempRow;
  764. return true;
  765. }
  766. m_dCurrentRow = tempRow;
  767. m_sLastError = "Error reading row\n";
  768. return false;
  769. }
  770. // Begin transaction
  771. void CSpreadSheet::BeginTransaction()
  772. {
  773. m_bTransaction = true;
  774. }
  775. // Save changes to spreadsheet
  776. bool CSpreadSheet::Commit()
  777. {
  778. if (m_bExcel) // If file is an Excel spreadsheet
  779. {
  780. m_Database->OpenEx(m_sDsn, CDatabase::noOdbcDialog);
  781. if (m_bAppend)
  782. {
  783. // Delete old sheet if it exists
  784. m_stempString= "[" + m_sSheetName + "$A1:IV65536]";
  785. m_stempSql.Format ("DROP TABLE %s", m_stempString);
  786. try
  787. {
  788. m_Database->ExecuteSQL(m_stempSql);
  789. }
  790. catch(CDBException *e)
  791. {
  792. m_sLastError = e->m_strError;
  793. m_Database->Close();
  794. return false;
  795. }
  796. // Create new sheet
  797. m_stempSql.Format("CREATE TABLE [%s$A1:IV65536] (", m_sSheetName);
  798. for (int j = 0; j < m_aFieldNames.GetSize(); j++)
  799. {
  800. m_stempSql = m_stempSql + "[" + m_aFieldNames.GetAt(j) +"]" + " char(255), ";
  801. }
  802. m_stempSql.Delete(m_stempSql.GetLength()-2, 2);
  803. m_stempSql += ")";
  804. }
  805. else
  806. {
  807. // Create new sheet
  808. m_stempSql.Format("CREATE TABLE [%s] (", m_sSheetName);
  809. for (int i = 0; i < m_aFieldNames.GetSize(); i++)
  810. {
  811. m_stempSql = m_stempSql + "[" + m_aFieldNames.GetAt(i) +"]" + " char(255), ";
  812. }
  813. m_stempSql.Delete(m_stempSql.GetLength()-2, 2);
  814. m_stempSql += ")";
  815. }
  816. try
  817. {
  818. m_Database->ExecuteSQL(m_stempSql);
  819. if (!m_bAppend)
  820. {
  821. m_dTotalColumns = m_aFieldNames.GetSize();
  822. m_bAppend = true;
  823. }
  824. }
  825. catch(CDBException *e)
  826. {
  827. m_sLastError = e->m_strError;
  828. m_Database->Close();
  829. return false;
  830. }
  831. // Save changed data
  832. for (int k = 1; k < m_dTotalRows; k++)
  833. {
  834. ReadRow(m_atempArray, k+1);
  835. // Create Insert SQL
  836. m_stempSql.Format("INSERT INTO [%s$A1:IV%d] (", m_sSheetName, k);
  837. for (int i = 0; i < m_atempArray.GetSize(); i++)
  838. {
  839. m_stempString.Format("[%s], ", m_aFieldNames.GetAt(i));
  840. m_stempSql = m_stempSql + m_stempString;
  841. }
  842. m_stempSql.Delete(m_stempSql.GetLength()-2, 2);
  843. m_stempSql += ") VALUES (";
  844. for (int j = 0; j < m_atempArray.GetSize(); j++)
  845. {
  846. m_stempString.Format("'%s', ", m_atempArray.GetAt(j));
  847. m_stempSql = m_stempSql + m_stempString;
  848. }
  849. m_stempSql.Delete(m_stempSql.GetLength()-2, 2);
  850. m_stempSql += ")";
  851. // Add row
  852. try
  853. {
  854. m_Database->ExecuteSQL(m_stempSql);
  855. }
  856. catch(CDBException *e)
  857. {
  858. m_sLastError = e->m_strError;
  859. m_Database->Close();
  860. return false;
  861. }
  862. }
  863. m_Database->Close();
  864. m_bTransaction = false;
  865. return true;
  866. }
  867. else // if file is a text delimited file
  868. {
  869. try
  870. {
  871. CFile *File = NULL;
  872. File = new CFile(m_sFile, CFile::modeCreate | CFile::modeWrite | CFile::shareDenyNone);
  873. if (File != NULL)
  874. {
  875. CArchive *Archive = NULL;
  876. Archive = new CArchive(File, CArchive::store);
  877. if (Archive != NULL)
  878. {
  879. for (int i = 0; i < m_aRows.GetSize(); i++)
  880. {
  881. Archive->WriteString(m_aRows.GetAt(i));
  882. Archive->WriteString("\r\n");
  883. }
  884. delete Archive;
  885. delete File;
  886. m_bTransaction = false;
  887. return true;
  888. }
  889. delete File;
  890. }
  891. }
  892. catch(...)
  893. {
  894. }
  895. m_sLastError = "Error writing file\n";
  896. return false;
  897. }
  898. }
  899. // Undo changes to spreadsheet
  900. bool CSpreadSheet::RollBack()
  901. {
  902. if (Open())
  903. {
  904. m_bTransaction = false;
  905. return true;
  906. }
  907. m_sLastError = "Error in returning to previous state\n";
  908. return false;
  909. }
  910. bool CSpreadSheet::Convert(CString SheetOrSeparator)
  911. {
  912. // Prepare file
  913. m_stempString = m_sFile;
  914. m_stempString.Delete(m_stempString.GetLength()-4, 4);
  915. if (m_bExcel) // If file is an Excel spreadsheet
  916. {
  917. m_stempString += ".csv";
  918. CSpreadSheet tempSheet(m_stempString, SheetOrSeparator, false);
  919. // Stop convert if text delimited file exists
  920. if (tempSheet.GetTotalColumns() != 0)
  921. {
  922. return false;
  923. }
  924. tempSheet.BeginTransaction();
  925. for (int i = 1; i <= m_dTotalRows; i++)
  926. {
  927. if (!ReadRow(m_atempArray, i))
  928. {
  929. return false;
  930. }
  931. if (!tempSheet.AddRow(m_atempArray, i))
  932. {
  933. return false;
  934. }
  935. }
  936. if (!tempSheet.Commit())
  937. {
  938. return false;
  939. }
  940. return true;
  941. }
  942. else // if file is a text delimited file
  943. {
  944. m_stempString += ".xls";
  945. CSpreadSheet tempSheet(m_stempString, SheetOrSeparator, false);
  946. // Stop convert if Excel file exists
  947. if (tempSheet.GetTotalColumns() != 0)
  948. {
  949. return false;
  950. }
  951. GetFieldNames(m_atempArray);
  952. // Check for duplicate header row field
  953. bool duplicate = false;
  954. for (int i = 0; i < m_atempArray.GetSize(); i++)
  955. {
  956. for (int j = 0; j < m_atempArray.GetSize(); j++)
  957. {
  958. if ((i != j) && (m_atempArray.GetAt(i) == m_atempArray.GetAt(j)))
  959. {
  960. m_sLastError.Format("Duplicate header row field:%s\n", m_atempArray.GetAt(i));
  961. duplicate = true;
  962. }
  963. }
  964. }
  965. if (duplicate) // Create dummy header row
  966. {
  967. m_atempArray.RemoveAll();
  968. for (int k = 1; k <= m_dTotalColumns; k++)
  969. {
  970. m_stempString.Format("%d", k);
  971. m_atempArray.Add(m_stempString);
  972. }
  973. if (!tempSheet.AddHeaders(m_atempArray))
  974. {
  975. return false;
  976. }
  977. for (int l = 1; l <= m_dTotalRows; l++)
  978. {
  979. if (!ReadRow(m_atempArray, l))
  980. {
  981. return false;
  982. }
  983. if (!tempSheet.AddRow(m_atempArray, l+1))
  984. {
  985. return false;
  986. }
  987. }
  988. return true;
  989. }
  990. else
  991. {
  992. if (!tempSheet.AddHeaders(m_atempArray))
  993. {
  994. return false;
  995. }
  996. for (int l = 2; l <= m_dTotalRows; l++)
  997. {
  998. if (!ReadRow(m_atempArray, l))
  999. {
  1000. return false;
  1001. }
  1002. if (!tempSheet.AddRow(m_atempArray, l))
  1003. {
  1004. return false;
  1005. }
  1006. }
  1007. return true;
  1008. }
  1009. }
  1010. }
  1011. // Open a text delimited file for reading or writing
  1012. bool CSpreadSheet::Open()
  1013. {
  1014. if (m_bExcel) // If file is an Excel spreadsheet
  1015. {
  1016. m_Database->OpenEx(m_sDsn, CDatabase::noOdbcDialog);
  1017. // Open Sheet
  1018. m_rSheet = new CRecordset( m_Database );
  1019. m_sSql.Format("SELECT * FROM [%s$A1:IV65536]", m_sSheetName);
  1020. try
  1021. {
  1022. m_rSheet->Open(CRecordset::forwardOnly, m_sSql, CRecordset::readOnly);
  1023. }
  1024. catch(...)
  1025. {
  1026. delete m_rSheet;
  1027. m_rSheet = NULL;
  1028. m_Database->Close();
  1029. return false;
  1030. }
  1031. // Get number of columns
  1032. m_dTotalColumns = m_rSheet->m_nResultCols;
  1033. if (m_dTotalColumns != 0)
  1034. {
  1035. m_aRows.RemoveAll();
  1036. m_stempString.Empty();
  1037. m_bAppend = true;
  1038. m_dTotalRows++; // Keep count of total number of rows
  1039. // Get field names i.e header row
  1040. for (int i = 0; i < m_dTotalColumns; i++)
  1041. {
  1042. m_stempSql = m_rSheet->m_rgODBCFieldInfos[i].m_strName;
  1043. m_aFieldNames.Add(m_stempSql);
  1044. // Join up all the columns into a string
  1045. if (i != m_dTotalColumns-1) // Not last column
  1046. {
  1047. m_stempString = m_stempString + "\"" + m_stempSql + "\"" + m_sSeparator;
  1048. }
  1049. else // Last column
  1050. {
  1051. m_stempString = m_stempString + "\"" + m_stempSql + "\"";
  1052. }
  1053. }
  1054. // Store the header row as the first row in memory
  1055. m_aRows.Add(m_stempString);
  1056. // Read and store the rest of the rows in memory
  1057. while (!m_rSheet->IsEOF())
  1058. {
  1059. m_dTotalRows++; // Keep count of total number of rows
  1060. try
  1061. {
  1062. // Get all the columns in a row
  1063. m_stempString.Empty();
  1064. for (short column = 0; column < m_dTotalColumns; column++)
  1065. {
  1066. m_rSheet->GetFieldValue(column, m_stempSql);
  1067. // Join up all the columns into a string
  1068. if (column != m_dTotalColumns-1) // Not last column
  1069. {
  1070. m_stempString = m_stempString + "\"" + m_stempSql + "\"" + m_sSeparator;
  1071. }
  1072. else // Last column
  1073. {
  1074. m_stempString = m_stempString + "\"" + m_stempSql + "\"";
  1075. }
  1076. }
  1077. // Store the obtained row in memory
  1078. m_aRows.Add(m_stempString);
  1079. m_rSheet->MoveNext();
  1080. }
  1081. catch (...)
  1082. {
  1083. m_sLastError = "Error reading row\n";
  1084. delete m_rSheet;
  1085. m_rSheet = NULL;
  1086. m_Database->Close();
  1087. return false;
  1088. }
  1089. }
  1090. }
  1091. m_rSheet->Close();
  1092. delete m_rSheet;
  1093. m_rSheet = NULL;
  1094. m_Database->Close();
  1095. m_dCurrentRow = 1;
  1096. return true;
  1097. }
  1098. else // if file is a text delimited file
  1099. {
  1100. try
  1101. {
  1102. CFile *File = NULL;
  1103. File = new CFile(m_sFile, CFile::modeRead | CFile::shareDenyNone);
  1104. if (File != NULL)
  1105. {
  1106. CArchive *Archive = NULL;
  1107. Archive = new CArchive(File, CArchive::load);
  1108. if (Archive != NULL)
  1109. {
  1110. m_aRows.RemoveAll();
  1111. // Read and store all rows in memory
  1112. while(Archive->ReadString(m_stempString))
  1113. {
  1114. m_aRows.Add(m_stempString);
  1115. }
  1116. ReadRow(m_aFieldNames, 1); // Get field names i.e header row
  1117. delete Archive;
  1118. delete File;
  1119. // Get total number of rows
  1120. m_dTotalRows = m_aRows.GetSize();
  1121. // Get the largest number of columns
  1122. for (int i = 0; i < m_aRows.GetSize(); i++)
  1123. {
  1124. ReadRow(m_atempArray, i);
  1125. if (m_atempArray.GetSize() > m_dTotalColumns)
  1126. {
  1127. m_dTotalColumns = m_atempArray.GetSize();
  1128. }
  1129. }
  1130. if (m_dTotalColumns != 0)
  1131. {
  1132. m_bAppend = true;
  1133. }
  1134. return true;
  1135. }
  1136. delete File;
  1137. }
  1138. }
  1139. catch(...)
  1140. {
  1141. }
  1142. m_sLastError = "Error in opening file\n";
  1143. return false;
  1144. }
  1145. }
  1146. // Convert Excel column in alphabet into column number
  1147. short CSpreadSheet::CalculateColumnNumber(CString column, bool Auto)
  1148. {
  1149. if (Auto)
  1150. {
  1151. int firstLetter, secondLetter;
  1152. column.MakeUpper();
  1153. if (column.GetLength() == 1)
  1154. {
  1155. firstLetter = column.GetAt(0);
  1156. return (firstLetter - 65 + 1); // 65 is A in ascii
  1157. }
  1158. else if (column.GetLength() == 2)
  1159. {
  1160. firstLetter = column.GetAt(0);
  1161. secondLetter = column.GetAt(1);
  1162. return ((firstLetter - 65 + 1)*26 + (secondLetter - 65 + 1)); // 65 is A in ascii
  1163. }
  1164. }
  1165. // Check if it is a valid field name
  1166. for (int i = 0; i < m_aFieldNames.GetSize(); i++)
  1167. {
  1168. if (!column.Compare(m_aFieldNames.GetAt(i)))
  1169. {
  1170. return (i + 1);
  1171. }
  1172. }
  1173. m_sLastError = "Invalid field name or column alphabet\n";
  1174. return 0;
  1175. }
  1176. // Get the name of the Excel-ODBC driver
  1177. void CSpreadSheet::GetExcelDriver()
  1178. {
  1179. char szBuf[2001];
  1180. WORD cbBufMax = 2000;
  1181. WORD cbBufOut;
  1182. char *pszBuf = szBuf;
  1183. // Get the names of the installed drivers ("odbcinst.h" has to be included )
  1184. if(!SQLGetInstalledDrivers(szBuf,cbBufMax,& cbBufOut))
  1185. {
  1186. m_sExcelDriver = "";
  1187. }
  1188. // Search for the driver...
  1189. do
  1190. {
  1191. if( strstr( pszBuf, "Excel" ) != 0 )
  1192. {
  1193. // Found !
  1194. m_sExcelDriver = CString( pszBuf );
  1195. break;
  1196. }
  1197. pszBuf = strchr( pszBuf, '\0' ) + 1;
  1198. }
  1199. while( pszBuf[1] != '\0' );
  1200. }