/************************************************************************ * * TextConverter.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-2010 by Henrik Just * * All Rights Reserved. * * Version 1.2 (2010-03-26) * */ package writer2latex.xhtml; import java.util.Hashtable; import java.util.Stack; import java.util.Vector; import java.util.LinkedList; import java.util.Locale; import java.text.Collator; //import org.w3c.dom.Document; import org.w3c.dom.Node; import org.w3c.dom.NodeList; import org.w3c.dom.Element; import writer2latex.util.Misc; import writer2latex.office.FontDeclaration; import writer2latex.office.XMLString; import writer2latex.office.IndexMark; import writer2latex.office.ListCounter; import writer2latex.office.ListStyle; import writer2latex.office.PropertySet; import writer2latex.office.StyleWithProperties; import writer2latex.office.OfficeReader; import writer2latex.office.TocReader; // Helper class (a struct) to contain information about an alphabetical // index entry. final class AlphabeticalEntry { String sWord; // the word for the index int nIndex; // the original index of this entry } // Helper class (a struct) to contain information about a toc entry // (ie. a heading, other paragraph or toc-mark) final class TocEntry { Element onode; // the original node String sLabel = null; // generated label for the entry int nFileIndex; // the file index for the generated content int nOutlineLevel; // the outline level for this heading int[] nOutlineNumber; // the natural outline number for this heading } // Helper class (a struct) to point back to indexes that should be processed final class IndexData { int nOutFileIndex; // the index of the out file containing the index Element onode; // the original node Element chapter; // the chapter containing this toc Element hnode; // a div node where the index should be added } public class TextConverter extends ConverterHelper { // Data used to handle splitting over several files // TODO: Accessor methods for sections int nSplit = 0; // The outline level at which to split files (0=no split) int nRepeatLevels = 5; // The number of levels to repeat when splitting (0=no repeat) private int nLastSplitLevel = 1; // The outline level at which the last split occured private int nDontSplitLevel = 0; // if > 0 splitting is forbidden boolean bAfterHeading=false; // last element was a top level heading protected Stack sections = new Stack(); // stack of nested sections Element[] currentHeading = new Element[7]; // Last headings (repeated when splitting) // Counters for generated numbers private ListCounter outlineNumbering; private Hashtable listCounters = new Hashtable(); private String sCurrentListLabel = null; // Mode used to handle floats (depends on source doc type and config) private int nFloatMode; // Data used for index bookkeeping private Vector indexes = new Vector(); // Data used to handle Alphabetical Index Vector index = new Vector(); // All words for the index private int nIndexIndex = -1; // Current index used for id's (of form idxN) private int nAlphabeticalIndex = -1; // File containing alphabetical index // Data used to handle Table of Contents private Vector tocEntries = new Vector(); // All potential(!) toc items private int nTocFileIndex = -1; // file index for main toc private Element currentChapter = null; // Node for the current chapter (level 1) heading private int nTocIndex = -1; // Current index for id's (of form tocN) private ListCounter naturalOutline = new ListCounter(); // Current "natural" outline number // Style names for foot- and endnotes private String sFntCitBodyStyle = null; private String sFntCitStyle = null; private String sEntCitBodyStyle = null; private String sEntCitStyle = null; // Gather the footnotes and endnotes private LinkedList footnotes = new LinkedList(); private LinkedList endnotes = new LinkedList(); // Sometimes we have to create an inlinenode in a block context // (labels for footnotes and endnotes) // We put it here and insert it in the first paragraph/heading to come: private Node asapNode = null; // When generating toc, a few things should be done differently private boolean bInToc = false; public TextConverter(OfficeReader ofr, XhtmlConfig config, Converter converter) { super(ofr,config,converter); nSplit = config.getXhtmlSplitLevel(); nRepeatLevels = config.getXhtmlRepeatLevels(); nFloatMode = ofr.isText() && config.xhtmlFloatObjects() ? DrawConverter.FLOATING : DrawConverter.ABSOLUTE; outlineNumbering = new ListCounter(ofr.getOutlineStyle()); // Styles for footnotes and endnotes PropertySet notes = ofr.getFootnotesConfiguration(); if (notes!=null) { sFntCitBodyStyle = notes.getProperty(XMLString.TEXT_CITATION_BODY_STYLE_NAME); sFntCitStyle = notes.getProperty(XMLString.TEXT_CITATION_STYLE_NAME); } notes = ofr.getEndnotesConfiguration(); if (notes!=null) { sEntCitBodyStyle = notes.getProperty(XMLString.TEXT_CITATION_BODY_STYLE_NAME); sEntCitStyle = notes.getProperty(XMLString.TEXT_CITATION_STYLE_NAME); } } /** Converts an office node as a complete text document * * @param onode the Office node containing the content to convert */ public void convertTextContent(Element onode) { Element hnode = converter.nextOutFile(); // Create form if (nSplit==0) { Element form = getDrawCv().createForm(); if (form!=null) { hnode.appendChild(form); hnode = form; } } // Convert content traverseBlockText(onode,hnode); // Generate all indexes int nIndexCount = indexes.size(); for (int i=0; i=0; i--) { TocEntry entry = tocEntries.get(i); if (XMLString.TEXT_H.equals(entry.onode.getTagName()) && entry.nFileIndex==nIndex && entry.nOutlineLevel<=nSplit) { entryCurrent = entry; break; } } if (entryCurrent==null) { entryCurrent = fakeEntry; if (nIndex==0) { bHasFrontMatter=true; } } // Determine the maximum outline level to include int nMaxLevel = entryCurrent.nOutlineLevel; if (nMaxLevelnPrevFileIndex+1) { // Skipping a file index means we have passed an index for (int k=nPrevFileIndex+1; k0) { inline.appendChild(converter.createTextNode(entry.sLabel)); if (!entry.sLabel.endsWith(" ")) { inline.appendChild(converter.createTextNode(" ")); } } traverseInlineText(entry.onode,inline); } } } if (nPrevFileIndex0) { Element block = converter.createElement(xpar.getBlockElement(sDisplayName)); if (!"(none)".equals(xpar.getBlockCss(sDisplayName))) { block.setAttribute("class",xpar.getBlockCss(sDisplayName)); } hnode.appendChild(block); curHnode = block; } boolean bMoreParagraphs = true; do { handleParagraph(child,curHnode); bMoreParagraphs = false; if (++i1) { // we cannot split due to a nested structure return node; } if (bAfterHeading && nLevel-nLastSplitLevel<=nRepeatLevels) { // we cannot split because we are right after a heading and the // maximum number of parent headings on the page is not reached return node; } if (nSplit>=nLevel && converter.outFileHasContent()) { // No objections, this is a level that causes splitting return converter.nextOutFile(); } return node; } /* Process a text:section tag (returns current html node) */ private Node handleSection(Node onode, Node hnode) { String sName = Misc.getAttribute(onode,XMLString.TEXT_NAME); String sStyleName = Misc.getAttribute(onode,XMLString.TEXT_STYLE_NAME); Element div = converter.createElement("div"); hnode.appendChild(div); converter.addTarget(div,sName+"|region"); StyleInfo sectionInfo = new StyleInfo(); getSectionSc().applyStyle(sStyleName,sectionInfo); applyStyle(sectionInfo,div); sections.push(onode); Node newhnode = traverseBlockText(onode, div); sections.pop(); return newhnode.getParentNode(); } private void handleHeading(Node onode, Node hnode, boolean bAfterSplit) { int nListLevel = getOutlineLevel((Element)onode); boolean bUnNumbered = "true".equals(Misc.getAttribute(onode,XMLString.TEXT_IS_LIST_HEADER)); boolean bRestart = "true".equals(Misc.getAttribute(onode,XMLString.TEXT_RESTART_NUMBERING)); int nStartValue = Misc.getPosInteger(Misc.getAttribute(onode,XMLString.TEXT_START_VALUE),1)-1; handleHeading(onode, hnode, bAfterSplit, ofr.getOutlineStyle(), nListLevel, bUnNumbered, bRestart, nStartValue); } /* * Process a text:h tag */ private void handleHeading(Node onode, Node hnode, boolean bAfterSplit, ListStyle listStyle, int nListLevel, boolean bUnNumbered, boolean bRestart, int nStartValue) { // Note: nListLevel may in theory be different from the outline level, // though the ui in OOo does not allow this // Numbering: It is possible to define outline numbering in CSS2 // using counters; but this is not supported by Mozilla 1.0. // TODO: Offer CSS2 solution as an alternative later. // Note: Conditional styles are not supported int nLevel = getOutlineLevel((Element)onode); if (nLevel<=6) { if (nLevel==1) { currentChapter = (Element) onode; } // If split output, add headings of higher levels if (bAfterSplit && nSplit>0) { int nFirst = nLevel-nRepeatLevels; if (nFirst<0) { nFirst=0; } for (int i=nFirst; i0) { Element span = converter.createElement("span"); StyleInfo info = new StyleInfo(); info.sClass = "SectionNumber"; if (listStyle!=null) { String sTextStyleName = listStyle.getLevelProperty( nListLevel,XMLString.TEXT_STYLE_NAME); getTextSc().applyStyle(sTextStyleName, info); } getTextSc().applyStyle(info, span); heading.appendChild(span); span.appendChild( converter.createTextNode(sLabel) ); } // Add to toc if (!bInToc) { converter.addTarget(heading,"toc"+(++nTocIndex)); // Add in external content if (nLevel<=nSplit) { converter.addContentEntry(sLabel+(sLabel.length()>0 ? " " : "")+Misc.getPCDATA(onode), nLevel, null); } // Add to real toc TocEntry entry = new TocEntry(); entry.onode = (Element) onode; entry.sLabel = sLabel; entry.nFileIndex = converter.getOutFileIndex(); entry.nOutlineLevel = nLevel; entry.nOutlineNumber = naturalOutline.step(nLevel).getValues(); tocEntries.add(entry); } traverseInlineText(onode,heading); // Keep track of current headings for split output currentHeading[nLevel] = heading; for (int i=nLevel+1; i<=6; i++) { currentHeading[i] = null; } } else { // beyond h6 - export as ordinary paragraph handleParagraph(onode,hnode); } } /* * Process a text:p tag */ private void handleParagraph(Node onode, Node hnode) { boolean bIsEmpty = OfficeReader.isWhitespaceContent(onode); if (config.ignoreEmptyParagraphs() && bIsEmpty) { return; } String sStyleName = Misc.getAttribute(onode,XMLString.TEXT_STYLE_NAME); Element par; if (ofr.isSpreadsheet()) { // attach inline text directly to parent (always a table cell) par = (Element) hnode; } else { // Hack because createParagraph doesn't work the way we need here :-( Element temp = converter.createElement("temp"); par = createParagraph(temp, sStyleName); prependAsapNode(par); traverseFloats(onode,hnode,par); hnode.appendChild(temp.getFirstChild()); } // Maybe add to toc if (ofr.isIndexSourceStyle(getParSc().getRealParStyleName(sStyleName))) { converter.addTarget(par,"toc"+(++nTocIndex)); TocEntry entry = new TocEntry(); entry.onode = (Element) onode; entry.sLabel = sCurrentListLabel; entry.nFileIndex = converter.getOutFileIndex(); tocEntries.add(entry); } sCurrentListLabel = null; if (!bIsEmpty) { par = createTextBackground(par, sStyleName); traverseInlineText(onode,par); } else { // An empty paragraph (this includes paragraphs that only contains // whitespace) is ignored by the browser, hence we add   par.appendChild( converter.createTextNode("\u00A0") ); } } private void prependAsapNode(Node node) { if (asapNode!=null) { // May float past a split; check this first if (asapNode.getOwnerDocument()!=node.getOwnerDocument()) { asapNode = converter.importNode(asapNode,true); } node.appendChild(asapNode); asapNode = null; } } /////////////////////////////////////////////////////////////////////////// // LISTS /////////////////////////////////////////////////////////////////////////// // Helper: Get a list counter for a list style private ListCounter getListCounter(ListStyle style) { if (style==ofr.getOutlineStyle()) { // Outline numbering has a special counter return outlineNumbering; } else if (style!=null) { // Get existing or create new counter if (listCounters.containsKey(style.getName())) { return listCounters.get(style.getName()); } else { ListCounter counter = new ListCounter(style); listCounters.put(style.getName(),counter); return counter; } } else { // No style, return a dummy return new ListCounter(); } } // Helper: Check if a list contains any items private boolean hasItems(Node onode) { Node child = onode.getFirstChild(); while (child!=null) { if (Misc.isElement(child,XMLString.TEXT_LIST_ITEM) || Misc.isElement(child,XMLString.TEXT_LIST_HEADER)) { return true; } child = child.getNextSibling(); } return false; } // TODO: Merge these three methods /* * Process a text:ordered-list tag. */ private void handleOL (Node onode, int nLevel, String sStyleName, Node hnode) { if (hasItems(onode)) { // add an OL element Element list = converter.createElement("ol"); StyleInfo listInfo = new StyleInfo(); getListSc().applyStyle(nLevel,sStyleName,listInfo); applyStyle(listInfo,list); hnode.appendChild(list); traverseList(onode,nLevel,sStyleName,list); } } /* * Process a text:unordered-list tag. */ private void handleUL (Node onode, int nLevel, String sStyleName, Node hnode) { if (hasItems(onode)) { // add an UL element Element list = converter.createElement("ul"); StyleInfo listInfo = new StyleInfo(); getListSc().applyStyle(nLevel,sStyleName,listInfo); applyStyle(listInfo,list); hnode.appendChild(list); traverseList(onode,nLevel,sStyleName,list); } } private void handleList(Node onode, int nLevel, String sStyleName, Node hnode) { // In OpenDocument, we should use the style to determine the type of list String sStyleName1 = Misc.getAttribute(onode,XMLString.TEXT_STYLE_NAME); if (sStyleName1!=null) { sStyleName = sStyleName1; } ListStyle style = ofr.getListStyle(sStyleName); if (style!=null && style.isNumber(nLevel)) { handleOL(onode,nLevel,sStyleName,hnode); } else { handleUL(onode,nLevel,sStyleName,hnode); } } /* * Process the contents of a list (Changed as suggested by Nick Bower) * The option xhtml_use_list_hack triggers some *invalid* code: * - the attribute start on ol (is valid in html 4 transitional) * - the attribute value on li (is valid in html 4 transitional) * (these attributes are supposed to be replaced by css, but browsers * generally don't support that) * - generates
    1. ...
