/************************************************************************ * * TableReader.java * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License version 2.1, as published by the Free Software Foundation. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, * MA 02111-1307 USA * * Copyright: 2002-2008 by Henrik Just * * All Rights Reserved. */ // Version 1.0 (2008-11-22) package writer2latex.office; import java.util.LinkedList; import java.util.Vector; import org.w3c.dom.Element; import org.w3c.dom.Node; import org.w3c.dom.NodeList; import writer2latex.util.Misc; /** *

This class reads a table from a table:table or table:sub-table element * and presents it as an n by m grid. In addition it gives access to the * absolute and relative widths of tables, columns and cells.

*/ public class TableReader { //private OfficeReader ofr; private Element tableNode; private LinkedList cols = new LinkedList(); private LinkedList rows = new LinkedList(); private LinkedList> cells = new LinkedList>(); private int nMaxCols = 1; // real number of columns (count to last non-empty) private int nMaxRows = 1; // real number of rows (count to last non-empty) private String[] sColWidth; private String[] sRelColWidth; private String sTableWidth; private String sRelTableWidth; private Vector printRanges; /** *

The constructor reads a table from a table:table or table:sub-table * node.

* @param ofr the OfficeReader object to get style information from * @param tableNode the table node */ public TableReader(OfficeReader ofr, Element tableNode) { //this.ofr = ofr; this.tableNode = tableNode; if (!tableNode.hasChildNodes()) { return; } // empty table! NodeList nl = tableNode.getChildNodes(); int nLen = nl.getLength(); for (int i = 0; i < nLen; i++) { Node child = nl.item(i); if (child.getNodeType() == Node.ELEMENT_NODE) { String sName = child.getNodeName(); if (sName.equals(XMLString.TABLE_TABLE_COLUMN)) { readTableColumn(child,false,false); } else if (sName.equals(XMLString.TABLE_TABLE_COLUMNS)) { readTableColumns(child,false,false); } else if (sName.equals(XMLString.TABLE_TABLE_COLUMN_GROUP)) { readTableColumnGroup(child,false,false); } else if (sName.equals(XMLString.TABLE_TABLE_HEADER_COLUMNS)) { readTableHeaderColumns(child,false,false); } else if (sName.equals(XMLString.TABLE_TABLE_ROW)) { readTableRow(child,false,false); } else if (sName.equals(XMLString.TABLE_TABLE_ROWS)) { readTableRows(child,false,false); } else if (sName.equals(XMLString.TABLE_TABLE_ROW_GROUP)) { readTableRowGroup(child,false,false); } else if (sName.equals(XMLString.TABLE_TABLE_HEADER_ROWS)) { readTableHeaderRows(child,false,false); } } } // Read table width from style StyleWithProperties tableStyle = ofr.getTableStyle(getTableStyleName()); if (tableStyle!=null) { sTableWidth = tableStyle.getProperty(XMLString.STYLE_WIDTH); sRelTableWidth = tableStyle.getProperty(XMLString.STYLE_REL_WIDTH); } // Determine column widths int nCols = cols.size(); sColWidth = new String[nCols]; sRelColWidth = new String[nCols]; int[] nRelColWidth = new int[nCols]; boolean bHasRelWidth=true; // set to false if some columns does not have a relative width set int nColSum = 0; for (int nCol=0; nColnMaxRows) { nMaxRows = nMaxRow; } nMaxCol = nCol + Misc.getPosInteger(cell.getAttribute( XMLString.TABLE_NUMBER_COLUMNS_SPANNED),1); if (nMaxCol>nMaxCols) { nMaxCols = nMaxCol; } } } } // Finally get the print ranges, if any printRanges = new Vector(); if (!"false".equals(tableNode.getAttribute(XMLString.TABLE_PRINT))) { TableRangeParser parser = new TableRangeParser(tableNode.getAttribute(XMLString.TABLE_PRINT_RANGES)); while (parser.hasMoreRanges()) { int[] nRange = parser.getRange(); TableRange range = new TableRange(this); range.setFirstCol(nRange[0]); range.setFirstRow(nRange[1]); range.setLastCol(nRange[2]); range.setLastRow(nRange[3]); printRanges.add(range); } } } private void readTableColumn(Node node, boolean bHeader, boolean bDisplay) { int nRepeat = Misc.getPosInteger(Misc.getAttribute(node, XMLString.TABLE_NUMBER_COLUMNS_REPEATED),1); while (nRepeat-->0) { cols.add(new TableLine(node,bHeader,bDisplay)); } } private void readTableColumns(Node node, boolean bHeader, boolean bDisplay) { if (!node.hasChildNodes()) { return; } // no columns here! NodeList nl = node.getChildNodes(); int nLen = nl.getLength(); for (int i = 0; i < nLen; i++) { Node child = nl.item(i); if (child.getNodeType() == Node.ELEMENT_NODE) { String sName = child.getNodeName(); if (sName.equals(XMLString.TABLE_TABLE_COLUMN)) { readTableColumn(child,bHeader,bDisplay); } else if (sName.equals(XMLString.TABLE_TABLE_COLUMN_GROUP)) { readTableColumnGroup(child,bHeader,bDisplay); } } } } private void readTableColumnGroup(Node node, boolean bHeader, boolean bDisplay) { bDisplay = !"false".equals(Misc.getAttribute(node,XMLString.TABLE_DISPLAY)); if (!node.hasChildNodes()) { return; } // no columns here! NodeList nl = node.getChildNodes(); int nLen = nl.getLength(); for (int i = 0; i < nLen; i++) { Node child = nl.item(i); if (child.getNodeType() == Node.ELEMENT_NODE) { String sName = child.getNodeName(); if (sName.equals(XMLString.TABLE_TABLE_HEADER_COLUMNS)) { readTableHeaderColumns(child,bHeader,bDisplay); } else if (sName.equals(XMLString.TABLE_TABLE_COLUMN)) { readTableColumn(child,bHeader,bDisplay); } else if (sName.equals(XMLString.TABLE_TABLE_COLUMN_GROUP)) { readTableColumnGroup(child,bHeader,bDisplay); } } } } private void readTableHeaderColumns(Node node, boolean bHeader, boolean bDisplay) { readTableColumns(node,true,bDisplay); } private void readTableRow(Node node, boolean bHeader, boolean bDisplay) { int nRepeat = Misc.getPosInteger(Misc.getAttribute(node, XMLString.TABLE_NUMBER_ROWS_REPEATED),1); while (nRepeat-->0) { rows.add(new TableLine(node,bHeader,bDisplay)); // Read the cells in the row LinkedList row = new LinkedList(); if (node.hasChildNodes()) { NodeList nl = node.getChildNodes(); int nLen = nl.getLength(); for (int i = 0; i < nLen; i++) { Node child = nl.item(i); if (child.getNodeType() == Node.ELEMENT_NODE) { Element cell = (Element) child; String sName = cell.getTagName(); if (sName.equals(XMLString.TABLE_TABLE_CELL)) { int nColRepeat = Misc.getPosInteger(cell.getAttribute( XMLString.TABLE_NUMBER_COLUMNS_REPEATED),1); while (nColRepeat-->0) { row.add(cell); } } else if (sName.equals(XMLString.TABLE_COVERED_TABLE_CELL)) { int nColRepeat = Misc.getPosInteger(cell.getAttribute( XMLString.TABLE_NUMBER_COLUMNS_REPEATED),1); while (nColRepeat-->0) { row.add(cell); } } } } } cells.add(row); } } private void readTableRows(Node node, boolean bHeader, boolean bDisplay) { if (!node.hasChildNodes()) { return; } // no rows here! NodeList nl = node.getChildNodes(); int nLen = nl.getLength(); for (int i = 0; i < nLen; i++) { Node child = nl.item(i); if (child.getNodeType() == Node.ELEMENT_NODE) { String sName = child.getNodeName(); if (sName.equals(XMLString.TABLE_TABLE_ROW)) { readTableRow(child,bHeader,bDisplay); } else if (sName.equals(XMLString.TABLE_TABLE_ROW_GROUP)) { readTableRowGroup(child,bHeader,bDisplay); } } } } private void readTableRowGroup(Node node, boolean bHeader, boolean bDisplay) { bDisplay = !"false".equals(Misc.getAttribute(node,XMLString.TABLE_DISPLAY)); if (!node.hasChildNodes()) { return; } // no rows here! NodeList nl = node.getChildNodes(); int nLen = nl.getLength(); for (int i = 0; i < nLen; i++) { Node child = nl.item(i); if (child.getNodeType() == Node.ELEMENT_NODE) { String sName = child.getNodeName(); if (sName.equals(XMLString.TABLE_TABLE_HEADER_ROWS)) { readTableHeaderRows(child,bHeader,bDisplay); } else if (sName.equals(XMLString.TABLE_TABLE_ROW)) { readTableRow(child,bHeader,bDisplay); } else if (sName.equals(XMLString.TABLE_TABLE_ROW_GROUP)) { readTableRowGroup(child,bHeader,bDisplay); } } } } public boolean isSubTable() { return XMLString.TABLE_SUB_TABLE.equals(tableNode.getTagName()) || // old "true".equals(Misc.getAttribute(tableNode,XMLString.TABLE_IS_SUB_TABLE)); // oasis } private void readTableHeaderRows(Node node, boolean bHeader, boolean bDisplay) { readTableRows(node,true,bDisplay); } public String getTableName() { return tableNode.getAttribute(XMLString.TABLE_NAME); } public String getTableStyleName() { return tableNode.getAttribute(XMLString.TABLE_STYLE_NAME); } public String getTableWidth() { return sTableWidth; } public String getRelTableWidth() { return sRelTableWidth; } public int getRowCount() { return rows.size(); } public int getMaxRowCount() { return nMaxRows; } public int getFirstBodyRow() { int nRows = rows.size(); for (int nRow=0; nRow=cells.size()) { return null; } LinkedList row = cells.get(nRow); if (nCol<0 || nCol>=row.size()) { return null; } return (Element) row.get(nCol); } public String getCellStyleName(int nRow, int nCol) { Element cell = (Element) getCell(nRow,nCol); if (cell==null) { return null; } String s = cell.getAttribute(XMLString.TABLE_STYLE_NAME); // try the cell if (s!=null && s.length()>0) { return s; } s = getRow(nRow).getDefaultCellStyleName(); // try row default if (s!=null && s.length()>0) { return s; } s = getCol(nCol).getDefaultCellStyleName(); // try column default return s; } public String getCellWidth(int nRow, int nCol) { Element cell = (Element) getCell(nRow,nCol); if (cell==null) { return null; } int nCols = Misc.getPosInteger(cell.getAttribute(XMLString.TABLE_NUMBER_COLUMNS_SPANNED),1); String sWidth = sColWidth[nCol]; for (int i=nCol+1; i=rows.size()) { return null; } return rows.get(nRow); } public TableLine getCol(int nCol) { if (nCol<0 || nCol>=cols.size()) { return null; } return cols.get(nCol); } public int getPrintRangeCount() { return printRanges.size(); } public TableRange getPrintRange(int nIndex) { if (0<=nIndex && nIndex