All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.docx4j.convert.out.fo.FOPictWriterAbstract Maven / Gradle / Ivy

Go to download

docx4j is a library which helps you to work with the Office Open XML file format as used in docx documents, pptx presentations, and xlsx spreadsheets.

There is a newer version: 6.1.2
Show newest version
/*
   Licensed to Plutext Pty Ltd under one or more contributor license agreements.  
   
 *  This file is part of docx4j.

    docx4j is licensed under the Apache License, Version 2.0 (the "License"); 
    you may not use this file except in compliance with the License. 

    You may obtain a copy of the License at 

        http://www.apache.org/licenses/LICENSE-2.0 

    Unless required by applicable law or agreed to in writing, software 
    distributed under the License is distributed on an "AS IS" BASIS, 
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 
    See the License for the specific language governing permissions and 
    limitations under the License.

 */
package org.docx4j.convert.out.fo;

import java.util.HashMap;
import java.util.Map;

import javax.xml.transform.TransformerException;

import org.docx4j.XmlUtils;
import org.docx4j.convert.out.FORenderer;
import org.docx4j.convert.out.FOSettings;
import org.docx4j.convert.out.common.AbstractWmlConversionContext;
import org.docx4j.convert.out.common.ConversionSectionWrapper;
import org.docx4j.convert.out.common.writer.AbstractPictWriter;
import org.docx4j.convert.out.fo.renderers.AbstractFORenderer;
import org.docx4j.model.structure.PageDimensions;
import org.docx4j.vml.VmlAllCoreAttributes;
import org.docx4j.vml.wordprocessingDrawing.STVerticalAnchor;
import org.docx4j.vml.wordprocessingDrawing.STWrapType;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;

/**
 * Note that despite its name, this currently only handles v:textbox.
 * 
 * Images (ie ./v:shape/v:imagedata) are handled differently, by legacy code.
 * 
 * 
 * @author jharrop
 *
 */
public abstract class FOPictWriterAbstract extends AbstractPictWriter {
	
	protected static Logger log = LoggerFactory.getLogger(FOPictWriterAbstract.class);
	
	private static String XSL_FO = "http://www.w3.org/1999/XSL/Format";
	
	public FOPictWriterAbstract() {
		super();
	}
	
	public abstract boolean foRendererSupportsFoFloat();
	
	@Override
	public Node toNode(AbstractWmlConversionContext context, Object unmarshalledNode, 
			Node modelContent, TransformState state, Document doc)
			throws TransformerException {

		org.docx4j.wml.Pict pict = (org.docx4j.wml.Pict)unmarshalledNode;
		
		/*
            
              
                
                
              
              
                
                  
                    
                      
                        Content
                      
                    
                  
                
              
            
		 */

		// Get the shape or rectangle...
		org.docx4j.vml.VmlShapeElements shape = null;
		for (Object o : pict.getAnyAndAny() ) {
			
			o = XmlUtils.unwrap(o);
//			System.out.println(o.getClass().getName());
			if (o instanceof org.docx4j.vml.VmlShapeElements
					&& !(o instanceof org.docx4j.vml.CTShapetype)) {
				shape = (org.docx4j.vml.VmlShapeElements)o;
				log.debug("Found " + shape.getClass().getName());
				break;
			}
		}
		if (shape==null) {
			return context.getMessageWriter().message(context, 
					"Couldn't find v:shape (or v:rectangle etc) in w:pict.");
		}

		org.docx4j.vml.CTTextbox textBox = null;
		org.docx4j.vml.wordprocessingDrawing.CTWrap w10Wrap = null;  
		for (Object o : shape.getEGShapeElements() ) {
			
			o = XmlUtils.unwrap(o);
			
			if (o instanceof org.docx4j.vml.CTTextbox) {
				textBox = (org.docx4j.vml.CTTextbox)o;
			}
			if (o instanceof org.docx4j.vml.wordprocessingDrawing.CTWrap) {
				w10Wrap = (org.docx4j.vml.wordprocessingDrawing.CTWrap)o;
			}
		}
		if (textBox==null) {
			return context.getMessageWriter().message(context, 
					"Couldn't find v:textbox in w:shape.");
		}
		
		

		Map props = null;
		if (shape instanceof VmlAllCoreAttributes) {
			props = getProperties(((VmlAllCoreAttributes)shape).getStyle()); 
		} else {
			log.warn(shape.getClass().getName() + " does not implement VmlAllCoreAttributes, so can't access @style if present");
			return context.getMessageWriter().message(context, 
					shape.getClass().getName() + " does not implement VmlAllCoreAttributes, so can't access @style if present");
		}
		
		
//		// temp
//		if (props.size()==0) {
//			System.out.println(XmlUtils.marshaltoString(pict));
//		}
		
		
		boolean wrap = true;
		if (w10Wrap!=null) {
			
			if (w10Wrap.getType()!=null
					&& (w10Wrap.getType().equals(STWrapType.TOP_AND_BOTTOM)
					|| w10Wrap.getType().equals(STWrapType.SQUARE)
					|| w10Wrap.getType().equals(STWrapType.TIGHT)
					|| w10Wrap.getType().equals(STWrapType.THROUGH))) {
				
				wrap = false;
			}
			
			// the no wrap page top case
			if (w10Wrap.getAnchory()!=null
					&& (w10Wrap.getAnchory().equals(STVerticalAnchor.PAGE)  )) {
				
				wrap = false;
			} 
			
		}
			
		return handleVTextBox(context, modelContent, doc, shape, props, wrap);
	}

	
	
