You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

TestHSSFWorkbook.java 26KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692
  1. /* ====================================================================
  2. Licensed to the Apache Software Foundation (ASF) under one or more
  3. contributor license agreements. See the NOTICE file distributed with
  4. this work for additional information regarding copyright ownership.
  5. The ASF licenses this file to You under the Apache License, Version 2.0
  6. (the "License"); you may not use this file except in compliance with
  7. the License. You may obtain a copy of the License at
  8. http://www.apache.org/licenses/LICENSE-2.0
  9. Unless required by applicable law or agreed to in writing, software
  10. distributed under the License is distributed on an "AS IS" BASIS,
  11. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. See the License for the specific language governing permissions and
  13. limitations under the License.
  14. ==================================================================== */
  15. package org.apache.poi.hssf.usermodel;
  16. import java.io.*;
  17. import java.util.List;
  18. import junit.framework.AssertionFailedError;
  19. import org.apache.poi.hssf.HSSFTestDataSamples;
  20. import org.apache.poi.hssf.HSSFITestDataProvider;
  21. import org.apache.poi.hssf.model.HSSFFormulaParser;
  22. import org.apache.poi.hssf.model.InternalWorkbook;
  23. import org.apache.poi.hssf.model.InternalSheet;
  24. import org.apache.poi.hssf.record.*;
  25. import org.apache.poi.ss.formula.ptg.Area3DPtg;
  26. import org.apache.poi.util.LittleEndian;
  27. import org.apache.poi.util.TempFile;
  28. import org.apache.poi.ss.usermodel.BaseTestWorkbook;
  29. import org.apache.poi.ss.util.CellRangeAddress;
  30. import org.apache.poi.poifs.filesystem.DirectoryEntry;
  31. import org.apache.poi.poifs.filesystem.DirectoryNode;
  32. import org.apache.poi.poifs.filesystem.NPOIFSFileSystem;
  33. import org.apache.poi.poifs.filesystem.POIFSFileSystem;
  34. import org.apache.poi.POIDataSamples;
  35. import org.apache.poi.hpsf.ClassID;
  36. /**
  37. * Tests for {@link HSSFWorkbook}
  38. */
  39. public final class TestHSSFWorkbook extends BaseTestWorkbook {
  40. public TestHSSFWorkbook() {
  41. super(HSSFITestDataProvider.instance);
  42. }
  43. /**
  44. * gives test code access to the {@link InternalWorkbook} within {@link HSSFWorkbook}
  45. */
  46. public static InternalWorkbook getInternalWorkbook(HSSFWorkbook wb) {
  47. return wb.getWorkbook();
  48. }
  49. public void testSetRepeatingRowsAndColumns() {
  50. // Test bug 29747
  51. HSSFWorkbook b = new HSSFWorkbook( );
  52. b.createSheet();
  53. b.createSheet();
  54. b.createSheet();
  55. b.setRepeatingRowsAndColumns( 2, 0,1,-1,-1 );
  56. NameRecord nameRecord = b.getWorkbook().getNameRecord( 0 );
  57. assertEquals(3, nameRecord.getSheetNumber());
  58. }
  59. public void testWindowOneDefaults() {
  60. HSSFWorkbook b = new HSSFWorkbook( );
  61. try {
  62. assertEquals(b.getActiveSheetIndex(), 0);
  63. assertEquals(b.getFirstVisibleTab(), 0);
  64. } catch (NullPointerException npe) {
  65. fail("WindowOneRecord in Workbook is probably not initialized");
  66. }
  67. }
  68. /**
  69. * Tests for {@link HSSFWorkbook#isHidden()} etc
  70. */
  71. public void testHidden() {
  72. HSSFWorkbook wb = new HSSFWorkbook();
  73. WindowOneRecord w1 = wb.getWorkbook().getWindowOne();
  74. assertEquals(false, wb.isHidden());
  75. assertEquals(false, w1.getHidden());
  76. wb.setHidden(true);
  77. assertEquals(true, wb.isHidden());
  78. assertEquals(true, w1.getHidden());
  79. wb = HSSFTestDataSamples.writeOutAndReadBack(wb);
  80. w1 = wb.getWorkbook().getWindowOne();
  81. wb.setHidden(true);
  82. assertEquals(true, wb.isHidden());
  83. assertEquals(true, w1.getHidden());
  84. wb.setHidden(false);
  85. assertEquals(false, wb.isHidden());
  86. assertEquals(false, w1.getHidden());
  87. }
  88. public void testSheetClone() {
  89. // First up, try a simple file
  90. HSSFWorkbook b = new HSSFWorkbook();
  91. assertEquals(0, b.getNumberOfSheets());
  92. b.createSheet("Sheet One");
  93. b.createSheet("Sheet Two");
  94. assertEquals(2, b.getNumberOfSheets());
  95. b.cloneSheet(0);
  96. assertEquals(3, b.getNumberOfSheets());
  97. // Now try a problem one with drawing records in it
  98. b = HSSFTestDataSamples.openSampleWorkbook("SheetWithDrawing.xls");
  99. assertEquals(1, b.getNumberOfSheets());
  100. b.cloneSheet(0);
  101. assertEquals(2, b.getNumberOfSheets());
  102. }
  103. public void testReadWriteWithCharts() {
  104. HSSFWorkbook b;
  105. HSSFSheet s;
  106. // Single chart, two sheets
  107. b = HSSFTestDataSamples.openSampleWorkbook("44010-SingleChart.xls");
  108. assertEquals(2, b.getNumberOfSheets());
  109. assertEquals("Graph2", b.getSheetName(1));
  110. s = b.getSheetAt(1);
  111. assertEquals(0, s.getFirstRowNum());
  112. assertEquals(8, s.getLastRowNum());
  113. // Has chart on 1st sheet??
  114. // FIXME
  115. assertNotNull(b.getSheetAt(0).getDrawingPatriarch());
  116. assertNull(b.getSheetAt(1).getDrawingPatriarch());
  117. assertFalse(b.getSheetAt(0).getDrawingPatriarch().containsChart());
  118. // We've now called getDrawingPatriarch() so
  119. // everything will be all screwy
  120. // So, start again
  121. b = HSSFTestDataSamples.openSampleWorkbook("44010-SingleChart.xls");
  122. b = HSSFTestDataSamples.writeOutAndReadBack(b);
  123. assertEquals(2, b.getNumberOfSheets());
  124. s = b.getSheetAt(1);
  125. assertEquals(0, s.getFirstRowNum());
  126. assertEquals(8, s.getLastRowNum());
  127. // Two charts, three sheets
  128. b = HSSFTestDataSamples.openSampleWorkbook("44010-TwoCharts.xls");
  129. assertEquals(3, b.getNumberOfSheets());
  130. s = b.getSheetAt(1);
  131. assertEquals(0, s.getFirstRowNum());
  132. assertEquals(8, s.getLastRowNum());
  133. s = b.getSheetAt(2);
  134. assertEquals(0, s.getFirstRowNum());
  135. assertEquals(8, s.getLastRowNum());
  136. // Has chart on 1st sheet??
  137. // FIXME
  138. assertNotNull(b.getSheetAt(0).getDrawingPatriarch());
  139. assertNull(b.getSheetAt(1).getDrawingPatriarch());
  140. assertNull(b.getSheetAt(2).getDrawingPatriarch());
  141. assertFalse(b.getSheetAt(0).getDrawingPatriarch().containsChart());
  142. // We've now called getDrawingPatriarch() so
  143. // everything will be all screwy
  144. // So, start again
  145. b = HSSFTestDataSamples.openSampleWorkbook("44010-TwoCharts.xls");
  146. b = HSSFTestDataSamples.writeOutAndReadBack(b);
  147. assertEquals(3, b.getNumberOfSheets());
  148. s = b.getSheetAt(1);
  149. assertEquals(0, s.getFirstRowNum());
  150. assertEquals(8, s.getLastRowNum());
  151. s = b.getSheetAt(2);
  152. assertEquals(0, s.getFirstRowNum());
  153. assertEquals(8, s.getLastRowNum());
  154. }
  155. public void testSelectedSheet_bug44523() {
  156. HSSFWorkbook wb=new HSSFWorkbook();
  157. HSSFSheet sheet1 = wb.createSheet("Sheet1");
  158. HSSFSheet sheet2 = wb.createSheet("Sheet2");
  159. HSSFSheet sheet3 = wb.createSheet("Sheet3");
  160. HSSFSheet sheet4 = wb.createSheet("Sheet4");
  161. confirmActiveSelected(sheet1, true);
  162. confirmActiveSelected(sheet2, false);
  163. confirmActiveSelected(sheet3, false);
  164. confirmActiveSelected(sheet4, false);
  165. wb.setSelectedTab(1);
  166. // Demonstrate bug 44525:
  167. // Well... not quite, since isActive + isSelected were also added in the same bug fix
  168. if (sheet1.isSelected()) {
  169. throw new AssertionFailedError("Identified bug 44523 a");
  170. }
  171. wb.setActiveSheet(1);
  172. if (sheet1.isActive()) {
  173. throw new AssertionFailedError("Identified bug 44523 b");
  174. }
  175. confirmActiveSelected(sheet1, false);
  176. confirmActiveSelected(sheet2, true);
  177. confirmActiveSelected(sheet3, false);
  178. confirmActiveSelected(sheet4, false);
  179. }
  180. public void testSelectMultiple() {
  181. HSSFWorkbook wb=new HSSFWorkbook();
  182. HSSFSheet sheet1 = wb.createSheet("Sheet1");
  183. HSSFSheet sheet2 = wb.createSheet("Sheet2");
  184. HSSFSheet sheet3 = wb.createSheet("Sheet3");
  185. HSSFSheet sheet4 = wb.createSheet("Sheet4");
  186. HSSFSheet sheet5 = wb.createSheet("Sheet5");
  187. HSSFSheet sheet6 = wb.createSheet("Sheet6");
  188. wb.setSelectedTabs(new int[] { 0, 2, 3});
  189. assertEquals(true, sheet1.isSelected());
  190. assertEquals(false, sheet2.isSelected());
  191. assertEquals(true, sheet3.isSelected());
  192. assertEquals(true, sheet4.isSelected());
  193. assertEquals(false, sheet5.isSelected());
  194. assertEquals(false, sheet6.isSelected());
  195. wb.setSelectedTabs(new int[] { 1, 3, 5});
  196. assertEquals(false, sheet1.isSelected());
  197. assertEquals(true, sheet2.isSelected());
  198. assertEquals(false, sheet3.isSelected());
  199. assertEquals(true, sheet4.isSelected());
  200. assertEquals(false, sheet5.isSelected());
  201. assertEquals(true, sheet6.isSelected());
  202. assertEquals(true, sheet1.isActive());
  203. assertEquals(false, sheet2.isActive());
  204. assertEquals(true, sheet1.isActive());
  205. assertEquals(false, sheet3.isActive());
  206. wb.setActiveSheet(2);
  207. assertEquals(false, sheet1.isActive());
  208. assertEquals(true, sheet3.isActive());
  209. if (false) { // helpful if viewing this workbook in excel:
  210. sheet1.createRow(0).createCell(0).setCellValue(new HSSFRichTextString("Sheet1"));
  211. sheet2.createRow(0).createCell(0).setCellValue(new HSSFRichTextString("Sheet2"));
  212. sheet3.createRow(0).createCell(0).setCellValue(new HSSFRichTextString("Sheet3"));
  213. sheet4.createRow(0).createCell(0).setCellValue(new HSSFRichTextString("Sheet4"));
  214. try {
  215. File fOut = TempFile.createTempFile("sheetMultiSelect", ".xls");
  216. FileOutputStream os = new FileOutputStream(fOut);
  217. wb.write(os);
  218. os.close();
  219. } catch (IOException e) {
  220. throw new RuntimeException(e);
  221. }
  222. }
  223. }
  224. public void testActiveSheetAfterDelete_bug40414() {
  225. HSSFWorkbook wb=new HSSFWorkbook();
  226. HSSFSheet sheet0 = wb.createSheet("Sheet0");
  227. HSSFSheet sheet1 = wb.createSheet("Sheet1");
  228. HSSFSheet sheet2 = wb.createSheet("Sheet2");
  229. HSSFSheet sheet3 = wb.createSheet("Sheet3");
  230. HSSFSheet sheet4 = wb.createSheet("Sheet4");
  231. // confirm default activation/selection
  232. confirmActiveSelected(sheet0, true);
  233. confirmActiveSelected(sheet1, false);
  234. confirmActiveSelected(sheet2, false);
  235. confirmActiveSelected(sheet3, false);
  236. confirmActiveSelected(sheet4, false);
  237. wb.setActiveSheet(3);
  238. wb.setSelectedTab(3);
  239. confirmActiveSelected(sheet0, false);
  240. confirmActiveSelected(sheet1, false);
  241. confirmActiveSelected(sheet2, false);
  242. confirmActiveSelected(sheet3, true);
  243. confirmActiveSelected(sheet4, false);
  244. wb.removeSheetAt(3);
  245. // after removing the only active/selected sheet, another should be active/selected in its place
  246. if (!sheet4.isSelected()) {
  247. throw new AssertionFailedError("identified bug 40414 a");
  248. }
  249. if (!sheet4.isActive()) {
  250. throw new AssertionFailedError("identified bug 40414 b");
  251. }
  252. confirmActiveSelected(sheet0, false);
  253. confirmActiveSelected(sheet1, false);
  254. confirmActiveSelected(sheet2, false);
  255. confirmActiveSelected(sheet4, true);
  256. sheet3 = sheet4; // re-align local vars in this test case
  257. // Some more cases of removing sheets
  258. // Starting with a multiple selection, and different active sheet
  259. wb.setSelectedTabs(new int[] { 1, 3, });
  260. wb.setActiveSheet(2);
  261. confirmActiveSelected(sheet0, false, false);
  262. confirmActiveSelected(sheet1, false, true);
  263. confirmActiveSelected(sheet2, true, false);
  264. confirmActiveSelected(sheet3, false, true);
  265. // removing a sheet that is not active, and not the only selected sheet
  266. wb.removeSheetAt(3);
  267. confirmActiveSelected(sheet0, false, false);
  268. confirmActiveSelected(sheet1, false, true);
  269. confirmActiveSelected(sheet2, true, false);
  270. // removing the only selected sheet
  271. wb.removeSheetAt(1);
  272. confirmActiveSelected(sheet0, false, false);
  273. confirmActiveSelected(sheet2, true, true);
  274. // The last remaining sheet should always be active+selected
  275. wb.removeSheetAt(1);
  276. confirmActiveSelected(sheet0, true, true);
  277. }
  278. private static void confirmActiveSelected(HSSFSheet sheet, boolean expected) {
  279. confirmActiveSelected(sheet, expected, expected);
  280. }
  281. private static void confirmActiveSelected(HSSFSheet sheet,
  282. boolean expectedActive, boolean expectedSelected) {
  283. assertEquals("active", expectedActive, sheet.isActive());
  284. assertEquals("selected", expectedSelected, sheet.isSelected());
  285. }
  286. /**
  287. * If Sheet.getSize() returns a different result to Sheet.serialize(), this will cause the BOF
  288. * records to be written with invalid offset indexes. Excel does not like this, and such
  289. * errors are particularly hard to track down. This test ensures that HSSFWorkbook throws
  290. * a specific exception as soon as the situation is detected. See bugzilla 45066
  291. */
  292. public void testSheetSerializeSizeMismatch_bug45066() {
  293. HSSFWorkbook wb = new HSSFWorkbook();
  294. InternalSheet sheet = wb.createSheet("Sheet1").getSheet();
  295. List<RecordBase> sheetRecords = sheet.getRecords();
  296. // one way (of many) to cause the discrepancy is with a badly behaved record:
  297. sheetRecords.add(new BadlyBehavedRecord());
  298. // There is also much logic inside Sheet that (if buggy) might also cause the discrepancy
  299. try {
  300. wb.getBytes();
  301. throw new AssertionFailedError("Identified bug 45066 a");
  302. } catch (IllegalStateException e) {
  303. // Expected badly behaved sheet record to cause exception
  304. assertTrue(e.getMessage().startsWith("Actual serialized sheet size"));
  305. }
  306. }
  307. /**
  308. * Checks that us and HSSFName play nicely with named ranges
  309. * that point to deleted sheets
  310. */
  311. public void testNamesToDeleteSheets() {
  312. HSSFWorkbook b = HSSFTestDataSamples.openSampleWorkbook("30978-deleted.xls");
  313. assertEquals(3, b.getNumberOfNames());
  314. // Sheet 2 is deleted
  315. assertEquals("Sheet1", b.getSheetName(0));
  316. assertEquals("Sheet3", b.getSheetName(1));
  317. Area3DPtg ptg;
  318. NameRecord nr;
  319. HSSFName n;
  320. /* ======= Name pointing to deleted sheet ====== */
  321. // First at low level
  322. nr = b.getWorkbook().getNameRecord(0);
  323. assertEquals("On2", nr.getNameText());
  324. assertEquals(0, nr.getSheetNumber());
  325. assertEquals(1, nr.getExternSheetNumber());
  326. assertEquals(1, nr.getNameDefinition().length);
  327. ptg = (Area3DPtg)nr.getNameDefinition()[0];
  328. assertEquals(1, ptg.getExternSheetIndex());
  329. assertEquals(0, ptg.getFirstColumn());
  330. assertEquals(0, ptg.getFirstRow());
  331. assertEquals(0, ptg.getLastColumn());
  332. assertEquals(2, ptg.getLastRow());
  333. // Now at high level
  334. n = b.getNameAt(0);
  335. assertEquals("On2", n.getNameName());
  336. assertEquals("", n.getSheetName());
  337. assertEquals("#REF!$A$1:$A$3", n.getRefersToFormula());
  338. /* ======= Name pointing to 1st sheet ====== */
  339. // First at low level
  340. nr = b.getWorkbook().getNameRecord(1);
  341. assertEquals("OnOne", nr.getNameText());
  342. assertEquals(0, nr.getSheetNumber());
  343. assertEquals(0, nr.getExternSheetNumber());
  344. assertEquals(1, nr.getNameDefinition().length);
  345. ptg = (Area3DPtg)nr.getNameDefinition()[0];
  346. assertEquals(0, ptg.getExternSheetIndex());
  347. assertEquals(0, ptg.getFirstColumn());
  348. assertEquals(2, ptg.getFirstRow());
  349. assertEquals(0, ptg.getLastColumn());
  350. assertEquals(3, ptg.getLastRow());
  351. // Now at high level
  352. n = b.getNameAt(1);
  353. assertEquals("OnOne", n.getNameName());
  354. assertEquals("Sheet1", n.getSheetName());
  355. assertEquals("Sheet1!$A$3:$A$4", n.getRefersToFormula());
  356. /* ======= Name pointing to 3rd sheet ====== */
  357. // First at low level
  358. nr = b.getWorkbook().getNameRecord(2);
  359. assertEquals("OnSheet3", nr.getNameText());
  360. assertEquals(0, nr.getSheetNumber());
  361. assertEquals(2, nr.getExternSheetNumber());
  362. assertEquals(1, nr.getNameDefinition().length);
  363. ptg = (Area3DPtg)nr.getNameDefinition()[0];
  364. assertEquals(2, ptg.getExternSheetIndex());
  365. assertEquals(0, ptg.getFirstColumn());
  366. assertEquals(0, ptg.getFirstRow());
  367. assertEquals(0, ptg.getLastColumn());
  368. assertEquals(1, ptg.getLastRow());
  369. // Now at high level
  370. n = b.getNameAt(2);
  371. assertEquals("OnSheet3", n.getNameName());
  372. assertEquals("Sheet3", n.getSheetName());
  373. assertEquals("Sheet3!$A$1:$A$2", n.getRefersToFormula());
  374. }
  375. /**
  376. * result returned by getRecordSize() differs from result returned by serialize()
  377. */
  378. private static final class BadlyBehavedRecord extends Record {
  379. public BadlyBehavedRecord() {
  380. //
  381. }
  382. public short getSid() {
  383. return 0x777;
  384. }
  385. public int serialize(int offset, byte[] data) {
  386. return 4;
  387. }
  388. public int getRecordSize() {
  389. return 8;
  390. }
  391. }
  392. /**
  393. * The sample file provided with bug 45582 seems to have one extra byte after the EOFRecord
  394. */
  395. public void testExtraDataAfterEOFRecord() {
  396. try {
  397. HSSFTestDataSamples.openSampleWorkbook("ex45582-22397.xls");
  398. } catch (RecordFormatException e) {
  399. if (e.getCause() instanceof LittleEndian.BufferUnderrunException) {
  400. throw new AssertionFailedError("Identified bug 45582");
  401. }
  402. }
  403. }
  404. /**
  405. * Test to make sure that NameRecord.getSheetNumber() is interpreted as a
  406. * 1-based sheet tab index (not a 1-based extern sheet index)
  407. */
  408. public void testFindBuiltInNameRecord() {
  409. // testRRaC has multiple (3) built-in name records
  410. // The second print titles name record has getSheetNumber()==4
  411. HSSFWorkbook wb = HSSFTestDataSamples.openSampleWorkbook("testRRaC.xls");
  412. NameRecord nr;
  413. assertEquals(3, wb.getWorkbook().getNumNames());
  414. nr = wb.getWorkbook().getNameRecord(2);
  415. // TODO - render full row and full column refs properly
  416. assertEquals("Sheet2!$A$1:$IV$1", HSSFFormulaParser.toFormulaString(wb, nr.getNameDefinition())); // 1:1
  417. try {
  418. wb.setRepeatingRowsAndColumns(3, 4, 5, 8, 11);
  419. } catch (RuntimeException e) {
  420. if (e.getMessage().equals("Builtin (7) already exists for sheet (4)")) {
  421. // there was a problem in the code which locates the existing print titles name record
  422. throw new RuntimeException("Identified bug 45720b");
  423. }
  424. throw e;
  425. }
  426. wb = HSSFTestDataSamples.writeOutAndReadBack(wb);
  427. assertEquals(3, wb.getWorkbook().getNumNames());
  428. nr = wb.getWorkbook().getNameRecord(2);
  429. assertEquals("Sheet2!E:F,Sheet2!$A$9:$IV$12", HSSFFormulaParser.toFormulaString(wb, nr.getNameDefinition())); // E:F,9:12
  430. }
  431. /**
  432. * Test that the storage clsid property is preserved
  433. */
  434. public void test47920() throws IOException {
  435. POIFSFileSystem fs1 = new POIFSFileSystem(POIDataSamples.getSpreadSheetInstance().openResourceAsStream("47920.xls"));
  436. HSSFWorkbook wb = new HSSFWorkbook(fs1);
  437. ClassID clsid1 = fs1.getRoot().getStorageClsid();
  438. ByteArrayOutputStream out = new ByteArrayOutputStream(4096);
  439. wb.write(out);
  440. byte[] bytes = out.toByteArray();
  441. POIFSFileSystem fs2 = new POIFSFileSystem(new ByteArrayInputStream(bytes));
  442. ClassID clsid2 = fs2.getRoot().getStorageClsid();
  443. assertTrue(clsid1.equals(clsid2));
  444. }
  445. /**
  446. * Tests that we can work with both {@link POIFSFileSystem}
  447. * and {@link NPOIFSFileSystem}
  448. */
  449. public void testDifferentPOIFS() throws Exception {
  450. // Open the two filesystems
  451. DirectoryNode[] files = new DirectoryNode[2];
  452. files[0] = (new POIFSFileSystem(HSSFTestDataSamples.openSampleFileStream("Simple.xls"))).getRoot();
  453. files[1] = (new NPOIFSFileSystem(HSSFTestDataSamples.getSampeFile("Simple.xls"))).getRoot();
  454. // Open without preserving nodes
  455. for(DirectoryNode dir : files) {
  456. HSSFWorkbook workbook = new HSSFWorkbook(dir, false);
  457. HSSFSheet sheet = workbook.getSheetAt(0);
  458. HSSFCell cell = sheet.getRow(0).getCell(0);
  459. assertEquals("replaceMe", cell .getRichStringCellValue().getString());
  460. }
  461. // Now re-check with preserving
  462. for(DirectoryNode dir : files) {
  463. HSSFWorkbook workbook = new HSSFWorkbook(dir, true);
  464. HSSFSheet sheet = workbook.getSheetAt(0);
  465. HSSFCell cell = sheet.getRow(0).getCell(0);
  466. assertEquals("replaceMe", cell .getRichStringCellValue().getString());
  467. }
  468. }
  469. public void testWordDocEmbeddedInXls() throws IOException {
  470. // Open the two filesystems
  471. DirectoryNode[] files = new DirectoryNode[2];
  472. files[0] = (new POIFSFileSystem(HSSFTestDataSamples.openSampleFileStream("WithEmbeddedObjects.xls"))).getRoot();
  473. files[1] = (new NPOIFSFileSystem(HSSFTestDataSamples.getSampeFile("WithEmbeddedObjects.xls"))).getRoot();
  474. // Check the embedded parts
  475. for(DirectoryNode root : files) {
  476. HSSFWorkbook hw = new HSSFWorkbook(root, true);
  477. List<HSSFObjectData> objects = hw.getAllEmbeddedObjects();
  478. boolean found = false;
  479. for (int i = 0; i < objects.size(); i++) {
  480. HSSFObjectData embeddedObject = objects.get(i);
  481. if (embeddedObject.hasDirectoryEntry()) {
  482. DirectoryEntry dir = embeddedObject.getDirectory();
  483. if (dir instanceof DirectoryNode) {
  484. DirectoryNode dNode = (DirectoryNode)dir;
  485. if (hasEntry(dNode,"WordDocument")) {
  486. found = true;
  487. }
  488. }
  489. }
  490. }
  491. assertTrue(found);
  492. }
  493. }
  494. /**
  495. * Checks that we can open a workbook with NPOIFS, and write it out
  496. * again (via POIFS) and have it be valid
  497. * @throws IOException
  498. */
  499. public void testWriteWorkbookFromNPOIFS() throws IOException {
  500. InputStream is = HSSFTestDataSamples.openSampleFileStream("WithEmbeddedObjects.xls");
  501. NPOIFSFileSystem fs = new NPOIFSFileSystem(is);
  502. // Start as NPOIFS
  503. HSSFWorkbook wb = new HSSFWorkbook(fs.getRoot(), true);
  504. assertEquals(3, wb.getNumberOfSheets());
  505. assertEquals("Root xls", wb.getSheetAt(0).getRow(0).getCell(0).getStringCellValue());
  506. // Will switch to POIFS
  507. wb = HSSFTestDataSamples.writeOutAndReadBack(wb);
  508. assertEquals(3, wb.getNumberOfSheets());
  509. assertEquals("Root xls", wb.getSheetAt(0).getRow(0).getCell(0).getStringCellValue());
  510. }
  511. public void testCellStylesLimit() {
  512. HSSFWorkbook wb = new HSSFWorkbook();
  513. int numBuiltInStyles = wb.getNumCellStyles();
  514. int MAX_STYLES = 4030;
  515. int limit = MAX_STYLES - numBuiltInStyles;
  516. for(int i=0; i < limit; i++){
  517. HSSFCellStyle style = wb.createCellStyle();
  518. }
  519. assertEquals(MAX_STYLES, wb.getNumCellStyles());
  520. try {
  521. HSSFCellStyle style = wb.createCellStyle();
  522. fail("expected exception");
  523. } catch (IllegalStateException e){
  524. assertEquals("The maximum number of cell styles was exceeded. " +
  525. "You can define up to 4000 styles in a .xls workbook", e.getMessage());
  526. }
  527. assertEquals(MAX_STYLES, wb.getNumCellStyles());
  528. }
  529. public void testSetSheetOrderHSSF(){
  530. HSSFWorkbook wb = new HSSFWorkbook();
  531. HSSFSheet s1 = wb.createSheet("first sheet");
  532. HSSFSheet s2 = wb.createSheet("other sheet");
  533. HSSFName name1 = wb.createName();
  534. name1.setNameName("name1");
  535. name1.setRefersToFormula("'first sheet'!D1");
  536. HSSFName name2 = wb.createName();
  537. name2.setNameName("name2");
  538. name2.setRefersToFormula("'other sheet'!C1");
  539. HSSFRow s1r1 = s1.createRow(2);
  540. HSSFCell c1 = s1r1.createCell(3);
  541. c1.setCellValue(30);
  542. HSSFCell c2 = s1r1.createCell(2);
  543. c2.setCellFormula("SUM('other sheet'!C1,'first sheet'!C1)");
  544. HSSFRow s2r1 = s2.createRow(0);
  545. HSSFCell c3 = s2r1.createCell(1);
  546. c3.setCellFormula("'first sheet'!D3");
  547. HSSFCell c4 = s2r1.createCell(2);
  548. c4.setCellFormula("'other sheet'!D3");
  549. // conditional formatting
  550. HSSFSheetConditionalFormatting sheetCF = s1.getSheetConditionalFormatting();
  551. HSSFConditionalFormattingRule rule1 = sheetCF.createConditionalFormattingRule(
  552. CFRuleRecord.ComparisonOperator.BETWEEN, "'first sheet'!D1", "'other sheet'!D1");
  553. HSSFConditionalFormattingRule [] cfRules = { rule1 };
  554. CellRangeAddress[] regions = {
  555. new CellRangeAddress(2, 4, 0, 0), // A3:A5
  556. };
  557. sheetCF.addConditionalFormatting(regions, cfRules);
  558. wb.setSheetOrder("other sheet", 0);
  559. // names
  560. assertEquals("'first sheet'!D1", wb.getName("name1").getRefersToFormula());
  561. assertEquals("'other sheet'!C1", wb.getName("name2").getRefersToFormula());
  562. // cells
  563. assertEquals("SUM('other sheet'!C1,'first sheet'!C1)", c2.getCellFormula());
  564. assertEquals("'first sheet'!D3", c3.getCellFormula());
  565. assertEquals("'other sheet'!D3", c4.getCellFormula());
  566. // conditional formatting
  567. HSSFConditionalFormatting cf = sheetCF.getConditionalFormattingAt(0);
  568. assertEquals("'first sheet'!D1", cf.getRule(0).getFormula1());
  569. assertEquals("'other sheet'!D1", cf.getRule(0).getFormula2());
  570. }
  571. private boolean hasEntry(DirectoryNode dirNode, String entryName) {
  572. try {
  573. dirNode.getEntry(entryName);
  574. return true;
  575. } catch (FileNotFoundException e) {
  576. return false;
  577. }
  578. }
  579. }