#60656 - Support export file that contains emf and render it correctly

git-svn-id: https://svn.apache.org/repos/asf/poi/branches/hemf@1846472 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Andreas Beeker 2018-11-12 23:21:18 +00:00
parent 0e68ef5f84
commit 940daf0d92
13 changed files with 286 additions and 125 deletions

View File

@ -22,6 +22,8 @@ package org.apache.poi.sl.draw;
import java.awt.Font; import java.awt.Font;
import java.awt.Graphics2D; import java.awt.Graphics2D;
import java.util.Map; import java.util.Map;
import java.util.Set;
import java.util.TreeSet;
import org.apache.poi.common.usermodel.fonts.FontInfo; import org.apache.poi.common.usermodel.fonts.FontInfo;
import org.apache.poi.sl.draw.Drawable.DrawableHint; import org.apache.poi.sl.draw.Drawable.DrawableHint;
@ -33,6 +35,13 @@ import org.apache.poi.sl.draw.Drawable.DrawableHint;
*/ */
public class DrawFontManagerDefault implements DrawFontManager { public class DrawFontManagerDefault implements DrawFontManager {
protected final Set<String> knownSymbolFonts = new TreeSet<>(String.CASE_INSENSITIVE_ORDER);
public DrawFontManagerDefault() {
knownSymbolFonts.add("Wingdings");
knownSymbolFonts.add("Symbol");
}
@Override @Override
public FontInfo getMappedFont(Graphics2D graphics, FontInfo fontInfo) { public FontInfo getMappedFont(Graphics2D graphics, FontInfo fontInfo) {
return getFontWithFallback(graphics, Drawable.FONT_MAP, fontInfo); return getFontWithFallback(graphics, Drawable.FONT_MAP, fontInfo);
@ -49,11 +58,25 @@ public class DrawFontManagerDefault implements DrawFontManager {
public String mapFontCharset(Graphics2D graphics, FontInfo fontInfo, String text) { public String mapFontCharset(Graphics2D graphics, FontInfo fontInfo, String text) {
// TODO: find a real charset mapping solution instead of hard coding for Wingdings // TODO: find a real charset mapping solution instead of hard coding for Wingdings
String attStr = text; return (fontInfo != null && knownSymbolFonts.contains(fontInfo.getTypeface()))
if (fontInfo != null && "Wingdings".equalsIgnoreCase(fontInfo.getTypeface())) { ? mapSymbolChars(text)
: text;
}
/**
* Symbol fonts like "Wingdings" or "Symbol" have glyphs mapped to a Unicode private use range via the Java font loader,
* although a system font viewer might show you the glyphs in the ASCII range.
* This helper function maps the chars of the text string to the corresponding private use range chars.
*
* @param text the input string, typically consists of ASCII chars
* @return the mapped string, typically consists of chars in the range of 0xf000 to 0xf0ff
*
* @since POI 4.0.0
*/
public static String mapSymbolChars(String text) {
// wingdings doesn't contain high-surrogates, so chars are ok // wingdings doesn't contain high-surrogates, so chars are ok
boolean changed = false; boolean changed = false;
char chrs[] = attStr.toCharArray(); char chrs[] = text.toCharArray();
for (int i=0; i<chrs.length; i++) { for (int i=0; i<chrs.length; i++) {
// only change valid chars // only change valid chars
if ((0x20 <= chrs[i] && chrs[i] <= 0x7f) || if ((0x20 <= chrs[i] && chrs[i] <= 0x7f) ||
@ -63,11 +86,7 @@ public class DrawFontManagerDefault implements DrawFontManager {
} }
} }
if (changed) { return changed ? new String(chrs) : text;
attStr = new String(chrs);
}
}
return attStr;
} }
@Override @Override

View File

@ -138,20 +138,18 @@ public class HemfDraw {
Point2D pnt[] = { new Point2D.Double(), new Point2D.Double(), new Point2D.Double() }; Point2D pnt[] = { new Point2D.Double(), new Point2D.Double(), new Point2D.Double() };
// points-1 because of the first point int i=0;
final int pointCnt = hasStartPoint() ? points-2 : points;
for (int i=0; i+2<pointCnt; i+=3) {
// x (4 bytes): A 32-bit signed integer that defines the horizontal (x) coordinate of the point.
// y (4 bytes): A 32-bit signed integer that defines the vertical (y) coordinate of the point.
if (i==0) {
if (hasStartPoint()) { if (hasStartPoint()) {
if (i < points) {
size += readPoint(leis, pnt[0]); size += readPoint(leis, pnt[0]);
poly.moveTo(pnt[0].getX(), pnt[0].getY()); poly.moveTo(pnt[0].getX(), pnt[0].getY());
i++;
}
} else { } else {
poly.moveTo(0, 0); poly.moveTo(0, 0);
} }
}
for (; i+2<points; i+=3) {
size += readPoint(leis, pnt[0]); size += readPoint(leis, pnt[0]);
size += readPoint(leis, pnt[1]); size += readPoint(leis, pnt[1]);
size += readPoint(leis, pnt[2]); size += readPoint(leis, pnt[2]);
@ -758,6 +756,7 @@ public class HemfDraw {
size += LittleEndianConsts.INT_SIZE; size += LittleEndianConsts.INT_SIZE;
Point2D points[] = new Point2D[count]; Point2D points[] = new Point2D[count];
for (int i=0; i<count; i++) { for (int i=0; i<count; i++) {
points[i] = new Point2D.Double();
size += readPoint(leis, points[i]); size += readPoint(leis, points[i]);
} }
@ -783,12 +782,14 @@ public class HemfDraw {
case 0x04: case 0x04:
int mode2 = leis.readUByte(); int mode2 = leis.readUByte();
int mode3 = leis.readUByte(); int mode3 = leis.readUByte();
assert(mode2 == 0x04 && mode3 == 0x04); assert(mode2 == 0x04 && (mode3 == 0x04 || mode3 == 0x05));
poly.curveTo( poly.curveTo(
points[i].getX(), points[i].getY(), points[i].getX(), points[i].getY(),
points[i+1].getX(), points[i+1].getY(), points[i+1].getX(), points[i+1].getY(),
points[i+2].getX(), points[i+2].getY() points[i+2].getX(), points[i+2].getY()
); );
// update mode for closePath handling below
mode = mode3;
i+=2; i+=2;
break; break;
// PT_MOVETO // PT_MOVETO

View File

@ -683,33 +683,27 @@ public class HemfFill {
} }
static int readXForm(LittleEndianInputStream leis, AffineTransform xform) { static int readXForm(LittleEndianInputStream leis, AffineTransform xform) {
// mapping <java AffineTransform> = <xform>: // mapping <java AffineTransform> = <xform>
// m00 (scaleX) = eM11 (Horizontal scaling component) // m00 (scaleX) = eM11 (Horizontal scaling component)
// m11 (scaleY) = eM22 (Vertical scaling component) double m00 = leis.readFloat();
// m01 (shearX) = eM12 (Horizontal proportionality constant) // m01 (shearX) = eM12 (Horizontal proportionality constant)
double m01 = leis.readFloat();
// m10 (shearY) = eM21 (Vertical proportionality constant) // m10 (shearY) = eM21 (Vertical proportionality constant)
double m10 = leis.readFloat();
// m11 (scaleY) = eM22 (Vertical scaling component)
double m11 = leis.readFloat();
// m02 (translateX) = eDx (The horizontal translation component, in logical units.) // m02 (translateX) = eDx (The horizontal translation component, in logical units.)
double m02 = leis.readFloat();
// m12 (translateY) = eDy (The vertical translation component, in logical units.) // m12 (translateY) = eDy (The vertical translation component, in logical units.)
double m12 = leis.readFloat();
// A 32-bit floating-point value of the transform matrix. xform.setTransform(m00, m10, m01, m11, m02, m12);
double eM11 = leis.readFloat();
// A 32-bit floating-point value of the transform matrix.
double eM12 = leis.readFloat();
// A 32-bit floating-point value of the transform matrix.
double eM21 = leis.readFloat();
// A 32-bit floating-point value of the transform matrix.
double eM22 = leis.readFloat();
// A 32-bit floating-point value that contains a horizontal translation component, in logical units.
double eDx = leis.readFloat();
// A 32-bit floating-point value that contains a vertical translation component, in logical units.
double eDy = leis.readFloat();
xform.setTransform(eM11, eM21, eM12, eM22, eDx, eDy);
return 6 * LittleEndian.INT_SIZE; return 6 * LittleEndian.INT_SIZE;
} }

View File

@ -273,15 +273,15 @@ public class HemfFont extends HwmfFont {
// An 8-bit unsigned integer that specifies an italic font if set to 0x01; // An 8-bit unsigned integer that specifies an italic font if set to 0x01;
// otherwise, it MUST be set to 0x00. // otherwise, it MUST be set to 0x00.
italic = (leis.readUByte() == 0x01); italic = (leis.readUByte() != 0x00);
// An 8-bit unsigned integer that specifies an underlined font if set to 0x01; // An 8-bit unsigned integer that specifies an underlined font if set to 0x01;
// otherwise, it MUST be set to 0x00. // otherwise, it MUST be set to 0x00.
underline = (leis.readUByte() == 0x01); underline = (leis.readUByte() != 0x00);
// An 8-bit unsigned integer that specifies a strikeout font if set to 0x01; // An 8-bit unsigned integer that specifies a strikeout font if set to 0x01;
// otherwise, it MUST be set to 0x00. // otherwise, it MUST be set to 0x00.
strikeOut = (leis.readUByte() == 0x01); strikeOut = (leis.readUByte() != 0x00);
// An 8-bit unsigned integer that specifies the set of character glyphs. // An 8-bit unsigned integer that specifies the set of character glyphs.
// It MUST be a value in the WMF CharacterSet enumeration. // It MUST be a value in the WMF CharacterSet enumeration.
@ -441,7 +441,8 @@ public class HemfFont extends HwmfFont {
// A 32-bit unsigned integer that MUST be set to the value 0x08007664. // A 32-bit unsigned integer that MUST be set to the value 0x08007664.
int signature = leis.readInt(); int signature = leis.readInt();
assert (signature == 0x08007664); // some non-conformant applications don't write the magic code in
// assert (signature == 0x08007664);
// A 32-bit unsigned integer that specifies the number of elements in the // A 32-bit unsigned integer that specifies the number of elements in the
// Values array. It MUST be in the range 0 to 16, inclusive. // Values array. It MUST be in the range 0 to 16, inclusive.

View File

@ -27,7 +27,9 @@ import java.awt.geom.Point2D;
import java.awt.image.BufferedImage; import java.awt.image.BufferedImage;
import java.io.IOException; import java.io.IOException;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Arrays;
import java.util.List; import java.util.List;
import java.util.function.Function;
import org.apache.poi.hemf.draw.HemfGraphics; import org.apache.poi.hemf.draw.HemfGraphics;
import org.apache.poi.hwmf.draw.HwmfDrawProperties; import org.apache.poi.hwmf.draw.HwmfDrawProperties;
@ -443,8 +445,6 @@ public class HemfMisc {
protected HwmfBrushStyle brushStyle; protected HwmfBrushStyle brushStyle;
protected HwmfHatchStyle hatchStyle; protected HwmfHatchStyle hatchStyle;
protected int[] styleEntry;
protected final HwmfBitmapDib bitmap = new HwmfBitmapDib(); protected final HwmfBitmapDib bitmap = new HwmfBitmapDib();
@ -477,7 +477,8 @@ public class HemfMisc {
// A 32-bit unsigned integer that specifies the PenStyle. // A 32-bit unsigned integer that specifies the PenStyle.
// The value MUST be defined from the PenStyle enumeration table // The value MUST be defined from the PenStyle enumeration table
penStyle = HwmfPenStyle.valueOf((int) leis.readUInt()); final HemfPenStyle emfPS = HemfPenStyle.valueOf((int) leis.readUInt());
penStyle = emfPS;
// A 32-bit unsigned integer that specifies the width of the line drawn by the pen. // A 32-bit unsigned integer that specifies the width of the line drawn by the pen.
// If the pen type in the PenStyle field is PS_GEOMETRIC, this value is the width in logical // If the pen type in the PenStyle field is PS_GEOMETRIC, this value is the width in logical
@ -517,10 +518,14 @@ public class HemfMisc {
// If the pen type in the PenStyle field is PS_GEOMETRIC, the lengths are specified in logical // If the pen type in the PenStyle field is PS_GEOMETRIC, the lengths are specified in logical
// units; otherwise, the lengths are specified in device units. // units; otherwise, the lengths are specified in device units.
styleEntry = new int[numStyleEntries]; float[] dashPattern = new float[numStyleEntries];
for (int i = 0; i < numStyleEntries; i++) { for (int i = 0; i < numStyleEntries; i++) {
styleEntry[i] = (int) leis.readUInt(); dashPattern[i] = (int) leis.readUInt();
}
if (penStyle.getLineDash() == HwmfLineDash.USERSTYLE) {
emfPS.setLineDashes(dashPattern);
} }
size += numStyleEntries * LittleEndianConsts.INT_SIZE; size += numStyleEntries * LittleEndianConsts.INT_SIZE;
@ -533,8 +538,11 @@ public class HemfMisc {
@Override @Override
public String toString() { public String toString() {
// TODO: add style entries + bmp // TODO: add style entries + bmp
return super.toString().replaceFirst("\\{", return
"{ brushStyle: '"+brushStyle+"', hatchStyle: '"+hatchStyle+"', "); "{ brushStyle: '"+brushStyle+"'"+
", hatchStyle: '"+hatchStyle+"'"+
", dashPattern: "+ Arrays.toString(penStyle.getLineDashes())+
", "+super.toString().substring(1);
} }
} }
@ -602,7 +610,8 @@ public class HemfMisc {
@Override @Override
public void draw(HemfGraphics ctx) { public void draw(HemfGraphics ctx) {
AffineTransform tx = ctx.getInitTransform(); ctx.updateWindowMapMode();
AffineTransform tx = ctx.getTransform();
tx.concatenate(xForm); tx.concatenate(xForm);
ctx.setTransform(tx); ctx.setTransform(tx);
} }
@ -649,30 +658,46 @@ public class HemfMisc {
return; return;
} }
final AffineTransform tx;
switch (modifyWorldTransformMode) { switch (modifyWorldTransformMode) {
case MWT_LEFTMULTIPLY:
tx = ctx.getTransform();
tx.concatenate(adaptXForm(tx));
break;
case MWT_RIGHTMULTIPLY:
tx = ctx.getTransform();
tx.preConcatenate(adaptXForm(tx));
break;
case MWT_IDENTITY: case MWT_IDENTITY:
ctx.setTransform(ctx.getInitTransform()); ctx.updateWindowMapMode();
tx = ctx.getTransform();
break; break;
case MWT_LEFTMULTIPLY: {
AffineTransform tx = new AffineTransform(xForm);
tx.concatenate(ctx.getTransform());
ctx.setTransform(tx);
break;
}
case MWT_RIGHTMULTIPLY: {
AffineTransform tx = new AffineTransform(xForm);
tx.preConcatenate(ctx.getTransform());
ctx.setTransform(tx);
break;
}
default: default:
case MWT_SET: { case MWT_SET:
AffineTransform tx = ctx.getInitTransform(); ctx.updateWindowMapMode();
tx.concatenate(xForm); tx = ctx.getTransform();
ctx.setTransform(tx); tx.concatenate(adaptXForm(tx));
break; break;
} }
ctx.setTransform(tx);
} }
/**
* adapt xform depending on the base transformation (... experimental ...)
*/
private AffineTransform adaptXForm(AffineTransform other) {
// normalize signed zero
Function<Double,Double> nn = (d) -> (d == 0. ? 0. : d);
double yDiff = Math.signum(nn.apply(xForm.getTranslateY())) == Math.signum(nn.apply(other.getTranslateY())) ? 1. : -1.;
double xDiff = Math.signum(nn.apply(xForm.getTranslateX())) == Math.signum(nn.apply(other.getTranslateX())) ? 1. : -1.;
return new AffineTransform(
xForm.getScaleX() == 0 ? 1. : xForm.getScaleX(),
yDiff * xForm.getShearY(),
xDiff * xForm.getShearX(),
xForm.getScaleY() == 0. ? 1. : xForm.getScaleY(),
xForm.getTranslateX(),
xForm.getTranslateY()
);
} }
@Override @Override

View File

@ -0,0 +1,45 @@
/* ====================================================================
Licensed to the Apache Software Foundation (ASF) under one or more
contributor license agreements. See the NOTICE file distributed with
this work for additional information regarding copyright ownership.
The ASF licenses this file to You 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.apache.poi.hemf.record.emf;
import org.apache.poi.hwmf.record.HwmfPenStyle;
public class HemfPenStyle extends HwmfPenStyle {
private float[] dashPattern;
public static HemfPenStyle valueOf(int flag) {
HemfPenStyle ps = new HemfPenStyle();
ps.flag = flag;
return ps;
}
@Override
public float[] getLineDashes() {
return (getLineDash() == HwmfLineDash.USERSTYLE) ? dashPattern : super.getLineDashes();
}
public void setLineDashes(float[] dashPattern) {
this.dashPattern = (dashPattern == null) ? null : dashPattern.clone();
}
@Override
public HemfPenStyle clone() {
return (HemfPenStyle)super.clone();
}
}

View File

@ -43,6 +43,10 @@ public interface HemfRecord {
*/ */
long init(LittleEndianInputStream leis, long recordSize, long recordId) throws IOException; long init(LittleEndianInputStream leis, long recordSize, long recordId) throws IOException;
/**
* Draws the record, the default redirects to the parent WMF record drawing
* @param ctx the drawing context
*/
default void draw(HemfGraphics ctx) { default void draw(HemfGraphics ctx) {
if (this instanceof HwmfRecord) { if (this instanceof HwmfRecord) {
((HwmfRecord) this).draw(ctx); ((HwmfRecord) this).draw(ctx);

View File

@ -54,6 +54,8 @@ public class HemfRecordIterator implements Iterator<HemfRecord> {
return null; return null;
} }
final int readIndex = stream.getReadIndex();
final long recordId, recordSize; final long recordId, recordSize;
try { try {
recordId = stream.readUInt(); recordId = stream.readUInt();
@ -65,7 +67,7 @@ public class HemfRecordIterator implements Iterator<HemfRecord> {
HemfRecordType type = HemfRecordType.getById(recordId); HemfRecordType type = HemfRecordType.getById(recordId);
if (type == null) { if (type == null) {
throw new RecordFormatException("Undefined record of type:"+recordId); throw new RecordFormatException("Undefined record of type: "+recordId+" at "+Integer.toHexString(readIndex));
} }
final HemfRecord record = type.constructor.get(); final HemfRecord record = type.constructor.get();

View File

@ -141,14 +141,12 @@ public class HemfText {
dx.add((int) leis.readUInt()); dx.add((int) leis.readUInt());
size += LittleEndianConsts.INT_SIZE; size += LittleEndianConsts.INT_SIZE;
} }
} else {
// if there are no dx entries, reset the string end
strEnd = (int)recordSize;
} }
if (dx.size() < stringLength) { if (dx.size() < stringLength) {
// invalid dx array // invalid dx array
dx.clear(); dx.clear();
} }
strEnd = (int)recordSize;
break; break;
} }
default: default:

View File

@ -31,6 +31,7 @@ import java.awt.font.TextLayout;
import java.awt.geom.AffineTransform; import java.awt.geom.AffineTransform;
import java.awt.geom.Area; import java.awt.geom.Area;
import java.awt.geom.Dimension2D; import java.awt.geom.Dimension2D;
import java.awt.geom.NoninvertibleTransformException;
import java.awt.geom.Point2D; import java.awt.geom.Point2D;
import java.awt.geom.Rectangle2D; import java.awt.geom.Rectangle2D;
import java.awt.image.BufferedImage; import java.awt.image.BufferedImage;
@ -43,6 +44,7 @@ import java.util.NoSuchElementException;
import java.util.TreeMap; import java.util.TreeMap;
import org.apache.commons.codec.Charsets; import org.apache.commons.codec.Charsets;
import org.apache.poi.common.usermodel.fonts.FontCharset;
import org.apache.poi.common.usermodel.fonts.FontInfo; import org.apache.poi.common.usermodel.fonts.FontInfo;
import org.apache.poi.hwmf.record.HwmfBrushStyle; import org.apache.poi.hwmf.record.HwmfBrushStyle;
import org.apache.poi.hwmf.record.HwmfFont; import org.apache.poi.hwmf.record.HwmfFont;
@ -57,6 +59,7 @@ import org.apache.poi.hwmf.record.HwmfText;
import org.apache.poi.hwmf.record.HwmfText.WmfExtTextOutOptions; import org.apache.poi.hwmf.record.HwmfText.WmfExtTextOutOptions;
import org.apache.poi.sl.draw.DrawFactory; import org.apache.poi.sl.draw.DrawFactory;
import org.apache.poi.sl.draw.DrawFontManager; import org.apache.poi.sl.draw.DrawFontManager;
import org.apache.poi.sl.draw.DrawFontManagerDefault;
import org.apache.poi.util.LocaleUtil; import org.apache.poi.util.LocaleUtil;
public class HwmfGraphics { public class HwmfGraphics {
@ -152,7 +155,7 @@ public class HwmfGraphics {
int cap = ps.getLineCap().awtFlag; int cap = ps.getLineCap().awtFlag;
int join = ps.getLineJoin().awtFlag; int join = ps.getLineJoin().awtFlag;
float miterLimit = (float)getProperties().getPenMiterLimit(); float miterLimit = (float)getProperties().getPenMiterLimit();
float dashes[] = ps.getLineDash().dashes; float dashes[] = ps.getLineDashes();
boolean dashAlt = ps.isAlternateDash(); boolean dashAlt = ps.isAlternateDash();
// This value is not an integer index into the dash pattern array. // This value is not an integer index into the dash pattern array.
// Instead, it is a floating-point value that specifies a linear distance. // Instead, it is a floating-point value that specifies a linear distance.
@ -370,6 +373,17 @@ public class HwmfGraphics {
public void drawString(byte[] text, int length, Point2D reference, Dimension2D scale, Rectangle2D clip, WmfExtTextOutOptions opts, List<Integer> dx, boolean isUnicode) { public void drawString(byte[] text, int length, Point2D reference, Dimension2D scale, Rectangle2D clip, WmfExtTextOutOptions opts, List<Integer> dx, boolean isUnicode) {
final HwmfDrawProperties prop = getProperties(); final HwmfDrawProperties prop = getProperties();
final AffineTransform at = graphicsCtx.getTransform();
if (at.getScaleX() == 0. || at.getScaleY() == 0.) {
return;
}
try {
at.createInverse();
} catch (NoninvertibleTransformException e) {
return;
}
HwmfFont font = prop.getFont(); HwmfFont font = prop.getFont();
if (font == null || text == null || text.length == 0) { if (font == null || text == null || text.length == 0) {
return; return;
@ -390,11 +404,19 @@ public class HwmfGraphics {
} }
String textString = new String(text, charset).substring(0,length).trim(); String textString = new String(text, charset).substring(0,length).trim();
if (textString.isEmpty()) { if (textString.isEmpty()) {
return; return;
} }
DrawFontManager fontHandler = DrawFactory.getInstance(graphicsCtx).getFontManager(graphicsCtx);
FontInfo fontInfo = fontHandler.getMappedFont(graphicsCtx, font);
if (fontInfo.getCharset() == FontCharset.SYMBOL) {
textString = DrawFontManagerDefault.mapSymbolChars(textString);
}
AttributedString as = new AttributedString(textString); AttributedString as = new AttributedString(textString);
addAttributes(as, font); addAttributes(as, font, fontInfo.getTypeface());
// disabled for the time being, as the results aren't promising // disabled for the time being, as the results aren't promising
/* /*
@ -473,7 +495,6 @@ public class HwmfGraphics {
tx.transform(src, dst); tx.transform(src, dst);
final Shape clipShape = graphicsCtx.getClip(); final Shape clipShape = graphicsCtx.getClip();
final AffineTransform at = graphicsCtx.getTransform();
try { try {
if (clip != null) { if (clip != null) {
graphicsCtx.translate(-clip.getCenterX(), -clip.getCenterY()); graphicsCtx.translate(-clip.getCenterX(), -clip.getCenterY());
@ -503,11 +524,8 @@ public class HwmfGraphics {
} }
} }
private void addAttributes(AttributedString as, HwmfFont font) { private void addAttributes(AttributedString as, HwmfFont font, String typeface) {
DrawFontManager fontHandler = DrawFactory.getInstance(graphicsCtx).getFontManager(graphicsCtx); as.addAttribute(TextAttribute.FAMILY, typeface);
FontInfo fontInfo = fontHandler.getMappedFont(graphicsCtx, font);
as.addAttribute(TextAttribute.FAMILY, fontInfo.getTypeface());
as.addAttribute(TextAttribute.SIZE, getFontHeight(font)); as.addAttribute(TextAttribute.SIZE, getFontHeight(font));
if (font.isStrikeOut()) { if (font.isStrikeOut()) {
as.addAttribute(TextAttribute.STRIKETHROUGH, TextAttribute.STRIKETHROUGH_ON); as.addAttribute(TextAttribute.STRIKETHROUGH, TextAttribute.STRIKETHROUGH_ON);
@ -669,4 +687,11 @@ public class HwmfGraphics {
graphicsCtx.setTransform(at); graphicsCtx.setTransform(at);
} }
} }
/**
* @return the bounding box
*/
public Rectangle2D getBbox() {
return (Rectangle2D)bbox.clone();
}
} }

View File

@ -499,8 +499,8 @@ public class HwmfFill {
if (bitmap.isValid()) { if (bitmap.isValid()) {
ctx.drawImage(getImage(), srcBounds, dstBounds); ctx.drawImage(getImage(), srcBounds, dstBounds);
} else if (!dstBounds.isEmpty()) { } else if (!dstBounds.isEmpty()) {
BufferedImage bi = new BufferedImage((int)dstBounds.getWidth(), (int)dstBounds.getHeight(), BufferedImage.TYPE_INT_ARGB); BufferedImage bi = new BufferedImage(100, 100, BufferedImage.TYPE_INT_ARGB);
ctx.drawImage(bi, dstBounds, dstBounds); ctx.drawImage(bi, new Rectangle2D.Double(0,0,100,100), dstBounds);
} }
} }

View File

@ -142,7 +142,7 @@ public class HwmfPenStyle implements Cloneable {
private static final BitField SUBSECTION_JOIN = BitFieldFactory.getInstance(0x03000); private static final BitField SUBSECTION_JOIN = BitFieldFactory.getInstance(0x03000);
private static final BitField SUBSECTION_GEOMETRIC = BitFieldFactory.getInstance(0x10000); private static final BitField SUBSECTION_GEOMETRIC = BitFieldFactory.getInstance(0x10000);
private int flag; protected int flag;
public static HwmfPenStyle valueOf(int flag) { public static HwmfPenStyle valueOf(int flag) {
HwmfPenStyle ps = new HwmfPenStyle(); HwmfPenStyle ps = new HwmfPenStyle();
@ -162,6 +162,15 @@ public class HwmfPenStyle implements Cloneable {
return HwmfLineDash.valueOf(SUBSECTION_DASH.getValue(flag)); return HwmfLineDash.valueOf(SUBSECTION_DASH.getValue(flag));
} }
/**
* Convienence method which should be used instead of accessing {@link HwmfLineDash#dashes}
* directly, so an subclass can provide user-style dashes
*
* @return the dash pattern
*/
public float[] getLineDashes() {
return getLineDash().dashes;
}
/** /**
* The pen sets every other pixel (this style is applicable only for cosmetic pens). * The pen sets every other pixel (this style is applicable only for cosmetic pens).

View File

@ -18,6 +18,7 @@
package org.apache.poi.hwmf.record; package org.apache.poi.hwmf.record;
import static org.apache.poi.hwmf.record.HwmfDraw.boundsToString; import static org.apache.poi.hwmf.record.HwmfDraw.boundsToString;
import static org.apache.poi.hwmf.record.HwmfDraw.dimToString;
import static org.apache.poi.hwmf.record.HwmfDraw.normalizeBounds; import static org.apache.poi.hwmf.record.HwmfDraw.normalizeBounds;
import static org.apache.poi.hwmf.record.HwmfDraw.pointToString; import static org.apache.poi.hwmf.record.HwmfDraw.pointToString;
import static org.apache.poi.hwmf.record.HwmfDraw.readBounds; import static org.apache.poi.hwmf.record.HwmfDraw.readBounds;
@ -30,6 +31,7 @@ import java.awt.geom.Point2D;
import java.awt.geom.Rectangle2D; import java.awt.geom.Rectangle2D;
import java.io.IOException; import java.io.IOException;
import org.apache.poi.hwmf.draw.HwmfDrawProperties;
import org.apache.poi.hwmf.draw.HwmfGraphics; import org.apache.poi.hwmf.draw.HwmfGraphics;
import org.apache.poi.util.Dimension2DDouble; import org.apache.poi.util.Dimension2DDouble;
import org.apache.poi.util.LittleEndianConsts; import org.apache.poi.util.LittleEndianConsts;
@ -56,9 +58,15 @@ public class HwmfWindowing {
@Override @Override
public void draw(HwmfGraphics ctx) { public void draw(HwmfGraphics ctx) {
ctx.getProperties().setViewportOrg(origin.getX(), origin.getY()); final HwmfDrawProperties prop = ctx.getProperties();
Rectangle2D old = prop.getViewport();
double oldX = (old == null ? 0 : old.getX());
double oldY = (old == null ? 0 : old.getY());
if (oldX != origin.getX() || oldY != origin.getY()) {
prop.setViewportOrg(origin.getX(), origin.getY());
ctx.updateWindowMapMode(); ctx.updateWindowMapMode();
} }
}
@Override @Override
public String toString() { public String toString() {
@ -91,13 +99,19 @@ public class HwmfWindowing {
@Override @Override
public void draw(HwmfGraphics ctx) { public void draw(HwmfGraphics ctx) {
ctx.getProperties().setViewportExt(extents.getWidth(), extents.getHeight()); final HwmfDrawProperties prop = ctx.getProperties();
Rectangle2D old = prop.getViewport();
double oldW = (old == null ? 0 : old.getWidth());
double oldH = (old == null ? 0 : old.getHeight());
if (oldW != extents.getWidth() || oldH != extents.getHeight()) {
prop.setViewportExt(extents.getWidth(), extents.getHeight());
ctx.updateWindowMapMode(); ctx.updateWindowMapMode();
} }
}
@Override @Override
public String toString() { public String toString() {
return "{ width: "+extents.getWidth()+", height: "+extents.getHeight()+" }"; return dimToString(extents);
} }
} }
@ -121,10 +135,14 @@ public class HwmfWindowing {
@Override @Override
public void draw(HwmfGraphics ctx) { public void draw(HwmfGraphics ctx) {
Rectangle2D viewport = ctx.getProperties().getViewport(); final HwmfDrawProperties prop = ctx.getProperties();
Rectangle2D viewport = prop.getViewport();
if (offset.getX() != 0 || offset.getY() != 0) {
double x = (viewport == null) ? 0 : viewport.getX(); double x = (viewport == null) ? 0 : viewport.getX();
double y = (viewport == null) ? 0 : viewport.getY(); double y = (viewport == null) ? 0 : viewport.getY();
ctx.getProperties().setViewportOrg(x+offset.getX(), y+offset.getY()); prop.setViewportOrg(x + offset.getX(), y + offset.getY());
ctx.updateWindowMapMode();
}
} }
@Override @Override
@ -152,9 +170,15 @@ public class HwmfWindowing {
@Override @Override
public void draw(HwmfGraphics ctx) { public void draw(HwmfGraphics ctx) {
ctx.getProperties().setWindowOrg(getX(), getY()); final HwmfDrawProperties prop = ctx.getProperties();
final Rectangle2D old = prop.getWindow();
double oldX = (old == null ? 0 : old.getX());
double oldY = (old == null ? 0 : old.getY());
if (oldX != getX() || oldY != getY()) {
prop.setWindowOrg(getX(), getY());
ctx.updateWindowMapMode(); ctx.updateWindowMapMode();
} }
}
public double getY() { public double getY() {
return origin.getY(); return origin.getY();
@ -195,9 +219,15 @@ public class HwmfWindowing {
@Override @Override
public void draw(HwmfGraphics ctx) { public void draw(HwmfGraphics ctx) {
ctx.getProperties().setWindowExt(size.getWidth(), size.getHeight()); final HwmfDrawProperties prop = ctx.getProperties();
Rectangle2D old = prop.getWindow();
double oldW = (old == null ? 0 : old.getWidth());
double oldH = (old == null ? 0 : old.getHeight());
if (oldW != size.getWidth() || oldH != size.getHeight()) {
prop.setWindowExt(size.getWidth(), size.getHeight());
ctx.updateWindowMapMode(); ctx.updateWindowMapMode();
} }
}
public Dimension2D getSize() { public Dimension2D getSize() {
return size; return size;
@ -205,7 +235,7 @@ public class HwmfWindowing {
@Override @Override
public String toString() { public String toString() {
return "{ width: "+size.getWidth()+", height: "+size.getHeight()+" }"; return dimToString(size);
} }
} }
@ -229,10 +259,13 @@ public class HwmfWindowing {
@Override @Override
public void draw(HwmfGraphics ctx) { public void draw(HwmfGraphics ctx) {
Rectangle2D window = ctx.getProperties().getWindow(); final HwmfDrawProperties prop = ctx.getProperties();
ctx.getProperties().setWindowOrg(window.getX()+offset.getX(), window.getY()+offset.getY()); Rectangle2D old = prop.getWindow();
if (offset.getX() != 0 || offset.getY() != 0) {
prop.setWindowOrg(old.getX() + offset.getX(), old.getY() + offset.getY());
ctx.updateWindowMapMode(); ctx.updateWindowMapMode();
} }
}
@Override @Override
public String toString() { public String toString() {
@ -275,12 +308,15 @@ public class HwmfWindowing {
@Override @Override
public void draw(HwmfGraphics ctx) { public void draw(HwmfGraphics ctx) {
Rectangle2D window = ctx.getProperties().getWindow(); final HwmfDrawProperties prop = ctx.getProperties();
double width = window.getWidth() * scale.getWidth(); Rectangle2D old = prop.getWindow();
double height = window.getHeight() * scale.getHeight(); if (scale.getWidth() != 1.0 || scale.getHeight() != 1.0) {
double width = old.getWidth() * scale.getWidth();
double height = old.getHeight() * scale.getHeight();
ctx.getProperties().setWindowExt(width, height); ctx.getProperties().setWindowExt(width, height);
ctx.updateWindowMapMode(); ctx.updateWindowMapMode();
} }
}
@Override @Override
public String toString() { public String toString() {
@ -325,13 +361,15 @@ public class HwmfWindowing {
@Override @Override
public void draw(HwmfGraphics ctx) { public void draw(HwmfGraphics ctx) {
Rectangle2D viewport = ctx.getProperties().getViewport(); final HwmfDrawProperties prop = ctx.getProperties();
if (viewport == null) { final Rectangle2D old = prop.getViewport() == null ? prop.getWindow() : prop.getViewport();
viewport = ctx.getProperties().getWindow();
if (scale.getWidth() != 1.0 || scale.getHeight() != 1.0) {
double width = old.getWidth() * scale.getWidth();
double height = old.getHeight() * scale.getHeight();
prop.setViewportExt(width, height);
ctx.updateWindowMapMode();
} }
double width = viewport.getWidth() * scale.getWidth();
double height = viewport.getHeight() * scale.getHeight();
ctx.getProperties().setViewportExt(width, height);
} }
@Override @Override