	public Node handleVTextBox(AbstractWmlConversionContext context,
			Node modelContent, Document doc, 
			org.docx4j.vml.VmlShapeElements shape,
			Map props, 
			boolean wrap) {
	
		String mso_position_vertical_relative = props.get("mso-position-vertical-relative");
		String mso_position_vertical = props.get("mso-position-vertical");
//		String z_index = props.get("z_index"); // negative -> behind
//
		
//		String width = props.get("width");
//		String height = props.get("height");

		
		// How to get current page width??
		ConversionSectionWrapper csw = context.getSections().getCurrentSection();
		PageDimensions pageDimensions = csw.getPageDimensions();
		
		// pgSz.getW().intValue() - (pgMar.getLeft().intValue() + pgMar.getRight().intValue());
		int writableWidthTwips = pageDimensions.getWritableWidthTwips();
		float writableWidthPts = writableWidthTwips/20;
		int writableHeightTwips = pageDimensions.getWritableHeightTwips();
		float writableHeightPts = writableHeightTwips/20;

		// TODO - avoid reference to FORenderer here! See  
		FORenderer foRenderer  = ((FOSettings)context.getConversionSettings()).getCustomFoRenderer();
		log.debug(foRenderer.getClass().getName());
		
		
		if (wrap) {
			// Floats define a block that is "out of line" in drifts to the top/left/right side of the page;
			// the text *flows around it*.
			
			log.debug("textbox - wrapped text");

			if (mso_position_vertical_relative==null) {
				
				try {
					log.warn(XmlUtils.marshaltoString(shape));
				} catch (Exception e) {
					log.warn(e.getMessage());
				}
				return context.getMessageWriter().message(context, 
						"mso_position_vertical_relative==null.  What to do?");
				
			} else {

				if ( mso_position_vertical_relative.equals("text")) {
					
					// Just lay it out in the flow; graceful degradation
					Element ret = doc.createElementNS(XSL_FO, "block");  
					setBorders(ret); // for now, solid borders, so we can see the text box
					
					String margin_top = props.get("margin-top");
					if (margin_top==null) {
						log.error("margin top prop not found.  What to do?");
						XmlUtils.treeCopy(modelContent.getChildNodes(), ret);
						return ret;
					} else {
						float marginTop = parsePtsVal(margin_top);
						
						
						if (foRendererSupportsFoFloat()) {
							
							// TODO not tested; need a renderer which supports float
							
							String mso_position_horizontal_relative = props.get("mso-position-horizontal-relative");
							String mso_position_horizontal = props.get("mso-position-horizontal"); // eg center
							
							
							if (mso_position_horizontal_relative==null) {
								log.warn("No support for mso_position_horizontal_relative==null");							
								XmlUtils.treeCopy(modelContent.getChildNodes(), ret);
								return ret;
							} else if (!mso_position_horizontal_relative.equals("text") ) {
								log.warn("No support for mso_position_horizontal_relative==" + mso_position_horizontal_relative.equals("text"));							
								XmlUtils.treeCopy(modelContent.getChildNodes(), ret);
								return ret;
							} else {  // mso_position_horizontal_relative.equals("text")

								
								float ml = parsePtsVal(props.get("margin-left"));
								
								if (mso_position_horizontal==null) {
									
									log.warn("No support for mso_position_horizontal==null");							
									XmlUtils.treeCopy(modelContent.getChildNodes(), ret);								
									return ret;
								}
								else if ( mso_position_horizontal.equals("absolute")) {
									// eg in Word UI, "Absolute position to right of column"
									
									// We can only float left or right (not center etc)
									log.warn("Degrading absolute position to left/right");
									if (ml/writableWidthPts <= 0.5) {
										mso_position_horizontal = "left";
									} else {
										mso_position_horizontal = "right";					
									}
								}
								
								ret = doc.createElementNS(XSL_FO, "float");  
								setBorders(ret); // for now, solid borders, so we can see the text box
								
								if ( mso_position_horizontal.equals("left")) {

									ret.setAttribute("float",  "start");
																		
								} else if ( mso_position_horizontal.equals("center")) {
									
									log.warn("Degrading center to right");
									ret.setAttribute("float",  "end");
									
								} else if ( mso_position_horizontal.equals("right")) {
									
									ret.setAttribute("float",  "end");
								} 
								XmlUtils.treeCopy(modelContent.getChildNodes(), ret);								
									
							} 							
						} else {
							//  FOP does not support float, so the best we can do is ...
							
							if (marginTop<=0) {
								// The text box is above the para
								marginTopZeroCase( props, ret, writableWidthPts);
								XmlUtils.treeCopy(modelContent.getChildNodes(), ret);
							} else {
								// The text box is next to or below the para
								// We'll assume here it is next to (since we don't know how long the para is)
								// Implement with a table (simulating float) .. simple mindedly put the textbox either left or right.
								// Obviously this won't work if the para had 2 text boxes, one to left and other to right..
								
								ret = doc.createElementNS(XSL_FO, "table");  
								if (marginTopPositiveCase(foRenderer, props, doc, ret, writableWidthPts, modelContent.getChildNodes())) {
									
								} else {
									// Couldn't do it.
									ret = doc.createElementNS(XSL_FO, "block");  
									setBorders(ret); // for now, solid borders, so we can see the text box
									XmlUtils.treeCopy(modelContent.getChildNodes(), ret);								
								}
							}
						
						}
					}
					
					log.debug(XmlUtils.w3CDomNodeToString(ret));
					
					return ret;		
					
				}  else if (foRendererSupportsFoFloat()) {
					
					// NB supportsFloat code is untested.
					
					if  (mso_position_vertical_relative.equals("page")
							|| mso_position_vertical_relative.equals("top-margin-area")
							|| mso_position_vertical_relative.equals("bottom-margin-area")) { 

						if ( mso_position_vertical_relative.equals("page")) { 
							
							if (mso_position_vertical.equals("top")) {

								Element ret = doc.createElementNS(XSL_FO, "float"); 
								ret.setAttribute("float", "before");
								setBorders(ret); // for now, solid borders, so we can see the text box

								log.debug(XmlUtils.w3CDomNodeToString(ret));
								return ret;		
								
							} else if (mso_position_vertical.equals("bottom")) {
								
								Element ret = doc.createElementNS(XSL_FO, "footnote"); 
								Element footnoteBody = doc.createElementNS(XSL_FO, "footnote-body"); 
								ret.appendChild(footnoteBody);
								Element block = doc.createElementNS(XSL_FO, "block"); 
								footnoteBody.appendChild(block);
								setBorders(block); // for now, solid borders, so we can see the text box
								
								XmlUtils.treeCopy(modelContent.getChildNodes(), block);
								return ret;
								
								
							} else {
								
								log.warn("No support for mso_position_vertical==" + mso_position_vertical);							
								return context.getMessageWriter().message(context, 
										"TODO for fo:float capable renderer, support no-wrap + mso-position-vertical=" + mso_position_vertical);
							}
							
												
						} else if ( mso_position_vertical_relative.equals("top-margin-area")) { 
							
							Element ret = doc.createElementNS(XSL_FO, "float"); 
							ret.setAttribute("float", "before");
							setBorders(ret); // for now, solid borders, so we can see the text box

							log.debug(XmlUtils.w3CDomNodeToString(ret));
							return ret;		
												
						} else if (  mso_position_vertical_relative.equals("bottom-margin-area")) { 

							Element ret = doc.createElementNS(XSL_FO, "footnote"); 
							Element footnoteBody = doc.createElementNS(XSL_FO, "footnote-body"); 
							ret.appendChild(footnoteBody);
							Element block = doc.createElementNS(XSL_FO, "block"); 
							footnoteBody.appendChild(block);
							setBorders(block); // for now, solid borders, so we can see the text box
							
							XmlUtils.treeCopy(modelContent.getChildNodes(), block);
							return ret;
												
						} else {

							// Can't happen
							return context.getMessageWriter().message(context, 
									"TODO (how did we get here?) mso-position-vertical-relative=" + mso_position_vertical_relative);
							
						}
					}  else {

						return context.getMessageWriter().message(context, 
								"TODO for fo:float capable renderer, support no-wrap + mso-position-vertical-relative=" + mso_position_vertical_relative);
						
					}
						
				} else {
					
					return context.getMessageWriter().message(context, 
							"TODO for fo:float INcapable renderer, support no-wrap + mso-position-vertical-relative=" + mso_position_vertical_relative);
					
				}
				
			}
			
		} else {
			// not wrap, so use
			// block-container .. text does not wrap around 
			// .. see https://www.ecrion.com/help/products/xfrenderingserver/xfultrascalehelp4/absolut_positioning.htm  

			// not wrap: can use fo:block-container/@position= fixed to approximate relative to page
			log.debug("textbox - over/behind docx text");
			Element ret = doc.createElementNS(XSL_FO, "block-container");  
			XmlUtils.treeCopy(modelContent.getChildNodes(), ret);								
			
			setBorders(ret); // for now, solid borders, so we can see the text box
			
			
			String mso_position_horizontal_relative = props.get("mso-position-horizontal-relative");
			String mso_position_horizontal = props.get("mso-position-horizontal"); // eg center
			
			
			if (mso_position_horizontal_relative==null) {
				log.warn("No support for mso_position_horizontal_relative==null");							
			} else if (!mso_position_horizontal_relative.equals("text") ) {
				log.warn("No support for mso_position_horizontal_relative==" + mso_position_horizontal_relative.equals("text"));							
			} else {  // mso_position_horizontal_relative.equals("text")
				
				float boxWidth = parsePtsVal(props.get("width"));
			
				ret.setAttribute("width",  props.get("width") );
				ret.setAttribute("height",  props.get("height") );
				
				if (mso_position_horizontal==null) {
					
					log.warn("No support for mso_position_horizontal==null");							
					
				} else if ( mso_position_horizontal.equals("left")) {

					ret.setAttribute("left",  "0pt");
												
				} else if ( mso_position_horizontal.equals("center")) {
					
					// convert width:186.95pt to margin setting
					int marginLeft = Math.round((writableWidthPts - boxWidth)/2); 
					
					ret.setAttribute("left", marginLeft + "pt");
					
				} else if ( mso_position_horizontal.equals("right")) {
					
					// page width - box width
					int marginLeft = Math.round((writableWidthPts - boxWidth)); 
					ret.setAttribute("left", marginLeft + "pt");
					
				} else if ( mso_position_horizontal.equals("absolute")) {
					// eg in Word UI, "Absolute position to right of column"

					// eg margin-left:108pt	
					ret.setAttribute("margin-left", props.get("margin-left") );
				}
					
			} 					

			ret.setAttribute("z-index", props.get("z-index") );					

			
			String margin_top = props.get("margin-top");
			if (mso_position_vertical_relative==null) {
				
				log.warn(XmlUtils.marshaltoString(shape));
				return context.getMessageWriter().message(context, 
						"mso_position_vertical_relative==null.  What to do?");
				
			} else {

				if ( mso_position_vertical_relative.equals("text")) {

					ret.setAttribute("position", "absolute");
					
					// @top
					if (margin_top==null) {
						// ret.setAttribute("top", "0pt");
					} else {
						//float marginTop = parsePtsVal(margin_top);
						ret.setAttribute("top", margin_top);
					}
										
					log.debug(XmlUtils.w3CDomNodeToString(ret));
					
					return ret;		
					
				} else if ( mso_position_vertical_relative.equals("page")
						|| mso_position_vertical_relative.equals("top-margin-area")
						|| mso_position_vertical_relative.equals("bottom-margin-area")) { 

					ret.setAttribute("position", "fixed");  // relative to page
					
					if ( mso_position_vertical_relative.equals("page")) { 
						
						if (mso_position_vertical.equals("top")) {

							// @top
							if (margin_top==null) {
								// ret.setAttribute("top", "0pt");
							} else {
								//float marginTop = parsePtsVal(margin_top);
								ret.setAttribute("top", margin_top);
							}
							
						} else if (mso_position_vertical.equals("bottom")) {
							
							//float boxHeight = parsePtsVal(props.get("height"));						
							//int top = Math.round(writableHeightPts - boxHeight);
							
							int top = Math.round(writableHeightPts );
							ret.setAttribute("top", top + "pt");
							
						} else {
							
							log.warn("No support for mso_position_vertical==" + mso_position_vertical);							
							
						}
						
						log.debug(XmlUtils.w3CDomNodeToString(ret));
						return ret;		
											
					} else if ( mso_position_vertical_relative.equals("top-margin-area")) { 
						
						// @top
						if (margin_top==null) {
							// ret.setAttribute("top", "0pt");
						} else {
							//float marginTop = parsePtsVal(margin_top);
							ret.setAttribute("top", margin_top);
						}
						
						log.debug(XmlUtils.w3CDomNodeToString(ret));
						return ret;		
											
					} else if (  mso_position_vertical_relative.equals("bottom-margin-area")) { 

						//float boxHeight = parsePtsVal(props.get("height"));						
						//int top = Math.round(writableHeightPts - boxHeight);
						
						int top = Math.round(writableHeightPts );
						ret.setAttribute("top", top + "pt");
						
						log.debug(XmlUtils.w3CDomNodeToString(ret));
						return ret;		
											
					} else {

						// Can't happen
						return context.getMessageWriter().message(context, 
								"TODO (how did we get here?) mso-position-vertical-relative=" + mso_position_vertical_relative);
						
					}
				}  else {

					return context.getMessageWriter().message(context, 
							"TODO support no-wrap + mso-position-vertical-relative=" + mso_position_vertical_relative);
					
				}
				
			}
			
		}
		
	}
	
