/* * Copyright (c) 2006 Henri Sivonen * * Permission is hereby granted, free of charge, to any person obtaining a * copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, including without limitation * the rights to use, copy, modify, merge, publish, distribute, sublicense, * and/or sell copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER * DEALINGS IN THE SOFTWARE. */ package org.whattf.checker.table; import java.util.LinkedList; import org.whattf.checker.AttributeUtil; import org.whattf.checker.Checker; import org.xml.sax.Attributes; import org.xml.sax.SAXException; /** * Checks XHTML table integrity: overlapping cells, spanning past the end * of row group, etc. * * @version $Id$ * @author hsivonen */ public final class TableChecker extends Checker { /** * Constructor. */ public TableChecker() { super(); } /** * Holds the current table. (Premature optimization to avoid * peeking the top of the stack all the time.) */ private Table current; /** * A stack for holding the tables that are open and ancestors of * the current table. Grows from the tail. */ private final LinkedList<Table> stack = new LinkedList<Table>(); /** * Pushes the current table onto the stack and creates a new one. */ private void push() { if (current != null) { stack.addLast(current); } current = new Table(this); } /** * Ends the current table, discards it and pops the top of the * stack to be the new current table. * * @throws SAXException if ending the table throws */ private void pop() throws SAXException { if (current == null) { throw new IllegalStateException("Bug!"); } current.end(); if (stack.isEmpty()) { current = null; } else { current = stack.removeLast(); } } /** * @see org.whattf.checker.Checker#startElement(java.lang.String, * java.lang.String, java.lang.String, org.xml.sax.Attributes) */ public void startElement(String uri, String localName, String qName, Attributes atts) throws SAXException { if ("http://www.w3.org/1999/xhtml".equals(uri)) { if ("table".equals(localName)) { push(); } else if (current != null) { if ("td".equals(localName)) { current.startCell(false, atts); } else if ("th".equals(localName)) { current.startCell(true, atts); } else if ("tr".equals(localName)) { current.startRow(); } else if ("tbody".equals(localName) || "thead".equals(localName) || "tfoot".equals(localName)) { current.startRowGroup(localName); } else if ("col".equals(localName)) { current.startCol(AttributeUtil.parseNonNegativeInteger(atts.getValue( "", "span"))); } else if ("colgroup".equals(localName)) { current.startColGroup(AttributeUtil.parseNonNegativeInteger(atts.getValue( "", "span"))); } } } } /** * @see org.whattf.checker.Checker#endElement(java.lang.String, * java.lang.String, java.lang.String) */ public void endElement(String uri, String localName, String qName) throws SAXException { if ("http://www.w3.org/1999/xhtml".equals(uri)) { if ("table".equals(localName)) { pop(); } else if (current != null) { if ("td".equals(localName)) { current.endCell(); } else if ("th".equals(localName)) { current.endCell(); } else if ("tr".equals(localName)) { current.endRow(); } else if ("tbody".equals(localName) || "thead".equals(localName) || "tfoot".equals(localName)) { current.endRowGroup(); } else if ("col".equals(localName)) { current.endCol(); } else if ("colgroup".equals(localName)) { current.endColGroup(); } } } } /** * @see org.whattf.checker.Checker#reset() */ public void reset() { stack.clear(); current = null; } }