diff --git a/src/main/java/writer2latex/xhtml/Converter.java b/src/main/java/writer2latex/xhtml/Converter.java index b79fdb9..10c8397 100644 --- a/src/main/java/writer2latex/xhtml/Converter.java +++ b/src/main/java/writer2latex/xhtml/Converter.java @@ -57,6 +57,7 @@ import writer2latex.office.XMLString; import writer2latex.util.ExportNameCollection; import writer2latex.util.Misc; import writer2latex.xhtml.content.DrawParser; +import writer2latex.xhtml.content.ListParser; import writer2latex.xhtml.content.MathParser; import writer2latex.xhtml.content.PageContainer; import writer2latex.xhtml.content.TableParser; @@ -86,6 +87,7 @@ public class Converter extends BasicConverter { // The helpers private Styles styles; private TextParser textParser; + private ListParser listParser; private TableParser tableParser; private DrawParser drawParser; private MathParser mathParser; @@ -192,6 +194,8 @@ public class Converter extends BasicConverter { public Styles getStylesParser() { return styles; } public TextParser getTextParser() { return textParser; } + + public ListParser getListParser() { return listParser; } public TableParser getTableParser() { return tableParser; } @@ -276,6 +280,7 @@ public class Converter extends BasicConverter { styles = new Styles(ofr,config,this,nType); textParser = new TextParser(ofr,config,this); + listParser = new ListParser(ofr,config,this); tableParser = new TableParser(ofr,config,this); drawParser = new DrawParser(ofr,config,this); mathParser = new MathParser(ofr,config,this,nType!=XhtmlDocument.XHTML10 && nType!=XhtmlDocument.XHTML11); @@ -837,6 +842,8 @@ public class Converter extends BasicConverter { meta.setAttribute("content",sValue); head.appendChild(meta); } + + } \ No newline at end of file diff --git a/src/main/java/writer2latex/xhtml/Parser.java b/src/main/java/writer2latex/xhtml/Parser.java index 1d2482b..bdc8728 100644 --- a/src/main/java/writer2latex/xhtml/Parser.java +++ b/src/main/java/writer2latex/xhtml/Parser.java @@ -29,6 +29,7 @@ import org.w3c.dom.Element; import writer2latex.office.OfficeReader; import writer2latex.xhtml.content.DrawParser; +import writer2latex.xhtml.content.ListParser; import writer2latex.xhtml.content.MathParser; import writer2latex.xhtml.content.TableParser; import writer2latex.xhtml.content.TextParser; @@ -93,6 +94,9 @@ public class Parser { protected PageStyleParser getPageSc() { return converter.getStylesParser().getPageSc(); } protected TextParser getTextCv() { return converter.getTextParser(); } + + protected ListParser getListParser() { return converter.getListParser(); } + protected TableParser getTableCv() { return converter.getTableParser(); } diff --git a/src/main/java/writer2latex/xhtml/content/ListParser.java b/src/main/java/writer2latex/xhtml/content/ListParser.java new file mode 100644 index 0000000..d4c00a5 --- /dev/null +++ b/src/main/java/writer2latex/xhtml/content/ListParser.java @@ -0,0 +1,369 @@ +package writer2latex.xhtml.content; + +import static writer2latex.office.XMLString.TEXT_CONTINUE_NUMBERING; +import static writer2latex.office.XMLString.TEXT_H; +import static writer2latex.office.XMLString.TEXT_LIST; +import static writer2latex.office.XMLString.TEXT_LIST_HEADER; +import static writer2latex.office.XMLString.TEXT_LIST_ITEM; +import static writer2latex.office.XMLString.TEXT_ORDERED_LIST; +import static writer2latex.office.XMLString.TEXT_P; +import static writer2latex.office.XMLString.TEXT_RESTART_NUMBERING; +import static writer2latex.office.XMLString.TEXT_START_VALUE; +import static writer2latex.office.XMLString.TEXT_STYLE_NAME; +import static writer2latex.office.XMLString.TEXT_UNORDERED_LIST; + +import java.util.Hashtable; + +import org.w3c.dom.Element; +import org.w3c.dom.Node; +import org.w3c.dom.NodeList; + +import writer2latex.office.ListCounter; +import writer2latex.office.ListStyle; +import writer2latex.office.OfficeReader; +import writer2latex.util.Misc; +import writer2latex.xhtml.Converter; +import writer2latex.xhtml.Parser; +import writer2latex.xhtml.StyleInfo; +import writer2latex.xhtml.XhtmlConfig; + +public class ListParser extends Parser { + + private Hashtable listCounters = new Hashtable(); + protected String sCurrentListLabel = null; + protected ListStyle currentListStyle = null; + protected int nCurrentListLevel = 0; + ListCounter outlineNumbering; + + + public ListParser(OfficeReader ofr, XhtmlConfig config, Converter converter) { + super(ofr, config, converter); + outlineNumbering = new ListCounter(ofr.getOutlineStyle()); + } + + protected 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(); + } + } + + protected void insertListLabel(ListStyle style, int nLevel, String sDefaultStyle, String sPrefix, String sLabel, Element hnode) { + if (sLabel!=null && sLabel.length()>0) { + if (sPrefix!=null) { + Element prefix = converter.createElement("span"); + prefix.setAttribute("class", "chapter-name"); + hnode.appendChild(prefix); + prefix.appendChild( converter.createTextNode(sPrefix)); + } + StyleInfo info = new StyleInfo(); + if (style!=null) { + String sTextStyleName = style.getLevelProperty(nLevel,TEXT_STYLE_NAME); + getTextSP().readStyle(sTextStyleName, info); + } + + if (info.sTagName==null) { info.sTagName = "span"; } + if (info.sClass==null) { info.sClass = sDefaultStyle; } + + Element content = converter.createElement(info.sTagName); + getTextSP().writeStyle(info, content); + hnode.appendChild(content); + content.appendChild( converter.createTextNode(sLabel) ); + } + } + + private boolean hasItems(Node onode) { + Node child = onode.getFirstChild(); + while (child!=null) { + if (Misc.isElement(child,TEXT_LIST_ITEM) || + Misc.isElement(child,TEXT_LIST_HEADER)) { + return true; + } + child = child.getNextSibling(); + } + return false; + } + + 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); + getListSc().readParentStyle(nLevel, sStyleName, listInfo); + writeStyle(listInfo,list); + hnode.appendChild(list); + traverseList(onode,nLevel,sStyleName,list); + } + } + + 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); + writeStyle(listInfo,list); + hnode.appendChild(list); + traverseList(onode,nLevel,sStyleName,list); + } + } + + protected 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,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); + } + } + + 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,TEXT_CONTINUE_NUMBERING)); + if (!bContinueNumbering && counter!=null) { + counter.restart(nLevel); + } + if (config.listFormatting()==XhtmlConfig.CSS1_HACK && 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(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(TEXT_ORDERED_LIST) || // old + child1.getTagName().equals(TEXT_UNORDERED_LIST) || // old + child1.getTagName().equals(TEXT_LIST)) { // oasis + bIsImmediateNestedList = true; + } + + if (config.listFormatting()==XhtmlConfig.CSS1_HACK && bIsImmediateNestedList) { + traverseListItem(child,nLevel,styleName,hnode); + } + else { + // add an li element + //if (counter!=null) { + sCurrentListLabel = counter.step(nLevel).getLabel(); + //} + currentListStyle = ofr.getListStyle(styleName); + nCurrentListLevel = nLevel; + Element item = converter.createElement("li"); + StyleInfo info = new StyleInfo(); + getPresentationSc().applyOutlineStyle(nLevel,info); + writeStyle(info,item); + hnode.appendChild(item); + if (config.listFormatting()==XhtmlConfig.CSS1_HACK) { + boolean bRestart = "true".equals(Misc.getAttribute(child, + TEXT_RESTART_NUMBERING)); + int nStartValue = Misc.getPosInteger(Misc.getAttribute(child, + 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(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); + } + } + } + } + } + + 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 splitting is forbidden - boolean bAfterHeading=false; // last element was a top level heading Element[] currentHeading = new Element[7]; // Last headings (repeated when splitting) private int nCharacterCount = 0; // The number of text characters in the current document @@ -255,9 +253,8 @@ public class TextParser extends Parser { return parseText(onode,0,null,hnode); } - private Node parseText(Node onode, int nLevel, String styleName, Node hnode) { + public Node parseText(Node onode, int nLevel, String styleName, Node hnode) { if (!onode.hasChildNodes()) { return hnode; } - bAfterHeading = false; NodeList nList = onode.getChildNodes(); int nLen = nList.getLength(); int i = 0; @@ -268,7 +265,6 @@ public class TextParser extends Parser { if (child.getNodeType() == Node.ELEMENT_NODE) { String nodeName = child.getNodeName(); // Block splitting - nDontSplitLevel++; if (OfficeReader.isDrawElement(child)) { getDrawCv().handleDrawElement((Element)child,(Element)hnode,null,nFloatMode); @@ -366,13 +362,11 @@ public class TextParser extends Parser { } hnode = processPageBreaks(child, hnode,style); inList = true; - if (listIsOnlyHeadings(child)) { - nDontSplitLevel--; - hnode = handleFakeList(child,nLevel+1,styleName,hnode); - nDontSplitLevel++; + if (getListParser().listIsOnlyHeadings(child)) { + hnode = getListParser().handleFakeList(child,nLevel+1,styleName,hnode); } else { - handleList(child,nLevel+1,styleName,hnode); + getListParser().handleList(child,nLevel+1,styleName,hnode); } inList = false; } @@ -388,9 +382,7 @@ public class TextParser extends Parser { } else if (nodeName.equals(TEXT_SECTION)) { hnode = processPageBreaks(child, hnode,null); - nDontSplitLevel--; hnode = handleSection(child,hnode); - nDontSplitLevel++; } else if (nodeName.equals(TEXT_TABLE_OF_CONTENT)) { if (!ofr.getTocReader((Element)child).isByChapter()) { @@ -424,13 +416,8 @@ public class TextParser extends Parser { else if (nodeName.equals(TEXT_SEQUENCE_DECLS)) { //handleSeqeuenceDecls(child); } - // Reenable splitting - nDontSplitLevel--; - // Remember if this was a heading - if (nDontSplitLevel==0) { - bAfterHeading = nodeName.equals(TEXT_H); - hnode = getDrawCv().flushFullscreenFrames((Element)hnode); - } + //TODO:IS IT NEEDED? + hnode = getDrawCv().flushFullscreenFrames((Element)hnode); } i++; } @@ -507,7 +494,7 @@ public class TextParser extends Parser { /* * Process a text:h tag */ - private void handleHeading(Element onode, Element hnode, boolean bAfterSplit, ListStyle listStyle, int nListLevel, boolean bUnNumbered, boolean bRestart, int nStartValue) { + public void handleHeading(Element onode, Element 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 @@ -541,7 +528,7 @@ public class TextParser extends Parser { ListCounter counter = null; String sLabel = ""; if (!bUnNumbered) { - counter = getListCounter(listStyle); + counter = getListParser().getListCounter(listStyle); if (bRestart) { counter.restart(nListLevel, nStartValue); } @@ -598,7 +585,7 @@ public class TextParser extends Parser { // Prepend numbering if (!bUnNumbered) { - insertListLabel(listStyle, nListLevel, "SectionNumber", null, sLabel, heading); + getListParser().insertListLabel(listStyle, nListLevel, "SectionNumber", null, sLabel, heading); } // Add to toc @@ -666,7 +653,7 @@ public class TextParser extends Parser { if (!bIsEmpty) { par = createTextBackground(par, styleName); if (config.listFormatting()==XhtmlConfig.HARD_LABELS) { - insertListLabel(currentListStyle, nCurrentListLevel, "ItemNumber", null, sCurrentListLabel, par); + getListParser().insertListLabel(currentListStyle, nCurrentListLevel, "ItemNumber", null, sCurrentListLabel, par); } sCurrentListLabel = null; traverseInlineText(onode,par); @@ -699,384 +686,7 @@ public class TextParser extends Parser { } - /////////////////////////////////////////////////////////////////////////// - // 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: Insert a list label formatted with a list style - private void insertListLabel(ListStyle style, int nLevel, String sDefaultStyle, String sPrefix, String sLabel, Element hnode) { - if (sLabel!=null && sLabel.length()>0) { - if (sPrefix!=null) { - Element prefix = converter.createElement("span"); - prefix.setAttribute("class", "chapter-name"); - hnode.appendChild(prefix); - prefix.appendChild( converter.createTextNode(sPrefix)); - } - StyleInfo info = new StyleInfo(); - if (style!=null) { - String sTextStyleName = style.getLevelProperty(nLevel,TEXT_STYLE_NAME); - getTextSP().readStyle(sTextStyleName, info); - } - - if (info.sTagName==null) { info.sTagName = "span"; } - if (info.sClass==null) { info.sClass = sDefaultStyle; } - - Element content = converter.createElement(info.sTagName); - getTextSP().writeStyle(info, content); - hnode.appendChild(content); - content.appendChild( converter.createTextNode(sLabel) ); - } - } - - // Helper: Check if a list contains any items - private boolean hasItems(Node onode) { - Node child = onode.getFirstChild(); - while (child!=null) { - if (Misc.isElement(child,TEXT_LIST_ITEM) || - Misc.isElement(child,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); - getListSc().readParentStyle(nLevel, sStyleName, listInfo); - writeStyle(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); - writeStyle(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,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,TEXT_CONTINUE_NUMBERING)); - if (!bContinueNumbering && counter!=null) { - counter.restart(nLevel); - } - if (config.listFormatting()==XhtmlConfig.CSS1_HACK && 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(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(TEXT_ORDERED_LIST) || // old - child1.getTagName().equals(TEXT_UNORDERED_LIST) || // old - child1.getTagName().equals(TEXT_LIST)) { // oasis - bIsImmediateNestedList = true; - } - - if (config.listFormatting()==XhtmlConfig.CSS1_HACK && bIsImmediateNestedList) { - traverseListItem(child,nLevel,styleName,hnode); - } - else { - // add an li element - //if (counter!=null) { - sCurrentListLabel = counter.step(nLevel).getLabel(); - //} - currentListStyle = ofr.getListStyle(styleName); - nCurrentListLevel = nLevel; - Element item = converter.createElement("li"); - StyleInfo info = new StyleInfo(); - getPresentationSc().applyOutlineStyle(nLevel,info); - writeStyle(info,item); - hnode.appendChild(item); - if (config.listFormatting()==XhtmlConfig.CSS1_HACK) { - boolean bRestart = "true".equals(Misc.getAttribute(child, - TEXT_RESTART_NUMBERING)); - int nStartValue = Misc.getPosInteger(Misc.getAttribute(child, - 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(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