ExcelWorkbook.cs 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Text;
  4. using Biff8Excel.Interfaces;
  5. using System.IO;
  6. using System.Runtime.InteropServices;
  7. using System.Diagnostics;
  8. using Biff8Excel.Formulas;
  9. using Microsoft.Win32;
  10. using System.Collections;
  11. namespace Biff8Excel.Excel
  12. {
  13. public class ExcelWorkbook: IDisposable
  14. {
  15. string m_fname; // file name
  16. string m_password =""; // password protect the sheet contents
  17. //int m_WorkbookSizeOnDisk; // Size in bytes of the saved workbook
  18. bool bFontsCreated;
  19. bool bActiveSheetSet;
  20. List<BoundSheet> mColBoundSheets; // One boundsheet record for each worksheet
  21. WorkSheetCollection mColSheets; // Collection or worksheets
  22. List<ExcelFont> mColFonts; // Collection of Font Objects
  23. List<ExcelFormat> mColFormats; // Collection of Format Objects
  24. List<ExcelCellStyle> mColCellStyls; // Collection of workbook cell styles
  25. // records
  26. ExcelSst recSST;
  27. Helper recHelper;
  28. ExternSheet recExternSheet;
  29. Records.BOF recBOF;
  30. Records.Windows1 recWindow1;
  31. Records.SupBook recSupBook;
  32. ExcelBinaryWriter bw;
  33. //---------------------------------------------------------------------------------------
  34. // Procedure : Function Calculate
  35. // : calculates any Formula cells, required if the workbook is to be viewed
  36. // : using the free Excel 97 viewer, as this program does not calculate formulae,
  37. // : it only reads the result of a formula value stored on disk by the full version of Excel.
  38. //---------------------------------------------------------------------------------------
  39. internal bool Calculate()
  40. {
  41. ExcelCell oCell;
  42. ushort oldFailCount;
  43. ushort failCount =0;
  44. ExcelCellDictionary ColFormulaCells; // Collection of formula cells in the worksheet
  45. ExcelCellDictionary ColNumberFormulaCells; // Collection of all number and formula cells in the worksheet
  46. // build a collection of number and formula cells
  47. ColFormulaCells = new ExcelCellDictionary();
  48. ColNumberFormulaCells = new ExcelCellDictionary();
  49. foreach (ExcelWorksheet sheet in mColSheets)
  50. {
  51. if (sheet.TotalRows > 0)
  52. {
  53. //arrRows = sheet.Rows;
  54. foreach (KeyValuePair<ushort, ExcelCellLinkDictionary> kv in sheet.Rows)
  55. {
  56. foreach (KeyValuePair<ushort, ExcelCell> kvc in kv.Value)
  57. {
  58. oCell = kvc.Value ;
  59. if (oCell.CellType == Globals.CELL_TYPE_FORMULA)
  60. {
  61. ColFormulaCells.Add(sheet.sheetNumber.ToString("000") + "!" + oCell.CellReference, oCell);
  62. ColNumberFormulaCells.Add(sheet.sheetNumber.ToString("000") + "!" + oCell.CellReference, oCell);
  63. }
  64. else if (oCell.CellType == Globals.CELL_TYPE_NUMBER)
  65. {
  66. ColNumberFormulaCells.Add(sheet.sheetNumber.ToString("000") + "!" + oCell.CellReference, oCell);
  67. }
  68. }
  69. }
  70. } // end if
  71. }
  72. // Loop through the Formula cells and calculate each in turn
  73. // If a formula contains a cell reference and that cell has a formula then
  74. // it may be that it cannot be resolved until the nested formula is resolved so
  75. // we may need to loop more than once through the formula cells to resolve all formulas
  76. do
  77. {
  78. oldFailCount = failCount;
  79. failCount = 0;
  80. foreach (DictionaryEntry oCells in ColFormulaCells)
  81. {
  82. if ( !((ExcelCell)oCells.Value).CalculationResolved)
  83. {
  84. ((ExcelCell)oCells.Value).Calculate(ColNumberFormulaCells);
  85. if (((ExcelCell)oCells.Value).CalculationResolved == false)
  86. failCount++;
  87. }
  88. }
  89. // (failCount should always be less than or equal to oldFailCount)
  90. // if failcount and OldFailCount match then
  91. // we cannot resolve any more formulas so exit
  92. } while (oldFailCount != failCount);
  93. return (failCount == 0);
  94. }
  95. public ExcelCell CreateCell()
  96. {
  97. ExcelCell ec = new ExcelCell(this);
  98. ec.pWorkbook = this;
  99. return ec;
  100. }
  101. //private void CreateDefaultSheets()
  102. //{
  103. // BoundSheet bs;
  104. // ExcelWorksheet ws;
  105. // for (int i = 1; i < 4; i++)
  106. // {
  107. // bs = new BoundSheet();
  108. // bs.SheetName = "Sheet" + i.ToString();
  109. // ws = new ExcelWorksheet(this);
  110. // mColBoundSheets.Add(bs);
  111. // mColSheets.Add(ws);
  112. // }
  113. //}
  114. private void CreateDefaultsCellStyles()
  115. {
  116. ExcelCellStyle cs;
  117. for (int i = 1; i < 22; i++)
  118. {
  119. //cs = new ExcelCellStyle(this);
  120. cs = this.CreateStyle();
  121. // Same across all styles
  122. cs.VerticalAlignment = EnumVerticalAlignment.Bottom;
  123. cs.PatternForeColour = EnumColours.SystemBorder;
  124. cs.PatternBackColour = EnumColours.SystemBackground;
  125. cs.CellIsLocked = true;
  126. if (i == 1)
  127. cs.StyleType = EnumType.Style;
  128. else if (i >= 2 & i <= 3)
  129. {
  130. cs.StyleType = EnumType.Style;
  131. cs.FontIndex = 1;
  132. cs.UseParentStyleOptions = 0xF4;
  133. }
  134. if (i >= 4 && i <= 5)
  135. {
  136. cs.StyleType = EnumType.Style;
  137. cs.FontIndex = 2;
  138. cs.UseParentStyleOptions = 0xF4;
  139. }
  140. if (i >= 6 && i <= 15)
  141. {
  142. cs.StyleType = EnumType.Style;
  143. cs.UseParentStyleOptions = 0xF4;
  144. }
  145. if (i == 16)
  146. {
  147. cs.StyleType = EnumType.Cell;
  148. }
  149. if (i >= 17 && i <= 21)
  150. {
  151. cs.StyleType = EnumType.Style;
  152. cs.FontIndex = 1;
  153. cs.UseParentStyleOptions = 0xF8;
  154. if (i == 17) // 43
  155. cs.FormatIndex = 0x2B;
  156. else if (i == 18)
  157. cs.FormatIndex = 0x29; // 41
  158. else if (i == 19)
  159. cs.FormatIndex = 0x2C; // 44
  160. else if (i == 20)
  161. cs.FormatIndex = 0x2A; // 42
  162. else if (i == 21)
  163. cs.FormatIndex = 0x9; // 9
  164. }
  165. mColCellStyls.Add(cs);
  166. }
  167. }
  168. internal ushort CreateNewFont(ExcelFont fnt)
  169. {
  170. if (fnt.Name.Length == 0)
  171. throw new Biff8ExcelException("no font name");
  172. mColFonts.Add(fnt);
  173. return (ushort)mColFonts.Count; // one based ???
  174. }
  175. internal ushort CreateNewFormat(ExcelFormat fmt)
  176. {
  177. if (fmt.FormatString.Length == 0)
  178. return 0;
  179. mColFormats.Add(fmt);
  180. fmt.index = (ushort)(mColFormats.Count + 163); // zero based userdefined start at index 164
  181. return fmt.index;
  182. }
  183. internal ushort CreateNewStyle(ExcelCellStyle cs)
  184. {
  185. mColCellStyls.Add(cs);
  186. return (ushort)(mColCellStyls.Count - 1); // zero based
  187. }
  188. /// <summary>
  189. /// 新建样式
  190. /// </summary>
  191. /// <returns></returns>
  192. public ExcelCellStyle CreateStyle()
  193. {
  194. ExcelCellStyle cs = new ExcelCellStyle();
  195. cs.pWorkbook = this;
  196. return cs;
  197. }
  198. /// <summary>
  199. /// 新建Sheet
  200. /// </summary>
  201. /// <param name="sSheetName">显示的名称</param>
  202. /// <returns></returns>
  203. public ExcelWorksheet CreateSheet(string sSheetName)
  204. {
  205. return CreateSheet(sSheetName, true);
  206. }
  207. /// <summary>
  208. /// 新建Sheet
  209. /// </summary>
  210. /// <param name="sSheetName">显示的名称</param>
  211. /// <param name="bVisible">是否显示</param>
  212. /// <returns></returns>
  213. public ExcelWorksheet CreateSheet(string sSheetName, bool bVisible)
  214. {
  215. ExcelWorksheet ws;
  216. BoundSheet bs;
  217. // create the new worksheet;
  218. ws = new ExcelWorksheet(this); // create a new sheet
  219. //ws.Workbook = this; // pass in a reference to the parent workbook object
  220. ws.SheetName = sSheetName;
  221. ws.sheetNumber = (ushort)mColSheets.Count; // zero index
  222. mColSheets.Add(ws); // add it to the sheets collection
  223. // Add a Bound Sheet record
  224. bs = new BoundSheet();
  225. bs.SheetName = sSheetName;
  226. bs.Visible = (bVisible == true ? Visibility.Visible : Visibility.Hidden);
  227. mColBoundSheets.Add(bs);
  228. return ws;
  229. }
  230. internal Interfaces.IExternSheet ExterSheet
  231. {
  232. get { return recExternSheet; }
  233. }
  234. public ExcelWorkbook()
  235. {
  236. mColFonts = new List<ExcelFont>();
  237. mColFormats = new List<ExcelFormat>();
  238. mColCellStyls = new List<ExcelCellStyle>();
  239. mColSheets = new WorkSheetCollection();
  240. mColBoundSheets = new List<BoundSheet>();
  241. recSST = new ExcelSst();
  242. recSupBook = new Biff8Excel.Records.SupBook();
  243. recExternSheet = new ExternSheet();
  244. recHelper = new Helper();
  245. recWindow1 = new Biff8Excel.Records.Windows1();
  246. recWindow1.ActivewWorksheet = 1;
  247. recHelper = new Helper();
  248. recWindow1 = new Biff8Excel.Records.Windows1();
  249. recWindow1.ActivewWorksheet = 1;
  250. recHelper.ExcelWorkSheets = mColSheets;
  251. recExternSheet.Workbook = this;
  252. CreateDefaultsCellStyles();
  253. bw = new ExcelBinaryWriter();
  254. // 'Call CreateDefaultSheets
  255. }
  256. internal List<ExcelFont> Fonts
  257. {
  258. get { return mColFonts; }
  259. }
  260. internal List<ExcelFormat> Formats
  261. {
  262. get { return mColFormats; }
  263. }
  264. /// <summary>
  265. /// 得到已在的Sheet
  266. /// </summary>
  267. /// <param name="sheet">Sheet名称或索引号</param>
  268. /// <returns></returns>
  269. public ExcelWorksheet GetSheet(object sheet)
  270. {
  271. if (sheet is string)
  272. return mColSheets[sheet.ToString()];
  273. else if (sheet is int)
  274. return mColSheets[Convert.ToInt32(sheet)];
  275. throw new Biff8ExcelException("no type sheet");
  276. //if (sheet.ToString() == "0") sheet = 1;
  277. //return mColSheets[sheet.ToString()];
  278. }
  279. public Helper Helper
  280. {
  281. get { return recHelper; }
  282. }
  283. public string Password
  284. {
  285. set { m_password = value; }
  286. }
  287. internal List<ExcelCellStyle> Styles
  288. {
  289. get { return mColCellStyls; }
  290. }
  291. internal ExcelSst SharedStringTable
  292. {
  293. get { return recSST; }
  294. }
  295. /// <summary>
  296. /// 设定活动的Sheet(只能有一个活动的sheet)
  297. /// </summary>
  298. ///// <param name="sheet">Sheet名称或索引号</param>
  299. public object SetActiveSheet
  300. {
  301. set
  302. {
  303. ExcelWorksheet oSheet;
  304. int tmp;
  305. if (bActiveSheetSet)
  306. return;
  307. bActiveSheetSet = true;
  308. if (int.TryParse(value.ToString(), out tmp))
  309. {
  310. recWindow1.ActivewWorksheet = (ushort)(tmp - 1); //zero based
  311. oSheet = mColSheets[tmp];
  312. if (oSheet != null)
  313. oSheet.Selected = true;
  314. }
  315. else
  316. {
  317. mColSheets[value.ToString()].Selected = true;
  318. recWindow1.ActivewWorksheet = mColSheets[value.ToString()].sheetNumber;
  319. }
  320. }
  321. }
  322. internal ushort SheetIndexFromName(string sSheetName)
  323. {
  324. ushort count = 0;
  325. bool bfound = false ;
  326. foreach (BoundSheet bs in mColBoundSheets)
  327. {
  328. if (bs.SheetName.ToUpper() == sSheetName.ToUpper())
  329. {
  330. bfound = true;
  331. break;
  332. }
  333. count++;
  334. }
  335. //return (bfound ? count : (ushort)-1);
  336. if (bfound)
  337. return count;
  338. else
  339. throw new Biff8ExcelException("Invalid Sheet Name");
  340. }
  341. public void Save(string fname)
  342. {
  343. byte[] buffer;
  344. int count;
  345. uint offset;
  346. uint endOfWorkbook; // position of the end of the workbook stream
  347. string s;
  348. m_fname = fname;
  349. if (this.Calculate() == false)
  350. {
  351. // If MsgBox("Some Formula cells could not be calculated, do you want to create the sheet anyway?" & vbCrLf & vbCrLf & _
  352. // "This will only cause a problem if you intent the workbook to be opened using the Microsoft Excel 97 Viewer " & _
  353. // "as this program does not recalculate formulae but reads the value from disk" & vbCrLf & vbCrLf & _
  354. // "Create anyway?", vbInformation + vbYesNo, "Warning") = vbNo Then
  355. }
  356. // This will create the Sheets for saving to disk,
  357. // this is required as any cell styles, fonts, formats etc will only be created
  358. // during the saving of a sheet, as we need to no the offsets in the stream for each sheet
  359. // we need to precalculate the offsets before the workbook records are written
  360. foreach (ExcelWorksheet sheet in mColSheets)
  361. {
  362. // set the sheet password if supplied
  363. if (m_password.Length > 0)
  364. {
  365. sheet.password = m_password;
  366. }
  367. sheet.CreateSheetBuffer();
  368. }
  369. //ExcelBinaryWriter bw = new ExcelBinaryWriter();
  370. recBOF = new Biff8Excel.Records.BOF();
  371. recBOF.DataType = Biff8Excel.Records.EnumBofTypes.bofWorkbook;
  372. buffer = recBOF.GetByte();
  373. bw.Write(buffer); // 20
  374. // Not implemented client implementations of these options
  375. if (mColSheets.Count > 0)
  376. this.SetActiveSheet = 1;
  377. //this.SetActiveSheet(1);
  378. recWindow1.Width = 15360; // twips 1024 x
  379. recWindow1.Height = 11520; // twips 768
  380. recWindow1.VerticalScrollBar = true;
  381. recWindow1.HorizontalScrollBar = true;
  382. recWindow1.DisplayTabsAtBottom = true;
  383. recWindow1.SelectedWorksheets = 1;
  384. recWindow1.TabWidth = 600; // twips
  385. buffer = recWindow1.GetByte();
  386. bw.Write(buffer);
  387. // Default font
  388. //this.SetDefaultFont("Arial", 10); // sets default font if not already set
  389. this.SetDefaultFont("宋体", 12);
  390. foreach (ExcelFont fnt in mColFonts)
  391. {
  392. buffer = fnt.WriteRecord();
  393. bw.Write(buffer);
  394. }
  395. //Default Formats
  396. this.WriteDefaultFormats();
  397. // formats
  398. foreach (ExcelFormat fmt in mColFormats)
  399. {
  400. buffer = fmt.WriteRecord();
  401. bw.Write(buffer);
  402. }
  403. // Cell Styles;
  404. foreach (ExcelCellStyle cs in mColCellStyls)
  405. {
  406. buffer = cs.WriteRecord();
  407. bw.Write(buffer);
  408. }
  409. // Styles
  410. this.WriteDefaultStyles();
  411. // Bound Sheets
  412. // The bound sheet record for a worksheet holds the absolute byte
  413. // array position of the BOF record of a worksheet
  414. // we don't know this yet so we will have to write this record
  415. // again at the end of the stream write, this is just a place
  416. // holder in the stream for these records
  417. foreach (BoundSheet bs in mColBoundSheets)
  418. {
  419. bs.StreamOffset = (uint)bw.Position; // save this records position in the stream
  420. //bs.StreamOffset = offsets;
  421. buffer = bs.WriteRecord;
  422. bw.Write(buffer);
  423. }
  424. // Sup Book Record (only using 1)
  425. recSupBook.TotalSheets = (ushort)mColBoundSheets.Count;
  426. buffer = recSupBook.GetByte();
  427. bw.Write(buffer);
  428. // ExternSheet, Holds indexes to the Sheets used in formulas
  429. buffer = recExternSheet.WriteRecord();
  430. bw.Write(buffer);
  431. //recSST.AbsolutOffset = (uint)bw.Position;
  432. //Shared String Table
  433. buffer = recSST.WriteRecord();
  434. bw.Write(buffer);
  435. // EOF
  436. s = "0A000000";
  437. bw.Write(Globals.WriteBytes(s));
  438. // re-write the bound sheet record with the
  439. // calculated sheet stream positions
  440. //offset = bw.Position;
  441. offset = (uint)bw.Position; // end of workbook stream
  442. //offset = offsets;
  443. endOfWorkbook = offset; // save end of workbook position for moving back later on
  444. count = 1;
  445. foreach (BoundSheet bs in mColBoundSheets)
  446. {
  447. bw.Position = (int)bs.StreamOffset;
  448. bs.SheetOffset = offset;
  449. buffer = bs.WriteRecord;
  450. bw.Write(buffer);
  451. offset += mColSheets[count].SizeOnDisk;
  452. count++;
  453. }
  454. bw.Position = (int)endOfWorkbook; // restore the stream pointer;
  455. // Write each Sheet
  456. foreach (ExcelWorksheet ws in mColSheets)
  457. {
  458. ws.UpdateAbsoluteOffsets((uint)bw.Position);
  459. bw.Write(ws.DiskBuffer);
  460. }
  461. // Com的结构化保存(Biff8的要求)
  462. COM.Storage file;
  463. COM.COMStream Data;
  464. if (File.Exists(m_fname))
  465. {
  466. File.SetAttributes(m_fname, FileAttributes.Normal);
  467. File.Delete(m_fname);
  468. }
  469. file = COM.Storage.CreateDocFile(m_fname, COM.StorageMode.ReadWrite | COM.StorageMode.ShareExclusive);
  470. Data = file.CreateStream("workbook");
  471. Data.WriteBytes(bw.Stream);
  472. //FileStream fs;
  473. //fs = new FileStream(m_fname, FileMode.OpenOrCreate);
  474. //fs.Write(bw.Stream, 0, bw.Stream.Length);
  475. //fs.Close();
  476. file.Commit(0);
  477. file.Dispose();
  478. Data = null;
  479. }
  480. /// <summary>
  481. /// 设定默认字体
  482. /// </summary>
  483. /// <param name="Name">字体名</param>
  484. /// <param name="Size">字体尺寸</param>
  485. public void SetDefaultFont(string name, ushort size)
  486. {
  487. if (bFontsCreated)
  488. return;
  489. ExcelFont f;
  490. bFontsCreated = true;
  491. // Create 4 Fonts (All the Same???)
  492. for (int i = 1; i < 5; i++)
  493. {
  494. f = new ExcelFont();
  495. f.Name = name;
  496. f.Size = size;
  497. f.Colour = EnumColours.SystemText;
  498. f.Bold = false;
  499. mColFonts.Add(f);
  500. }
  501. }
  502. internal string[] SheetNames
  503. {
  504. get
  505. {
  506. if (mColSheets.Count == 0)
  507. return null;
  508. ushort x = 0;
  509. string[] s = new string[mColSheets.Count];
  510. foreach (ExcelWorksheet ws in mColSheets)
  511. {
  512. s[x] = ws.SheetName;
  513. x++;
  514. }
  515. return s;
  516. }
  517. }
  518. private void WriteDefaultStyles()
  519. {
  520. string s;
  521. s = "93020400108003FF"
  522. + "93020400118006FF"
  523. + "93020400128004FF"
  524. + "93020400138007FF"
  525. + "93020400008000FF"
  526. + "93020400148005FF";
  527. bw.Write(Globals.WriteBytes(s));
  528. }
  529. private void WriteDefaultFormats()
  530. {
  531. string s;
  532. s = "1E041800050013000022A322232C2323303B5C2D22A322232C232330"
  533. + "1E041D00060018000022A322232C2323303B5B5265645D5C2D22A322232C232330"
  534. + "1E041E00070019000022A322232C2323302E30303B5C2D22A322232C2323302E3030"
  535. + "1E04230008001E000022A322232C2323302E30303B5B5265645D5C2D22A322232C2323302E3030"
  536. + "1E0435002A003000005F2D22A3222A20232C2323305F2D3B5C2D22A3222A20232C2323305F2D3B5F2D22A3222A20222D225F2D3B5F2D405F2D"
  537. + "1E042C0029002700005F2D2A20232C2323305F2D3B5C2D2A20232C2323305F2D3B5F2D2A20222D225F2D3B5F2D405F2D"
  538. + "1E043D002C003800005F2D22A3222A20232C2323302E30305F2D3B5C2D22A3222A20232C2323302E30305F2D3B5F2D22A3222A20222D223F3F5F2D3B5F2D405F2D"
  539. + "1E0434002B002F00005F2D2A20232C2323302E30305F2D3B5C2D2A20232C2323302E30305F2D3B5F2D2A20222D223F3F5F2D3B5F2D405F2D";
  540. bw.Write(Globals.WriteBytes(s));
  541. }
  542. //internal int WorkbookSizeOnDisk
  543. //{
  544. // get { return m_WorkbookSizeOnDisk; }
  545. //}
  546. #region IDisposable 成员
  547. public void Dispose()
  548. {
  549. mColFonts = null;
  550. mColFormats = null;
  551. mColCellStyls = null;
  552. mColSheets.Dispose();
  553. mColSheets = null;
  554. mColBoundSheets = null;
  555. recSST.Dispose();
  556. recSST = null;
  557. recSupBook = null;
  558. recExternSheet.Dispose();
  559. recExternSheet = null;
  560. recHelper.Dispose();
  561. recHelper = null;
  562. recWindow1 = null;
  563. }
  564. #endregion
  565. }
  566. }