/* * 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.txt; import java.awt.Point; import java.awt.geom.Rectangle2D; import java.util.Iterator; import java.util.LinkedList; import org.apache.fop.area.CTM; /** * This keeps information about the current state when writing to txt, i.e. * manages coordinate transformation matrices for getting absolute coordinates. */ public class TXTState { /** Keeps all coordinate transformation matrices during rendering. */ private LinkedList stackCTM = new LinkedList(); /** * Current result coordinate transformation matrix. It's product of * all matrices in order, saved in stackCTM. */ private CTM resultCTM = new CTM(); /** * Constructs a newly allocated TXTState object. */ public TXTState() { } /** * Updates result coordinate transformation matrix * (i.e. resultCTM), multipliing it by given matrix. * * @param ctm CTM */ private void updateResultCTM(CTM ctm) { resultCTM = resultCTM.multiply(ctm); } /** * Recalculate current result coordinate transformation matrix. */ private void calcResultCTM() { resultCTM = new CTM(); for (Iterator i = stackCTM.iterator(); i.hasNext();) { updateResultCTM((CTM) i.next()); } } /** * Push the current coordinate transformation matrix onto the stack and * reevaluate resultCTM. * * @param ctm instance of CTM */ public void push(CTM ctm) { stackCTM.addLast(ctm); updateResultCTM(ctm); } /** * Pop the coordinate transformation matrix from the stack and reevaluate * resultCTM. */ public void pop() { stackCTM.removeLast(); calcResultCTM(); } /** * Modifies coordinate transformation matrix in such a way, so * x-shift and y-shift will be transformed in text positions. * * @param ctm CTM to modify * @return instance of CTM */ public CTM refineCTM(CTM ctm) { double[] da = ctm.toArray(); // refine x-shift da[4] = Helper.roundPosition((int) da[4], TXTRenderer.CHAR_WIDTH); // refine y-shift da[5] = Helper.roundPosition((int) da[5], TXTRenderer.CHAR_HEIGHT); return new CTM(da[0], da[1], da[2], da[3], da[4], da[5]); } /** * Transforms point using ctm. * * @param p Point * @param ctm CTM * @return transformed Point */ public Point transformPoint(Point p, CTM ctm) { Rectangle2D r = new Rectangle2D.Double(p.x, p.y, 0, 0); CTM nctm = refineCTM(ctm); r = nctm.transform(r); return new Point((int) r.getX(), (int) r.getY()); } /** * Transforms point (x, y) using resultCTM. * * @param x x-coordinate * @param y y-coordinate * @return transformed Point */ public Point transformPoint(int x, int y) { return transformPoint(new Point(x, y), resultCTM); } /** * @return current result coordinate transformation matrix */ public CTM getResultCTM() { return resultCTM; } }