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

org.apache.fop.render.java2d.Java2DGraphicsPainter Maven / Gradle / Ivy

The newest version!
/*
 * 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.
 */

/* $Id$ */

package org.apache.fop.render.java2d;

import java.awt.BasicStroke;
import java.awt.Color;
import java.awt.Graphics2D;
import java.awt.Point;
import java.awt.Rectangle;
import java.awt.geom.GeneralPath;
import java.awt.geom.Line2D;
import java.io.IOException;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;

import org.apache.fop.fo.Constants;
import org.apache.fop.render.intermediate.GraphicsPainter;
import org.apache.fop.traits.RuleStyle;
import org.apache.fop.util.ColorUtil;

class Java2DGraphicsPainter implements GraphicsPainter {

    /** logging instance */
    static final Log log = LogFactory.getLog(Java2DGraphicsPainter.class);


    private GeneralPath currentPath;

    private final Java2DPainter painter;


    Java2DGraphicsPainter(Java2DPainter painter) {
        this.painter = painter;
    }

    private Java2DGraphicsState getG2DState() {
        return this.painter.g2dState;
    }


    private Graphics2D getG2D() {
        return getG2DState().getGraph();
    }

    public void drawBorderLine(int x1, int y1, int x2, int y2,
            boolean horz, boolean startOrBefore, int style, Color color)
                    throws IOException {
        float w = x2 - x1;
        float h = y2 - y1;
        if ((w < 0) || (h < 0)) {
            log.error("Negative extent received. Border won't be painted.");
            return;
        }
        switch (style) {
        case Constants.EN_DASHED:
            getG2D().setColor(color);
            if (horz) {
                float unit = Math.abs(2 * h);
                int rep = (int)(w / unit);
                if (rep % 2 == 0) {
                    rep++;
                }
                unit = w / rep;
                float ym = y1 + (h / 2);
                BasicStroke s = new BasicStroke(h, BasicStroke.CAP_BUTT,
                        BasicStroke.JOIN_MITER, 10.0f, new float[] {unit}, 0);
                getG2D().setStroke(s);
                getG2D().draw(new Line2D.Float(x1, ym, x2, ym));
            } else {
                float unit = Math.abs(2 * w);
                int rep = (int)(h / unit);
                if (rep % 2 == 0) {
                    rep++;
                }
                unit = h / rep;
                float xm = x1 + (w / 2);
                BasicStroke s = new BasicStroke(w, BasicStroke.CAP_BUTT,
                        BasicStroke.JOIN_MITER, 10.0f, new float[] {unit}, 0);
                getG2D().setStroke(s);
                getG2D().draw(new Line2D.Float(xm, y1, xm, y2));
            }
            break;
        case Constants.EN_DOTTED:
            getG2D().setColor(color);
            if (horz) {
                float unit = Math.abs(2 * h);
                int rep = (int)(w / unit);
                if (rep % 2 == 0) {
                    rep++;
                }
                unit = w / rep;
                float ym = y1 + (h / 2);
                BasicStroke s = new BasicStroke(h, BasicStroke.CAP_ROUND,
                        BasicStroke.JOIN_MITER, 10.0f, new float[] {0, unit}, 0);
                getG2D().setStroke(s);
                getG2D().draw(new Line2D.Float(x1, ym, x2, ym));
            } else {
                float unit = Math.abs(2 * w);
                int rep = (int)(h / unit);
                if (rep % 2 == 0) {
                    rep++;
                }
                unit = h / rep;
                float xm = x1 + (w / 2);
                BasicStroke s = new BasicStroke(w, BasicStroke.CAP_ROUND,
                        BasicStroke.JOIN_MITER, 10.0f, new float[] {0, unit}, 0);
                getG2D().setStroke(s);
                getG2D().draw(new Line2D.Float(xm, y1, xm, y2));
            }
            break;
        case Constants.EN_DOUBLE:
            getG2D().setColor(color);
            if (horz) {
                float h3 = h / 3;
                float ym1 = y1 + (h3 / 2);
                float ym2 = ym1 + h3 + h3;
                BasicStroke s = new BasicStroke(h3);
                getG2D().setStroke(s);
                getG2D().draw(new Line2D.Float(x1, ym1, x2, ym1));
                getG2D().draw(new Line2D.Float(x1, ym2, x2, ym2));
            } else {
                float w3 = w / 3;
                float xm1 = x1 + (w3 / 2);
                float xm2 = xm1 + w3 + w3;
                BasicStroke s = new BasicStroke(w3);
                getG2D().setStroke(s);
                getG2D().draw(new Line2D.Float(xm1, y1, xm1, y2));
                getG2D().draw(new Line2D.Float(xm2, y1, xm2, y2));
            }
            break;
        case Constants.EN_GROOVE:
        case Constants.EN_RIDGE:
            float colFactor = (style == Constants.EN_GROOVE ? 0.4f : -0.4f);
            if (horz) {
                Color uppercol = ColorUtil.lightenColor(color, -colFactor);
                Color lowercol = ColorUtil.lightenColor(color, colFactor);
                float h3 = h / 3;
                float ym1 = y1 + (h3 / 2);
                getG2D().setStroke(new BasicStroke(h3));
                getG2D().setColor(uppercol);
                getG2D().draw(new Line2D.Float(x1, ym1, x2, ym1));
                getG2D().setColor(color);
                getG2D().draw(new Line2D.Float(x1, ym1 + h3, x2, ym1 + h3));
                getG2D().setColor(lowercol);
                getG2D().draw(new Line2D.Float(x1, ym1 + h3 + h3, x2, ym1 + h3 + h3));
            } else {
                Color leftcol = ColorUtil.lightenColor(color, -colFactor);
                Color rightcol = ColorUtil.lightenColor(color, colFactor);
                float w3 = w / 3;
                float xm1 = x1 + (w3 / 2);
                getG2D().setStroke(new BasicStroke(w3));
                getG2D().setColor(leftcol);
                getG2D().draw(new Line2D.Float(xm1, y1, xm1, y2));
                getG2D().setColor(color);
                getG2D().draw(new Line2D.Float(xm1 + w3, y1, xm1 + w3, y2));
                getG2D().setColor(rightcol);
                getG2D().draw(new Line2D.Float(xm1 + w3 + w3, y1, xm1 + w3 + w3, y2));
            }
            break;
        case Constants.EN_INSET:
        case Constants.EN_OUTSET:
            colFactor = (style == Constants.EN_OUTSET ? 0.4f : -0.4f);
            if (horz) {
                color = ColorUtil.lightenColor(color, (startOrBefore ? 1 : -1) * colFactor);
                getG2D().setStroke(new BasicStroke(h));
                float ym1 = y1 + (h / 2);
                getG2D().setColor(color);
                getG2D().draw(new Line2D.Float(x1, ym1, x2, ym1));
            } else {
                color = ColorUtil.lightenColor(color, (startOrBefore ? 1 : -1) * colFactor);
                float xm1 = x1 + (w / 2);
                getG2D().setStroke(new BasicStroke(w));
                getG2D().setColor(color);
                getG2D().draw(new Line2D.Float(xm1, y1, xm1, y2));
            }
            break;
        case Constants.EN_HIDDEN:
            break;
        default:
            getG2D().setColor(color);
            if (horz) {
                float ym = y1 + (h / 2);
                getG2D().setStroke(new BasicStroke(h));
                getG2D().draw(new Line2D.Float(x1, ym, x2, ym));
            } else {
                float xm = x1 + (w / 2);
                getG2D().setStroke(new BasicStroke(w));
                getG2D().draw(new Line2D.Float(xm, y1, xm, y2));
            }
        }
    }

