/* ==================================================================== Licensed to the Apache Software Foundation (ASF) under one or more contributor license agreements. See the NOTICE file distributed with this work for additional information regarding copyright ownership. The ASF licenses this file to You under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. ==================================================================== */ package org.apache.poi.hssf.usermodel; import java.awt.font.FontRenderContext; import java.awt.font.TextAttribute; import java.awt.font.TextLayout; import java.awt.geom.AffineTransform; import java.io.PrintWriter; import java.text.AttributedString; import java.text.DecimalFormat; import java.text.NumberFormat; import java.util.ArrayList; import java.util.Iterator; import java.util.List; import java.util.TreeMap; import org.apache.poi.ddf.EscherRecord; import org.apache.poi.hssf.model.Sheet; import org.apache.poi.hssf.model.Workbook; import org.apache.poi.hssf.record.CellValueRecordInterface; import org.apache.poi.hssf.record.DVRecord; import org.apache.poi.hssf.record.EscherAggregate; import org.apache.poi.hssf.record.Record; import org.apache.poi.hssf.record.RowRecord; import org.apache.poi.hssf.record.SCLRecord; import org.apache.poi.hssf.record.WSBoolRecord; import org.apache.poi.hssf.record.WindowTwoRecord; import org.apache.poi.hssf.record.aggregates.DataValidityTable; import org.apache.poi.hssf.record.formula.FormulaShifter; import org.apache.poi.hssf.util.PaneInformation; import org.apache.poi.ss.usermodel.CellStyle; import org.apache.poi.ss.usermodel.Row; import org.apache.poi.ss.util.CellRangeAddress; import org.apache.poi.hssf.util.Region; import org.apache.poi.util.POILogFactory; import org.apache.poi.util.POILogger; /** * High level representation of a worksheet. * @author Andrew C. Oliver (acoliver at apache dot org) * @author Glen Stampoultzis (glens at apache.org) * @author Libin Roman (romal at vistaportal.com) * @author Shawn Laubach (slaubach at apache dot org) (Just a little) * @author Jean-Pierre Paris (jean-pierre.paris at m4x dot org) (Just a little, too) * @author Yegor Kozlov (yegor at apache.org) (Autosizing columns) */ public class HSSFSheet implements org.apache.poi.ss.usermodel.Sheet { private static final int DEBUG = POILogger.DEBUG; /* Constants for margins */ public static final short LeftMargin = Sheet.LeftMargin; public static final short RightMargin = Sheet.RightMargin; public static final short TopMargin = Sheet.TopMargin; public static final short BottomMargin = Sheet.BottomMargin; public static final byte PANE_LOWER_RIGHT = (byte)0; public static final byte PANE_UPPER_RIGHT = (byte)1; public static final byte PANE_LOWER_LEFT = (byte)2; public static final byte PANE_UPPER_LEFT = (byte)3; /** * Used for compile-time optimization. This is the initial size for the collection of * rows. It is currently set to 20. If you generate larger sheets you may benefit * by setting this to a higher number and recompiling a custom edition of HSSFSheet. */ public final static int INITIAL_CAPACITY = 20; /** * reference to the low level Sheet object */ private Sheet sheet; /** stores HSSFRows by Integer (zero-based row number) key */ private TreeMap rows; protected Workbook book; protected HSSFWorkbook workbook; private int firstrow; private int lastrow; private static POILogger log = POILogFactory.getLogger(HSSFSheet.class); /** * Creates new HSSFSheet - called by HSSFWorkbook to create a sheet from * scratch. You should not be calling this from application code (its protected anyhow). * * @param workbook - The HSSF Workbook object associated with the sheet. * @see org.apache.poi.hssf.usermodel.HSSFWorkbook#createSheet() */ protected HSSFSheet(HSSFWorkbook workbook) { sheet = Sheet.createSheet(); rows = new TreeMap(); this.workbook = workbook; this.book = workbook.getWorkbook(); } /** * Creates an HSSFSheet representing the given Sheet object. Should only be * called by HSSFWorkbook when reading in an exisiting file. * * @param workbook - The HSSF Workbook object associated with the sheet. * @param sheet - lowlevel Sheet object this sheet will represent * @see org.apache.poi.hssf.usermodel.HSSFWorkbook#createSheet() */ protected HSSFSheet(HSSFWorkbook workbook, Sheet sheet) { this.sheet = sheet; rows = new TreeMap(); this.workbook = workbook; this.book = workbook.getWorkbook(); setPropertiesFromSheet(sheet); } HSSFSheet cloneSheet(HSSFWorkbook workbook) { return new HSSFSheet(workbook, sheet.cloneSheet()); } /** * used internally to set the properties given a Sheet object */ private void setPropertiesFromSheet(Sheet sheet) { RowRecord row = sheet.getNextRow(); boolean rowRecordsAlreadyPresent = row!=null; while (row != null) { createRowFromRecord(row); row = sheet.getNextRow(); } CellValueRecordInterface[] cvals = sheet.getValueRecords(); long timestart = System.currentTimeMillis(); if (log.check( POILogger.DEBUG )) log.log(DEBUG, "Time at start of cell creating in HSSF sheet = ", new Long(timestart)); HSSFRow lastrow = null; // Add every cell to its row for (int i = 0; i < cvals.length; i++) { CellValueRecordInterface cval = cvals[i]; long cellstart = System.currentTimeMillis(); HSSFRow hrow = lastrow; if (hrow == null || hrow.getRowNum() != cval.getRow()) { hrow = getRow( cval.getRow() ); lastrow = hrow; if (hrow == null) { // Some tools (like Perl module Spreadsheet::WriteExcel - bug 41187) skip the RowRecords // Excel, OpenOffice.org and GoogleDocs are all OK with this, so POI should be too. if (rowRecordsAlreadyPresent) { // if at least one row record is present, all should be present. throw new RuntimeException("Unexpected missing row when some rows already present"); } // create the row record on the fly now. RowRecord rowRec = new RowRecord(cval.getRow()); sheet.addRow(rowRec); hrow = createRowFromRecord(rowRec); } } if (log.check( POILogger.DEBUG )) log.log( DEBUG, "record id = " + Integer.toHexString( ( (Record) cval ).getSid() ) ); hrow.createCellFromRecord( cval ); if (log.check( POILogger.DEBUG )) log.log( DEBUG, "record took ", new Long( System.currentTimeMillis() - cellstart ) ); } if (log.check( POILogger.DEBUG )) log.log(DEBUG, "total sheet cell creation took ", new Long(System.currentTimeMillis() - timestart)); } /** * Create a new row within the sheet and return the high level representation * * @param rownum row number * @return High level HSSFRow object representing a row in the sheet * @see org.apache.poi.hssf.usermodel.HSSFRow * @see #removeRow(HSSFRow) */ public HSSFRow createRow(int rownum) { HSSFRow row = new HSSFRow(workbook, this, rownum); addRow(row, true); return row; } /** * Used internally to create a high level Row object from a low level row object. * USed when reading an existing file * @param row low level record to represent as a high level Row and add to sheet * @return HSSFRow high level representation */ private HSSFRow createRowFromRecord(RowRecord row) { HSSFRow hrow = new HSSFRow(workbook, this, row); addRow(hrow, false); return hrow; } /** * Remove a row from this sheet. All cells contained in the row are removed as well * * @param row representing a row to remove. */ public void removeRow(Row row) { HSSFRow hrow = (HSSFRow) row; if (rows.size() > 0) { Integer key = new Integer(row.getRowNum()); HSSFRow removedRow = (HSSFRow) rows.remove(key); if (removedRow != row) { if (removedRow != null) { rows.put(key, removedRow); } throw new RuntimeException("Specified row does not belong to this sheet"); } if (hrow.getRowNum() == getLastRowNum()) { lastrow = findLastRow(lastrow); } if (hrow.getRowNum() == getFirstRowNum()) { firstrow = findFirstRow(firstrow); } sheet.removeRow(hrow.getRowRecord()); } } /** * used internally to refresh the "last row" when the last row is removed. */ private int findLastRow(int lastrow) { if (lastrow < 1) { return -1; } int rownum = lastrow - 1; HSSFRow r = getRow(rownum); while (r == null && rownum > 0) { r = getRow(--rownum); } if (r == null) { return -1; } return rownum; } /** * used internally to refresh the "first row" when the first row is removed. */ private int findFirstRow(int firstrow) { int rownum = firstrow + 1; HSSFRow r = getRow(rownum); while (r == null && rownum <= getLastRowNum()) { r = getRow(++rownum); } if (rownum > getLastRowNum()) return -1; return rownum; } /** * add a row to the sheet * * @param addLow whether to add the row to the low level model - false if its already there */ private void addRow(HSSFRow row, boolean addLow) { rows.put(new Integer(row.getRowNum()), row); if (addLow) { sheet.addRow(row.getRowRecord()); } if (row.getRowNum() > getLastRowNum()) { lastrow = row.getRowNum(); } if (row.getRowNum() < getFirstRowNum()) { firstrow = row.getRowNum(); } } /** * Returns the logical row (not physical) 0-based. If you ask for a row that is not * defined you get a null. This is to say row 4 represents the fifth row on a sheet. * @param rowIndex row to get * @return HSSFRow representing the rownumber or null if its not defined on the sheet */ public HSSFRow getRow(int rowIndex) { return (HSSFRow) rows.get(new Integer(rowIndex)); } /** * Returns the number of phsyically defined rows (NOT the number of rows in the sheet) */ public int getPhysicalNumberOfRows() { return rows.size(); } /** * Gets the first row on the sheet * @return the number of the first logical row on the sheet, zero based */ public int getFirstRowNum() { return firstrow; } /** * Gets the number last row on the sheet. * Owing to idiosyncrasies in the excel file * format, if the result of calling this method * is zero, you can't tell if that means there * are zero rows on the sheet, or one at * position zero. For that case, additionally * call {@link #getPhysicalNumberOfRows()} to * tell if there is a row at position zero * or not. * @return the number of the last row contained in this sheet, zero based. */ public int getLastRowNum() { return lastrow; } /** * Creates a data validation object * @param dataValidation The Data validation object settings */ public void addValidationData(HSSFDataValidation dataValidation) { if (dataValidation == null) { throw new IllegalArgumentException("objValidation must not be null"); } DataValidityTable dvt = sheet.getOrCreateDataValidityTable(); DVRecord dvRecord = dataValidation.createDVRecord(workbook); dvt.addDataValidation(dvRecord); } /** * Get the DVRecords objects that are associated to this sheet * @return a list of DVRecord instances */ public List getDVRecords() { List dvRecords = new ArrayList(); List records = sheet.getRecords(); for(int index=0; index