	private void setBorders(Element ret) {
		ret.setAttribute("border-left-style", "solid");
		ret.setAttribute("border-top-style", "solid");
		ret.setAttribute("border-bottom-style", "solid");
		ret.setAttribute("border-right-style", "solid");
		
	}

	private Map getProperties(String s) {
		
		Map map = new HashMap();
		
		if (s==null) {
			log.warn("shape has no @style");
			return map;
		}
		
		for(final String entry : s.split(";")) {
		    final String[] parts = entry.split(":");
		    assert(parts.length == 2) : "Invalid entry: " + entry;
		    map.put(parts[0], parts[1]);
		}		
		return map;
		
	}

	private float parsePtsVal(String pts) {
		
		if (pts==null) {
			log.warn("No val!");
			return -99; // or exception?
			
		} else if (pts.contains("pt")) {

			pts = pts.substring(0, pts.indexOf("pt"));
			return Float.parseFloat(pts);
			
		} else if (pts.equals("0")) {
			
			return 0;
			
		} else {

			log.warn("Unit is not points! " + pts);
			return -99; // or exception?
			
		}
		
	}
	
	/**
	 * Wrap case, for renderers which don't support fo:float (eg Apache FOP)
	 * 
	 * @param props
	 * @param ret
	 * @param widthPts
	 */
	private void marginTopZeroCase(Map props, Element ret, float widthPts ) {

//		mso-position-horizontal:center
//		mso-position-horizontal-relative:text
		
		String mso_position_horizontal_relative = props.get("mso-position-horizontal-relative");
		String mso_position_horizontal = props.get("mso-position-horizontal"); // eg center
		
		
		if (mso_position_horizontal_relative==null) {
			log.warn("No support for mso_position_horizontal_relative==null");							
		} else if (!mso_position_horizontal_relative.equals("text") ) {
			log.warn("No support for mso_position_horizontal_relative==" + mso_position_horizontal_relative.equals("text"));							
		} else {  // mso_position_horizontal_relative.equals("text")

			// strategy is to use @margin-left for positioning
			// (since the margin is outside the block border).
			// This works for FOP.  If it didn't, I guess we could try using a table
			
			float boxWidth = parsePtsVal(props.get("width"));
			
			if (mso_position_horizontal==null) {
				
				log.warn("No support for mso_position_horizontal==null");							
				
			} else if ( mso_position_horizontal.equals("left")) {

				ret.setAttribute("margin-left",  "0pt");
				
				// page width - box width
				int marginRight = Math.round((widthPts - boxWidth)); 
				ret.setAttribute("margin-right", marginRight + "pt");
				
			} else if ( mso_position_horizontal.equals("center")) {
				
				// convert width:186.95pt to margin setting
				int marginLeft = Math.round((widthPts - boxWidth)/2); 
				
				ret.setAttribute("margin-left", marginLeft + "pt");
				ret.setAttribute("margin-right", marginLeft + "pt");
				
			} else if ( mso_position_horizontal.equals("right")) {
				
				ret.setAttribute("margin-right",  "0pt");
				// page width - box width
				int marginLeft = Math.round((widthPts - boxWidth)); 
				ret.setAttribute("margin-left", marginLeft + "pt");
				
			} else if ( mso_position_horizontal.equals("absolute")) {
				// eg in Word UI, "Absolute position to right of column"

				// eg margin-left:108pt	
				ret.setAttribute("margin-left", props.get("margin-left") );

				float ml = parsePtsVal(props.get("margin-left"));
				int mRight = Math.round((widthPts - (boxWidth+ml))); 
				ret.setAttribute("margin-right",  mRight + "pt");
			}
				
		} 
		
	}