    public void drawLine(Point start, Point end, int width, Color color,
            RuleStyle style) throws IOException {
        if (start.y != end.y) {
            //TODO Support arbitrary lines if necessary
            throw new UnsupportedOperationException(
                    "Can only deal with horizontal lines right now");
        }

        saveGraphicsState();
        int half = width / 2;
        int starty = start.y - half;
        Rectangle boundingRect = new Rectangle(start.x, start.y - half, end.x - start.x, width);
        getG2DState().updateClip(boundingRect);

        switch (style.getEnumValue()) {
        case Constants.EN_SOLID:
        case Constants.EN_DASHED:
        case Constants.EN_DOUBLE:
            drawBorderLine(start.x, start.y - half, end.x, end.y + half,
                    true, true, style.getEnumValue(), color);
            break;
        case Constants.EN_DOTTED:
            int shift = half; //This shifts the dots to the right by half a dot's width
            drawBorderLine(start.x + shift, start.y - half, end.x + shift, end.y + half,
                    true, true, style.getEnumValue(), color);
            break;
        case Constants.EN_GROOVE:
        case Constants.EN_RIDGE:
            getG2DState().updateColor(ColorUtil.lightenColor(color, 0.6f));
            moveTo(start.x, starty);
            lineTo(end.x, starty);
            lineTo(end.x, starty + 2 * half);
            lineTo(start.x, starty + 2 * half);
            closePath();
            getG2D().fill(currentPath);
            currentPath = null;
            getG2DState().updateColor(color);
            if (style.getEnumValue() == Constants.EN_GROOVE) {
                moveTo(start.x, starty);
                lineTo(end.x, starty);
                lineTo(end.x, starty + half);
                lineTo(start.x + half, starty + half);
                lineTo(start.x, starty + 2 * half);
            } else {
                moveTo(end.x, starty);
                lineTo(end.x, starty + 2 * half);
                lineTo(start.x, starty + 2 * half);
                lineTo(start.x, starty + half);
                lineTo(end.x - half, starty + half);
            }
            closePath();
            getG2D().fill(currentPath);
            currentPath = null;
            break;

        case Constants.EN_NONE:
            // No rule is drawn
            break;
        default:
            break;
        } // end switch
        restoreGraphicsState();
    }

    /** {@inheritDoc} */
    public void moveTo(int x, int y) throws IOException {
        if (currentPath == null) {
            currentPath = new GeneralPath();
        }
        currentPath.moveTo(x, y);
    }

    /** {@inheritDoc} */
    public void lineTo(int x, int y) throws IOException {
        if (currentPath == null) {
            currentPath = new GeneralPath();
        }
        currentPath.lineTo(x, y);
    }

    /** {@inheritDoc} */
    public void arcTo(double startAngle, double endAngle, int cx, int cy,
            int width, int height) throws IOException {
    }

    /** {@inheritDoc} */
    public void rotateCoordinates(double angle) throws IOException {
    }

    /** {@inheritDoc} */
    public void translateCoordinates(int xTranslate, int yTranslate)
            throws IOException {
    }

    /** {@inheritDoc} */
    public void scaleCoordinates(float xScale, float yScale)
            throws IOException {
    }

    /** {@inheritDoc} */
    public void closePath() throws IOException {
        currentPath.closePath();
    }

    /** {@inheritDoc} */
    public void clip() throws IOException {
        if (currentPath == null) {
            throw new IllegalStateException("No current path available!");
        }
        getG2DState().updateClip(currentPath);
        currentPath = null;
    }

    /** {@inheritDoc} */
    public void saveGraphicsState() throws IOException {
        this.painter.saveGraphicsState();
    }

    /** {@inheritDoc} */
    public void restoreGraphicsState() throws IOException {
        this.painter.restoreGraphicsState();
        this.currentPath = null;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy