/************************************************************************
*
* DrawConverter.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-2012 by Henrik Just
*
* All Rights Reserved.
*
* Version 1.4 (2012-04-07)
*
*/
/* TODO (impress2xhtml)
* Support master page content!
* New option: xhtml_draw_scaling: scale all draw objects this percentage
* (applies to applySize in this class + page size in PageStyleConverter)
* Certain options should have a fixed value for impress2xhtml:
* original_image_size: always false
* xhtml_formatting: always "convert_all"
* xhtml_frame_formatting: always "convert_all"
* xhtml_use_list_hack: always "true" (until list merge is fixed..)
* apply hard draw page background (see below)
* apply z-order for draw objects (frames)
* export notes (part of draw-page)
* export list-style-image for image bullets!
*/
package writer2latex.xhtml;
import java.util.Iterator;
import java.util.Vector;
import java.io.ByteArrayInputStream;
import java.io.IOException;
import org.xml.sax.SAXException;
import org.w3c.dom.Document;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
import org.w3c.dom.Element;
//import writer2latex.xmerge.EmbeddedBinaryObject;
import writer2latex.util.Misc;
import writer2latex.util.CSVList;
import writer2latex.util.SimpleXMLParser;
import writer2latex.xmerge.BinaryGraphicsDocument;
import writer2latex.office.EmbeddedObject;
import writer2latex.office.EmbeddedXMLObject;
import writer2latex.office.XMLString;
import writer2latex.office.MIMETypes;
import writer2latex.office.StyleWithProperties;
import writer2latex.office.FormReader;
import writer2latex.office.ControlReader;
//import writer2latex.office.MasterPage;
//import writer2latex.office.PageLayout;
import writer2latex.office.OfficeReader;
//import writer2latex.xhtml.XhtmlStyleMap;
public class DrawConverter extends ConverterHelper {
/** Identifies objects that should be displayed inline.
*/
public static final int INLINE = 0;
/** Identifies objects that should be displayed as floats, either alone
* or with text wrap (using the css attribute float:left or float:right)
*/
public static final int FLOATING = 1;
/** Identifies objects that should be positioned absolute (using the css
* attribute postion:absolute)
*/
public static final int ABSOLUTE = 2;
/** Identifies objects that should be placed centered */
public static final int CENTERED = 3;
/** Identifies objects that should fill the entire screen */
public static final int FULL_SCREEN = 4;
private FormReader form = null;
private String sScale;
private boolean bConvertToPx;
private int nImageSize;
private String sImageSplit;
private boolean bCoverImage;
private boolean bUseSVG;
// Frames in spreadsheet documents are collected here
private Vector
A draw element with a hyperlink is represented as two elements,
* eg. <draw:a><draw:image/></draw:a>
.
* We thus need methods to switch between the two elements.
This method takes a draw
-element.
* If this element is a hyperlink, the child element is returned.
* Otherwise the argument is returned unchanged.
draw:a
element
* @return the corresponding element
*/
public Element getRealDrawElement(Element onode) {
if (XMLString.DRAW_A.equals(onode.getTagName())) {
Node child = onode.getFirstChild();
while (child!=null) {
if (OfficeReader.isDrawElement(child)) { return (Element) child; }
child = child.getNextSibling();
}
return null; // empty anchor
}
return onode;
}
/** A draw element with a hyperlink is represented as two elements,
* eg. <draw:a><draw:image/></draw:a>
.
* We thus need methods to switch between the two elements.
This method takes a draw
-element.
* If this element is contained in a hyperlink, the hyperlink is returned.
* Otherwise null is returned.
draw:a
element
* @return the hyperlink element, if any
*/
public Element getDrawAnchor(Element onode) {
Element parent = (Element) onode.getParentNode();
// in oasis format, we need to skip the frame as well
if (XMLString.DRAW_FRAME.equals(parent.getTagName())) {
parent = (Element) parent.getParentNode();
}
if (XMLString.DRAW_A.equals(parent.getTagName())) { return parent; }
return null;
}
private Element getFrame(Element onode) {
if (ofr.isOpenDocument()) return (Element) onode.getParentNode();
else return onode;
}
// Add cover image (the first image in the document is taken out of context)
public Element insertCoverImage(Element hnode) {
Element currentNode = hnode;
if (bCoverImage) {
Element cover = ofr.getFirstImage();
if (cover!=null) {
converter.setCoverFile(null);
bCollectFullscreenFrames = false;
handleDrawElement(cover,currentNode,null,FULL_SCREEN);
bCollectFullscreenFrames = true;
// Add margin:0 to body
Element head = Misc.getChildByTagName(hnode.getOwnerDocument().getDocumentElement(),"head");
if (head!=null) {
Element style = converter.createElement("style");
head.appendChild(style);
style.setAttribute("type", "text/css");
style.appendChild(converter.createTextNode("body { margin:0 }"));
}
currentNode = getTextCv().doMaybeSplit(hnode, 0);
}
}
return currentNode;
}
// Flush all full screen images, returning the new document node
public Element flushFullscreenFrames(Element hnode) {
Element currentNode = hnode;
if (converter.isTopLevel() && !fullscreenFrames.isEmpty()) {
bCollectFullscreenFrames = false;
currentNode = getTextCv().doMaybeSplit(hnode, 0);
for (Element image : fullscreenFrames) {
handleDrawElement(image,currentNode,null,FULL_SCREEN);
currentNode = getTextCv().doMaybeSplit(hnode, 0);
}
fullscreenFrames.clear();
bCollectFullscreenFrames = true;
}
return currentNode;
}
public void flushFrames(Element hnode) {
bCollectFrames = false;
int nCount = frames.size();
for (int i=0; inMode
:
* DrawConverter.INLINE
: Presented inline. The hnode
* must accept inline content. An inline container must be
* provided.DrawConverter.FLOAT
: Presented as a float. The hnode
* must accept block/flow content. A block container must be
* provided.DrawConverter.ABSOLUTE
: Presented at an absolute
* position. A block container must be provided.Containers for block and inline elements should be supplied. * The containers may be identical (flow container).
*Note: A draw:text-box will be ignored in inline mode.
* @param onode the draw element * @param hnodeBlock the xhtml element to attach the converted element to if it's a block element * @param hnodeInline the xhtml element to attach the converted element to if it's an inline element * @param nMode identifies how the element should be presented */ public void handleDrawElement(Element onode, Element hnodeBlock, Element hnodeInline, int nMode) { if (bCollectFrames) { frames.add(onode); return; } String sName = onode.getNodeName(); if (sName.equals(XMLString.DRAW_OBJECT)) { handleDrawObject(onode,hnodeBlock,hnodeInline,nMode); } else if (sName.equals(XMLString.DRAW_OBJECT_OLE)) { handleDrawObject(onode,hnodeBlock,hnodeInline,nMode); } else if (sName.equals(XMLString.DRAW_IMAGE)) { handleDrawImage(onode,hnodeBlock,hnodeInline,nMode); } else if (sName.equals(XMLString.DRAW_TEXT_BOX)) { handleDrawTextBox(onode,hnodeBlock,hnodeInline,nMode); } else if (sName.equals(XMLString.DRAW_A)) { Element elm = getRealDrawElement(onode); if (elm!=null) { handleDrawElement(elm,hnodeBlock,hnodeInline,nMode); } } else if (sName.equals(XMLString.DRAW_FRAME)) { // OpenDocument embeds the draw element in a frame element handleDrawElement(Misc.getFirstChildElement(onode),hnodeBlock,hnodeInline,nMode); } else if (sName.equals(XMLString.DRAW_G)) { handleDrawGroup(onode,hnodeBlock,hnodeInline,nMode); } else if (sName.equals(XMLString.DRAW_CONTROL)) { handleDrawControl(onode,hnodeBlock,hnodeInline,nMode); } } private void handleDrawObject(Element onode, Element hnodeBlock, Element hnodeInline, int nMode) { // TODO: Placement if not inline // If possible, add the object inline. In pure block context, add a div. Element hnode; if (hnodeInline!=null) { hnode = hnodeInline; } else { Element div = converter.createElement("div"); hnodeBlock.appendChild(div); hnode = div; } String sHref = Misc.getAttribute(onode, XMLString.XLINK_HREF); if (sHref!=null) { // Embedded object in package or linked object if (ofr.isInPackage(sHref)) { // Embedded object in package if (sHref.startsWith("#")) { sHref=sHref.substring(1); } if (sHref.startsWith("./")) { sHref=sHref.substring(2); } EmbeddedObject object = converter.getEmbeddedObject(sHref); if (MIMETypes.MATH.equals(object.getType()) || MIMETypes.ODF.equals(object.getType())) { // Formula! EmbeddedXMLObject xmlObject = (EmbeddedXMLObject) object; // Document settings = object.getSettingsDOM(); Element replacementImage = null; if (ofr.isOpenDocument()) { // look for replacement image replacementImage = Misc.getChildByTagName(getFrame(onode),XMLString.DRAW_IMAGE); } try { hnode.appendChild(converter.createTextNode(" ")); getMathCv().convert(replacementImage,xmlObject.getContentDOM().getDocumentElement(),hnode); hnode.appendChild(converter.createTextNode(" ")); } catch (SAXException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } else { // unsupported object boolean bIgnore = true; if (ofr.isOpenDocument()) { // look for replacement image Element replacementImage = Misc.getChildByTagName(getFrame(onode),XMLString.DRAW_IMAGE); if (replacementImage!=null) { handleDrawImage(replacementImage,hnodeBlock,hnodeInline,nMode); bIgnore = false; } } if (bIgnore) { hnode.appendChild( converter.createTextNode("[Warning: object ignored]")); } } } else { // TODO: Linked object hnode.appendChild( converter.createTextNode("[Warning: Linked object ignored]")); } } else { // flat xml format Element formula = Misc.getChildByTagName(onode,XMLString.MATH); // Since OOo 3.2 if (formula==null) { formula = Misc.getChildByTagName(onode,XMLString.MATH_MATH); } if (formula != null) { Element replacementImage = null; if (ofr.isOpenDocument()) { // look for replacement image replacementImage = Misc.getChildByTagName(getFrame(onode),XMLString.DRAW_IMAGE); } hnode.appendChild(converter.createTextNode(" ")); getMathCv().convert(replacementImage,formula,hnode); hnode.appendChild(converter.createTextNode(" ")); } else { // unsupported object boolean bIgnore = true; if (ofr.isOpenDocument()) { // look for replacement image Element replacementImage = Misc.getChildByTagName(getFrame(onode),XMLString.DRAW_IMAGE); if (replacementImage!=null) { handleDrawImage(replacementImage,hnodeBlock,hnodeInline,nMode); bIgnore = false; } } if (bIgnore) { hnode.appendChild( converter.createTextNode("[Warning: object ignored]")); } } } } private void handleDrawImage(Element onode, Element hnodeBlock, Element hnodeInline, int nMode) { Element frame = getFrame(onode); // For EPUB document some images require special treatment: Cover image and full screen images if (bCollectFullscreenFrames && converter.isOPS()) { // First check whether this is the cover image if (bCoverImage && onode==ofr.getFirstImage()) { return; } // Next check to see if we should treat this image as a "full screen" image // (Currently only images are handled like this, hence the code is here rather than in handleDrawElement) else if (!"none".equals(sImageSplit) && converter.isTopLevel()) { StyleWithProperties style = ofr.getFrameStyle(frame.getAttribute(XMLString.DRAW_STYLE_NAME)); String sWidth = getFrameWidth(frame, style); String sHeight = getFrameHeight(frame, style); // It is if the image width exceeds a certain percentage of the current text width and the height is // greater than 1.33*the width (recommended by Michel "Coolmicro") if (sWidth!=null && sHeight!=null && Misc.sub(Misc.multiply("133%",sWidth), sHeight).startsWith("-") && Misc.sub(Misc.multiply(sImageSplit,converter.getContentWidth()), Misc.multiply(sScale,Misc.truncateLength(sWidth))).startsWith("-")) { fullscreenFrames.add(onode); return; } } } // Get the image from the ImageLoader BinaryGraphicsDocument bgd = null; String sFileName = null; String sHref = Misc.getAttribute(onode,XMLString.XLINK_HREF); if (sHref!=null && sHref.length()>0 && !ofr.isInPackage(sHref)) { // Linked image is not yet handled by ImageLoader. This is a temp. // solution (will go away when ImageLoader is finished) if (!converter.isOPS()) { // Cannot have linked images in EPUB, ignore the image sFileName = sHref; // In OpenDocument *package* format ../ means "leave the package" if (ofr.isOpenDocument() && ofr.isPackageFormat() && sFileName.startsWith("../")) { sFileName=sFileName.substring(3); } //String sExt = sHref.substring(sHref.lastIndexOf(".")).toLowerCase(); } } else { // embedded or base64 encoded image bgd = converter.getImageLoader().getImage(onode); if (bgd!=null) { sFileName = bgd.getFileName(); // If this is the cover image, add it to the converter result if (bCoverImage && onode==ofr.getFirstImage()) { converter.setCoverImageFile(bgd,null); } } } if (sFileName==null) { return; } // TODO: Add warning? // Create the image (sFileName contains the file name) Element imageElement = null; if (converter.nType==XhtmlDocument.HTML5 && bUseSVG && bgd!=null && MIMETypes.SVG.equals(bgd.getMIMEType())) { // In HTML5 we may embed SVG images directly in the document byte[] blob = bgd.getData(); try { Document dom = SimpleXMLParser.parse(new ByteArrayInputStream(blob)); if (dom!=null) { Element elm = hnodeInline!=null ? hnodeInline : hnodeBlock; imageElement = (Element) elm.getOwnerDocument().importNode(dom.getDocumentElement(), true); } else { System.out.println("Failed to parse SVG"); } } catch (IOException e) { // Will not happen with a byte array System.out.println("IOException parsing SVG"); e.printStackTrace(); } catch (SAXException e) { e.printStackTrace(); System.out.println("SAXException parsing SVG"); } } else { // In all other cases, create an img element if (bgd!=null) { converter.addDocument(bgd); } Element image = converter.createElement("img"); String sName = Misc.getAttribute(getFrame(onode),XMLString.DRAW_NAME); converter.addTarget(image,sName+"|graphic"); image.setAttribute("src",sFileName); // Add alternative text, using either alt.text, name or file name Element desc = Misc.getChildByTagName(frame,XMLString.SVG_DESC); if (desc==null) { desc = Misc.getChildByTagName(frame,XMLString.SVG_TITLE); } String sAltText = desc!=null ? Misc.getPCDATA(desc) : (sName!=null ? sName : sFileName); image.setAttribute("alt",sAltText); imageElement = image; } if (imageElement!=null) { // Now style it StyleInfo info = new StyleInfo(); String sStyleName = Misc.getAttribute(frame, XMLString.DRAW_STYLE_NAME); if (nMode!=FULL_SCREEN) { getFrameSc().applyStyle(sStyleName,info); } applyImageSize(frame,info.props,nMode,false); // Apply placement applyPlacement(frame, hnodeBlock, hnodeInline, nMode, imageElement, info); applyStyle(info,imageElement); addLink(onode,imageElement); } } private void handleDrawTextBox(Element onode, Element hnodeBlock, Element hnodeInline, int nMode) { // Create the div with id=name Element textbox = converter.createElement("div"); if (hnodeBlock!=null) { hnodeBlock.appendChild(textbox); } else { // cannot include the div inline, ignore return; } // Add name, if defined String sName = Misc.getAttribute(getFrame(onode),XMLString.DRAW_NAME); if (sName!=null) { converter.addTarget(textbox,sName+"|frame"); } // Now style it Element frame = getFrame(onode); StyleInfo info = new StyleInfo(); // Draw frame style String sStyleName = Misc.getAttribute(frame, XMLString.DRAW_STYLE_NAME); if (sStyleName!=null) { getFrameSc().applyStyle(sStyleName,info); } // Presentation frame style String sPresentationStyleName = Misc.getAttribute(frame, XMLString.PRESENTATION_STYLE_NAME); if (sPresentationStyleName!=null) { if ("outline".equals(Misc.getAttribute(frame, XMLString.PRESENTATION_CLASS))) { getPresentationSc().enterOutline(sPresentationStyleName); } getPresentationSc().applyStyle(sPresentationStyleName,info); } // Additional text formatting String sTextStyleName = Misc.getAttribute(frame, XMLString.DRAW_TEXT_STYLE_NAME); if (sTextStyleName!=null) { //getStyleCv().applyParStyle(sTextStyleName,info); } // Apply placement String sContentWidth = null; switch (nMode) { case INLINE: break; case ABSOLUTE: sContentWidth = applyImageSize(frame,info.props,nMode,false); info.props.addValue("margin-left","auto"); info.props.addValue("margin-right","auto"); applyPosition(frame,info.props); break; case CENTERED: info.props.addValue("margin-top","2px"); info.props.addValue("margin-bottom","2px"); info.props.addValue("margin-left","auto"); info.props.addValue("margin-right","auto"); sContentWidth = applyImageSize(frame,info.props,nMode,true); break; case FLOATING: sContentWidth = applyImageSize(frame,info.props,nMode,true); StyleWithProperties style = ofr.getFrameStyle(sStyleName); if (style!=null) { String sPos = style.getProperty(XMLString.STYLE_HORIZONTAL_POS); String sWrap = style.getProperty(XMLString.STYLE_WRAP); if (isLeft(sPos)) { if (mayWrapRight(sWrap)) { info.props.addValue("float","left"); } else { // TODO: Remove the margin-right attribute from the existing properties // and likewise for the other two cases below (requires new CSVList) info.props.addValue("margin-right", "auto"); } } else if (isRight(sPos)) { if (mayWrapLeft(sWrap)) { info.props.addValue("float","right"); } else { info.props.addValue("margin-left", "auto"); } } else if (isCenter(sPos)) { info.props.addValue("margin-left", "auto"); info.props.addValue("margin-right", "auto"); } else if (isFromLeft(sPos)) { if (mayWrapRight(sWrap)) { info.props.addValue("float","left"); } String sX = frame.getAttribute(XMLString.SVG_X); if (sX!=null && sX.length()>0) { info.props.addValue("margin-left",scale(sX)); } } } } //Finish applyStyle(info,textbox); if (sContentWidth!=null) { converter.pushContentWidth(sContentWidth); } getTextCv().traverseBlockText(onode,textbox); if (sContentWidth!=null) { converter.popContentWidth(); } getPresentationSc().exitOutline(); } private void handleDrawGroup(Element onode, Element hnodeBlock, Element hnodeInline, int nMode) { // TODO: style-name and z-index should be transferred to children Node child = onode.getFirstChild(); while (child!=null) { if (OfficeReader.isDrawElement(child)) { handleDrawElement((Element) child, hnodeBlock, hnodeInline, nMode); } child = child.getNextSibling(); } } ////////////////////////////////////////////////////////////////////////// // Forms private void handleDrawControl(Element onode, Element hnodeBlock, Element hnodeInline, int nMode) { // Get the control, if possible if (form==null) { return; } ControlReader control = ofr.isOpenDocument() ? ofr.getForms().getControl(Misc.getAttribute(onode,XMLString.DRAW_CONTROL)) : ofr.getForms().getControl(Misc.getAttribute(onode,XMLString.FORM_ID)); if (control==null || control.getOwnerForm()!=form) { return; } // Create the control element Element hcontrol = null; String sType = control.getControlType(); if (XMLString.FORM_TEXT.equals(sType)) { hcontrol = createInputText(control,false); } else if (XMLString.FORM_PASSWORD.equals(sType)) { hcontrol = createInputText(control,true); } else if (XMLString.FORM_FILE.equals(sType)) { hcontrol = createInputFile(control); } else if (XMLString.FORM_IMAGE.equals(sType)) { hcontrol = createInput(control,"image"); } else if (XMLString.FORM_HIDDEN.equals(sType)) { hcontrol = createInput(control,"hidden"); } else if (XMLString.FORM_CHECKBOX.equals(sType)) { hcontrol = createInputCheck(control,false); } else if (XMLString.FORM_RADIO.equals(sType)) { hcontrol = createInputCheck(control,true); } else if (XMLString.FORM_BUTTON.equals(sType)) { hcontrol = createInputButton(control); } else if (XMLString.FORM_FIXED_TEXT.equals(sType)) { hcontrol = createLabel(control); } else if (XMLString.FORM_TEXTAREA.equals(sType)) { hcontrol = createTextarea(control); } else if (XMLString.FORM_LISTBOX.equals(sType)) { hcontrol = createSelect(control); } // ignore other controls if (hcontrol!=null) { Element frame = onode; // controls are *not* contained in a draw:frame! StyleInfo info = new StyleInfo(); getFrameSc().applyStyle(frame.getAttribute(XMLString.DRAW_STYLE_NAME),info); applySize(frame,info.props,false); applyPlacement(frame,hnodeBlock,hnodeInline,nMode,hcontrol,info); applyStyle(info,hcontrol); } } private Element createInput(ControlReader control, String sType) { // Create the element Element input = converter.createElement("input"); input.setAttribute("type",sType); return input; } private Element createInputFile(ControlReader control) { Element input = converter.createElement("input"); input.setAttribute("type","file"); setCommonAttributes(control,input); setDisabled(control,input); setReadonly(control,input); setValue(control,input); return input; } private Element createInputText(ControlReader control, boolean bPassword) { Element input = converter.createElement("input"); input.setAttribute("type",bPassword ? "password" : "text"); setCommonAttributes(control,input); setName(control,input,true); setValue(control,input); setMaxLength(control,input); setDisabled(control,input); setReadonly(control,input); return input; } private Element createInputCheck(ControlReader control, boolean bRadio) { Element input = converter.createElement("input"); input.setAttribute("type",bRadio ? "radio" : "checkbox"); setCommonAttributes(control,input); setName(control,input,true); setValue(control,input); setChecked(control,input); setDisabled(control,input); setReadonly(control,input); // Add a label for the check/radio Element label = converter.createElement("label"); setFor(control,label); label.appendChild(input); label.appendChild(converter.createTextNode(control.getTypeAttribute(XMLString.FORM_LABEL))); return label; } private Element createInputButton(ControlReader control) { Element input = converter.createElement("input"); String sButtonType = control.getTypeAttribute(XMLString.FORM_BUTTON_TYPE); if ("submit".equals(sButtonType)) { input.setAttribute("type","submit"); } else if ("reset".equals(sButtonType)) { input.setAttribute("type","reset"); } else { // TODO: url button (using javascript) input.setAttribute("type","button"); } setCommonAttributes(control,input); setName(control,input,true); input.setAttribute("value",control.getTypeAttribute(XMLString.FORM_LABEL)); setDisabled(control,input); return input; } private Element createLabel(ControlReader control) { Element label = converter.createElement("label"); setCommonAttributes(control,label); setFor(control,label); label.setAttribute("value",control.getTypeAttribute(XMLString.FORM_LABEL)); label.appendChild(converter.createTextNode(control.getTypeAttribute(XMLString.FORM_LABEL))); return label; } private Element createTextarea(ControlReader control) { Element textarea = converter.createElement("textarea"); setCommonAttributes(control,textarea); setName(control,textarea,true); setDisabled(control,textarea); setReadonly(control,textarea); // rows & cols are required - but css will override them! textarea.setAttribute("rows","10"); textarea.setAttribute("cols","5"); // The value attribute should be used as content String s = control.getTypeAttribute(XMLString.FORM_VALUE); if (s!=null) { textarea.appendChild(converter.createTextNode(s)); } return textarea; } private Element createSelect(ControlReader control) { Element select = converter.createElement("select"); setCommonAttributes(control,select); setName(control,select,false); setSize(control,select); setMultiple(control,select); setDisabled(control,select); // Add options int nCount = control.getItemCount(); for (int i=0; i