instead of *
    1. ...
in case the first child of * a list item is a new list. This occurs when a list is *continued* at * level 2 or higher. This hack seems to be the only solution that * actually produces correct results in browsers :-( */ private void traverseList (Node onode, int nLevel, String styleName, Element hnode) { ListCounter counter = getListCounter(ofr.getListStyle(styleName)); // Restart numbering, if required if (counter!=null) { boolean bContinueNumbering = "true".equals(Misc.getAttribute(onode,XMLString.TEXT_CONTINUE_NUMBERING)); if (!bContinueNumbering && counter!=null) { counter.restart(nLevel); } if (config.xhtmlUseListHack() && counter.getValue(nLevel)>0) { hnode.setAttribute("start",Integer.toString(counter.getValue(nLevel)+1)); } } if (onode.hasChildNodes()) { NodeList nList = onode.getChildNodes(); int len = nList.getLength(); for (int i = 0; i < len; i++) { Node child = nList.item(i); if (child.getNodeType() == Node.ELEMENT_NODE) { String nodeName = child.getNodeName(); if (nodeName.equals(XMLString.TEXT_LIST_ITEM)) { // Check to see if first child is a new list boolean bIsImmediateNestedList = false; Element child1 = Misc.getFirstChildElement(child); if (child1.getTagName().equals(XMLString.TEXT_ORDERED_LIST) || // old child1.getTagName().equals(XMLString.TEXT_UNORDERED_LIST) || // old child1.getTagName().equals(XMLString.TEXT_LIST)) { // oasis bIsImmediateNestedList = true; } if (config.xhtmlUseListHack() && bIsImmediateNestedList) { traverseListItem(child,nLevel,styleName,hnode); } else { // add an li element sCurrentListLabel = counter.step(nLevel).getLabel(); Element item = converter.createElement("li"); StyleInfo info = new StyleInfo(); getPresentationSc().applyOutlineStyle(nLevel,info); applyStyle(info,item); hnode.appendChild(item); if (config.xhtmlUseListHack()) { boolean bRestart = "true".equals(Misc.getAttribute(child, XMLString.TEXT_RESTART_NUMBERING)); int nStartValue = Misc.getPosInteger(Misc.getAttribute(child, XMLString.TEXT_START_VALUE),1); if (bRestart) { item.setAttribute("value",Integer.toString(nStartValue)); if (counter!=null) { sCurrentListLabel = counter.restart(nLevel,nStartValue).getLabel(); } } } traverseListItem(child,nLevel,styleName,item); } } if (nodeName.equals(XMLString.TEXT_LIST_HEADER)) { // add an li element Element item = converter.createElement("li"); hnode.appendChild(item); item.setAttribute("style","list-style-type:none"); traverseListItem(child,nLevel,styleName,item); } } } } } /* * Process the contents of a list item * (a list header should only contain paragraphs, but we don't care) */ private void traverseListItem (Node onode, int nLevel, String styleName, Node hnode) { // First check if we have a single paragraph to be omitted // This should happen if we ignore styles and have no style-map // for the paragraph style used if (config.xhtmlFormatting()!=XhtmlConfig.CONVERT_ALL && onode.hasChildNodes()) { NodeList list = onode.getChildNodes(); int nLen = list.getLength(); int nParCount = 0; boolean bNoPTag = true; for (int i=0; i 0) { index.set(i,entryj); index.set(j,entryi); } } } // Generate the index Element table = converter.createElement("table"); table.setAttribute("style","width:100%"); div.appendChild(table); Element tr = converter.createElement("tr"); table.appendChild(tr); Element[] td = new Element[4]; for (int i=0; i<4; i++) { td[i] = converter.createElement("td"); td[i].setAttribute("style","vertical-align:top"); tr.appendChild(td[i]); } int nColEntries = nIndexIndex/4+1; int nColIndex = -1; for (int i=0; i<=nIndexIndex; i++) { if (i%nColEntries==0) { nColIndex++; } AlphabeticalEntry entry = index.get(i); Element p = createParagraph(td[nColIndex],sEntryStyleName); Element a = converter.createLink("idx"+entry.nIndex); p.appendChild(a); a.appendChild(converter.createTextNode(entry.sWord)); } } } /* * Process Bibliography */ private void handleBibliography (Node onode, Node hnode) { // Use the content, not the template // This is a temp. solution. Later we want to be able to create // hyperlinks from the bib-item to the actual entry in the bibliography, // so we have to recreate the bibliography from the template. Node body = Misc.getChildByTagName(onode,XMLString.TEXT_INDEX_BODY); if (body!=null) { Element div = converter.createElement("div"); converter.addTarget(div,"bibliography"); hnode.appendChild(div); //asapNode = converter.createTarget("bibliography"); Node title = Misc.getChildByTagName(body,XMLString.TEXT_INDEX_TITLE); if (title!=null) { traverseBlockText(title,div); } traverseBlockText(body,div); } } //////////////////////////////////////////////////////////////////////// // INLINE TEXT //////////////////////////////////////////////////////////////////////// /* Process floating frames bound to this inline text (ie. paragraph) */ private void traverseFloats(Node onode, Node hnodeBlock, Node hnodeInline) { Node child = onode.getFirstChild(); while (child!=null) { if (child.getNodeType()==Node.ELEMENT_NODE) { Element elm = (Element) child; String sTag = elm.getTagName(); if (OfficeReader.isDrawElement(elm)) { elm = getDrawCv().getRealDrawElement(elm); if (elm!=null) { String sAnchor = elm.getAttribute(XMLString.TEXT_ANCHOR_TYPE); // Convert only floating frames; text-boxes must always float if (!"as-char".equals(sAnchor)) { getDrawCv().handleDrawElement(elm,(Element)hnodeBlock, (Element)hnodeInline,nFloatMode); } else if (XMLString.DRAW_TEXT_BOX.equals(sTag)) { getDrawCv().handleDrawElement(elm,(Element)hnodeBlock, (Element)hnodeInline,DrawConverter.INLINE); } } } else if (OfficeReader.isTextElement(elm)) { // Do not descend into {foot|end}notes if (!OfficeReader.isNoteElement(elm)) { traverseFloats(elm,hnodeBlock,hnodeInline); } } } child = child.getNextSibling(); } } /* * Process inline text */ private void traverseInlineText (Node onode,Node hnode) { //String styleName = Misc.getAttribute(onode, XMLString.TEXT_STYLE_NAME); if (onode.hasChildNodes()) { NodeList nList = onode.getChildNodes(); int nLen = nList.getLength(); for (int i = 0; i < nLen; i++) { Node child = nList.item(i); short nodeType = child.getNodeType(); switch (nodeType) { case Node.TEXT_NODE: String s = child.getNodeValue(); if (s.length() > 0) { hnode.appendChild( converter.createTextNode(s) ); } break; case Node.ELEMENT_NODE: String sName = child.getNodeName(); if (OfficeReader.isDrawElement(child)) { Element elm = getDrawCv().getRealDrawElement((Element)child); if (elm!=null) { String sAnchor = (elm.getAttribute(XMLString.TEXT_ANCHOR_TYPE)); if ("as-char".equals(sAnchor)) { getDrawCv().handleDrawElement(elm,null,(Element)hnode,DrawConverter.INLINE); } } } else if (child.getNodeName().equals(XMLString.TEXT_S)) { if (config.ignoreDoubleSpaces()) { hnode.appendChild( converter.createTextNode(" ") ); } else { int count= Misc.getPosInteger(Misc.getAttribute(child,XMLString.TEXT_C),1); for ( ; count > 0; count--) { hnode.appendChild( converter.createTextNode("\u00A0") ); } } } else if (sName.equals(XMLString.TEXT_TAB_STOP)) { handleTabStop(child,hnode); } else if (sName.equals(XMLString.TEXT_TAB)) { // oasis handleTabStop(child,hnode); } else if (sName.equals(XMLString.TEXT_LINE_BREAK)) { if (!config.ignoreHardLineBreaks()) { hnode.appendChild( converter.createElement("br") ); } } else if (sName.equals(XMLString.TEXT_SPAN)) { handleSpan(child,hnode); } else if (sName.equals(XMLString.TEXT_A)) { handleAnchor(child,hnode); } else if (sName.equals(XMLString.TEXT_FOOTNOTE)) { handleFootnote(child,hnode); } else if (sName.equals(XMLString.TEXT_ENDNOTE)) { handleEndnote(child,hnode); } else if (sName.equals(XMLString.TEXT_NOTE)) { // oasis if ("endnote".equals(Misc.getAttribute(child,XMLString.TEXT_NOTE_CLASS))) { handleEndnote(child,hnode); } else { handleFootnote(child,hnode); } } else if (sName.equals(XMLString.TEXT_SEQUENCE)) { handleSequence(child,hnode); } else if (sName.equals(XMLString.TEXT_PAGE_NUMBER)) { handlePageNumber(child,hnode); } else if (sName.equals(XMLString.TEXT_PAGE_COUNT)) { handlePageCount(child,hnode); } else if (sName.equals(XMLString.TEXT_SEQUENCE_REF)) { handleSequenceRef(child,hnode); } else if (sName.equals(XMLString.TEXT_FOOTNOTE_REF)) { handleNoteRef(child,hnode); } else if (sName.equals(XMLString.TEXT_ENDNOTE_REF)) { handleNoteRef(child,hnode); } else if (sName.equals(XMLString.TEXT_NOTE_REF)) { // oasis handleNoteRef(child,hnode); } else if (sName.equals(XMLString.TEXT_REFERENCE_MARK)) { handleReferenceMark(child,hnode); } else if (sName.equals(XMLString.TEXT_REFERENCE_MARK_START)) { handleReferenceMark(child,hnode); } else if (sName.equals(XMLString.TEXT_REFERENCE_REF)) { handleReferenceRef(child,hnode); } else if (sName.equals(XMLString.TEXT_BOOKMARK)) { handleBookmark(child,hnode); } else if (sName.equals(XMLString.TEXT_BOOKMARK_START)) { handleBookmark(child,hnode); } else if (sName.equals(XMLString.TEXT_BOOKMARK_REF)) { handleBookmarkRef(child,hnode); } else if (sName.equals(XMLString.TEXT_ALPHABETICAL_INDEX_MARK)) { handleAlphabeticalIndexMark(child,hnode); } else if (sName.equals(XMLString.TEXT_ALPHABETICAL_INDEX_MARK_START)) { handleAlphabeticalIndexMarkStart(child,hnode); } else if (sName.equals(XMLString.TEXT_TOC_MARK)) { handleTocMark(child,hnode); } else if (sName.equals(XMLString.TEXT_TOC_MARK_START)) { handleTocMark(child,hnode); } else if (sName.equals(XMLString.TEXT_BIBLIOGRAPHY_MARK)) { handleBibliographyMark(child,hnode); } else if (sName.equals(XMLString.OFFICE_ANNOTATION)) { converter.handleOfficeAnnotation(child,hnode); } else if (sName.startsWith("text:")) { traverseInlineText(child,hnode); } // other tags are ignored; break; default: // Do nothing } } } } private void handleTabStop(Node onode, Node hnode) { // xhtml does not have tab stops, but we export and ASCII TAB character, which the // user may choose to format if (config.getXhtmlTabstopStyle().length()>0) { Element span = converter.createElement("span"); hnode.appendChild(span); span.setAttribute("class",config.getXhtmlTabstopStyle()); span.appendChild(converter.createTextNode("\t")); } else { hnode.appendChild(converter.createTextNode("\t")); } } private void handleSpan(Node onode, Node hnode) { if (!bInToc) { String sStyleName = Misc.getAttribute(onode,XMLString.TEXT_STYLE_NAME); Element span = createInline((Element) hnode,sStyleName); traverseInlineText(onode,span); } else { traverseInlineText(onode,hnode); } } private void traversePCDATA(Node onode, Node hnode) { if (onode.hasChildNodes()) { NodeList nl = onode.getChildNodes(); int nLen = nl.getLength(); for (int i=0; i0 && n>0) { hnode = converter.nextOutFile(); } for (int i=0; i0) { Element span = converter.createElement("span"); span.setAttribute("style",sBack); node.appendChild(span); return span; } else { return node; } } /* Create a styled inline node */ private Element createInline(Element node, String sStyleName) { StyleInfo info = new StyleInfo(); getTextSc().applyStyle(sStyleName,info); Element newNode = node; if (info.hasAttributes() || !"span".equals(info.sTagName)) { // We need to create a new element newNode = converter.createElement(info.sTagName); node.appendChild(newNode); applyStyle(info,newNode); } return applyAttributes(newNode,ofr.getTextStyle(sStyleName)); } private int getOutlineLevel(Element node) { return ofr.isOpenDocument() ? Misc.getPosInteger(node.getAttribute(XMLString.TEXT_OUTLINE_LEVEL),1): Misc.getPosInteger(node.getAttribute(XMLString.TEXT_LEVEL),1); } }