/************************************************************************
*
* 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-2010 by Henrik Just
*
* All Rights Reserved.
*
* Version 1.2 (2010-02-19)
*
*/
/* 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.IOException;
import org.xml.sax.SAXException;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
import org.w3c.dom.Element;
import writer2latex.xmerge.EmbeddedObject;
//import writer2latex.xmerge.EmbeddedBinaryObject;
import writer2latex.xmerge.EmbeddedXMLObject;
import writer2latex.util.Misc;
import writer2latex.util.CSVList;
import writer2latex.xmerge.BinaryGraphicsDocument;
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;
private FormReader form = null;
private String sScale;
private boolean bConvertToPx;
private boolean bOriginalImageSize;
// 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;
}
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(); try { hnode.appendChild(converter.createTextNode(" ")); getMathCv().convert(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 Node formula = Misc.getChildByTagName(onode,XMLString.MATH); // Since OOo 3.2 if (formula==null) { formula = Misc.getChildByTagName(onode,XMLString.MATH_MATH); } if (formula != null) { hnode.appendChild(converter.createTextNode(" ")); getMathCv().convert(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) { // Get the image from the ImageLoader 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) 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 BinaryGraphicsDocument bgd = converter.getImageLoader().getImage(onode); if (bgd!=null) { converter.addDocument(bgd); sFileName = bgd.getFileName(); } } if (sFileName==null) { return; } // TODO: Add warning? // Create the image (sFileName contains the file name) 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 frame = getFrame(onode); 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); // Now style it StyleInfo info = new StyleInfo(); String sStyleName = Misc.getAttribute(frame, XMLString.DRAW_STYLE_NAME); getFrameSc().applyStyle(sStyleName,info); if (!bOriginalImageSize) { applySize(frame,info.props,false); } // Apply placement applyPlacement(frame, hnodeBlock, hnodeInline, nMode, image, info); applyStyle(info,image); addLink(onode,image); } 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 sStyleName = Misc.getAttribute(frame, XMLString.PRESENTATION_STYLE_NAME); if (sStyleName!=null) { if ("outline".equals(Misc.getAttribute(frame, XMLString.PRESENTATION_CLASS))) { getPresentationSc().enterOutline(sStyleName); } getPresentationSc().applyStyle(sStyleName,info); } // Additional text formatting sStyleName = Misc.getAttribute(frame, XMLString.DRAW_TEXT_STYLE_NAME); if (sStyleName!=null) { //getStyleCv().applyParStyle(sStyleName,info); } // Apply placement switch (nMode) { case INLINE: break; case ABSOLUTE: applySize(frame,info.props,false); applyPosition(frame,info.props); break; case CENTERED: info.props.addValue("maring-top","2px"); info.props.addValue("maring-bottom","2px"); info.props.addValue("margin-left","auto"); info.props.addValue("margin-right","auto"); applySize(frame,info.props,true); break; case FLOATING: applySize(frame,info.props,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) && mayWrapRight(sWrap)) { info.props.addValue("float","left"); } else if (isRight(sPos) && mayWrapLeft(sWrap)) { info.props.addValue("float","right"); } 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); getTextCv().traverseBlockText(onode,textbox); 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