	/**
	 * Wrap case, for renderers which don't support fo:float (eg Apache FOP).  Try to degrade gracefully.
	 * 
	 * @param props
	 * @param doc
	 * @param ret
	 * @param widthPts
	 * @param childNodes
	 * @return
	 */
	private boolean marginTopPositiveCase(FORenderer foRenderer, Map props, Document doc, Element ret, float widthPts,
			NodeList childNodes) {

//		mso-position-horizontal:center
//		mso-position-horizontal-relative:text
		
		String mso_position_horizontal_relative = props.get("mso-position-horizontal-relative");
		String mso_position_horizontal = props.get("mso-position-horizontal"); // eg center
		
		
		if (mso_position_horizontal_relative==null) {
			log.warn("No support for mso_position_horizontal_relative==null");
			return false;
		} else if (!mso_position_horizontal_relative.equals("text") ) {
			log.warn("No support for mso_position_horizontal_relative==" + mso_position_horizontal_relative.equals("text"));
			return false;
		} else {  // mso_position_horizontal_relative.equals("text")

			// strategy is to use @margin-left for positioning
			// (since the margin is outside the block border).
			// This works for FOP.  If it didn't, I guess we could try using a table
			
			float boxWidth = parsePtsVal(props.get("width"));
			
			if (mso_position_horizontal==null) {
				
				log.warn("No support for mso_position_horizontal==null");							
				return false;
			}
			float ml = parsePtsVal(props.get("margin-left"));
			
			if ( mso_position_horizontal.equals("absolute")) {
				// eg in Word UI, "Absolute position to right of column"
				
				// our simple minded approach here is to treat as left, right or centre ..

				if (ml/widthPts < 0.334) {
					// fake left
					mso_position_horizontal = "left";
				} else if (ml/widthPts < 0.665) {
					// fake right
					mso_position_horizontal = "right";
				} else {
					// fake center
					mso_position_horizontal = "center";					
				}
			}
			
			if ( mso_position_horizontal.equals("left")) {
				
				Element tcol1 = doc.createElementNS(XSL_FO, "table-column");
				ret.appendChild(tcol1);
				tcol1.setAttribute("column-number", "1"); // optional?
				// width of left cell is ml + boxWidth
				int col1W = Math.round(ml + boxWidth);
				tcol1.setAttribute("column-width", col1W +"pt");

				Element tcol2 = doc.createElementNS(XSL_FO, "table-column");
				ret.appendChild(tcol2);
				tcol2.setAttribute("column-number", "2"); // optional?
				// width of left cell is ml + boxWidth
				int col2W = Math.round(widthPts-col1W);
				tcol2.setAttribute("column-width", col2W +"pt");
				
				/*
		          
		            
		              

						
		                
		              
		              

						
						
		              
		            
		          	
		          */			
				Element tbody = doc.createElementNS(XSL_FO, "table-body");
				ret.appendChild(tbody);
				Element trow = doc.createElementNS(XSL_FO, "table-row");
				tbody.appendChild(trow);
				
				// Cell 1 is the text box
				Element tc1 = doc.createElementNS(XSL_FO, "table-cell");
				trow.appendChild(tc1);				
				Element block = doc.createElementNS(XSL_FO, "block");  

				
				setBorders(block); // for now, solid borders, so we can see the text box
				tc1.appendChild(block);
				XmlUtils.treeCopy(childNodes, block);
				
				// Cell 2 is the p content proper (which we don't have)
				Element tc2 = doc.createElementNS(XSL_FO, "table-cell");
				trow.appendChild(tc2);	
//				tc2.appendChild(
//						doc.createComment("CONTENT GOES HERE"));
				Element placeholder = doc.createElementNS(XSL_FO, "block");  
				placeholder.setTextContent("#TEXTBOX#"); // magic string
				tc2.appendChild(placeholder);
				
				// TODO here the architecture needs further thought
				if (foRenderer instanceof AbstractFORenderer) {
					
					((AbstractFORenderer)foRenderer).TEXTBOX_POSTPROCESSING_REQUIRED = true;
						// currently done in the render step in FORendererApacheFOP
						// Maybe it should be done in AbstractFOExporter.postprocess,
						// but then we'd need one of those per renderer vendor?
				} else {
					log.warn("TODO: implement TEXTBOX_POSTPROCESSING_REQUIRED for " + foRenderer.getClass().getName() );
				}
				
				return true;
				
			} else if ( mso_position_horizontal.equals("center")) {
				
				// use 3 cells
				// No, can't do it...
				// best we could do would be to put the textbox centred
				// below the text.
				log.warn("Can't support mso_position_horizontal:center");
				
				return false;
				
			} else if ( mso_position_horizontal.equals("right")) {

				Element tcol1 = doc.createElementNS(XSL_FO, "table-column");
				ret.appendChild(tcol1);
				tcol1.setAttribute("column-number", "1"); // optional?
				// width of left cell is ml + boxWidth
				int col1W = Math.round(widthPts-boxWidth);
				tcol1.setAttribute("column-width", col1W +"pt");

				Element tcol2 = doc.createElementNS(XSL_FO, "table-column");
				ret.appendChild(tcol2);
				tcol2.setAttribute("column-number", "2"); // optional?
				// width of left cell is ml + boxWidth
				int col2W = Math.round(boxWidth);
				tcol2.setAttribute("column-width", col2W +"pt");
				
				Element tbody = doc.createElementNS(XSL_FO, "table-body");
				ret.appendChild(tbody);
				Element trow = doc.createElementNS(XSL_FO, "table-row");
				tbody.appendChild(trow);

				// Cell 1 is the p content proper (which we don't have)
				Element tc1 = doc.createElementNS(XSL_FO, "table-cell");
				trow.appendChild(tc1);	
//				tc2.appendChild(
//						doc.createComment("CONTENT GOES HERE"));
				Element placeholder = doc.createElementNS(XSL_FO, "block");  
				placeholder.setTextContent("#TEXTBOX#"); // magic string
				tc1.appendChild(placeholder);
				
				// Cell 2 is the text box
				Element tc2 = doc.createElementNS(XSL_FO, "table-cell");
				trow.appendChild(tc2);				
				Element block = doc.createElementNS(XSL_FO, "block");  
				setBorders(block); // for now, solid borders, so we can see the text box
				tc2.appendChild(block);
				XmlUtils.treeCopy(childNodes, block);
				
				// TODO here the architecture needs further thought
				if (foRenderer instanceof AbstractFORenderer) {
					
					((AbstractFORenderer)foRenderer).TEXTBOX_POSTPROCESSING_REQUIRED = true;
						// currently done in the render step in FORendererApacheFOP
						// Maybe it should be done in AbstractFOExporter.postprocess,
						// but then we'd need one of those per renderer vendor?
				} else {
					log.warn("TODO: implement TEXTBOX_POSTPROCESSING_REQUIRED for " + foRenderer.getClass().getName() );
				}
				
				return true;
				
			} 
			return false;
				
		} 
		
	}
	
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy