/* FractionAtom.java * ========================================================================= * This file is originally part of the JMathTeX Library - http://jmathtex.sourceforge.net * * Copyright (C) 2004-2007 Universiteit Gent * Copyright (C) 2009 DENIZET Calixte * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or (at * your option) any later version. * * This program 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 * General Public License for more details. * * A copy of the GNU General Public License can be found in the file * LICENSE.txt provided with the source distribution of this program (see * the META-INF directory in the source jar). This license can also be * found on the GNU website at http://www.gnu.org/licenses/gpl.html. * * If you did not receive a copy of the GNU General Public License along * with this program, contact the lead developer, or write to the Free * Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA * 02110-1301, USA. * * Linking this library statically or dynamically with other modules * is making a combined work based on this library. Thus, the terms * and conditions of the GNU General Public License cover the whole * combination. * * As a special exception, the copyright holders of this library give you * permission to link this library with independent modules to produce * an executable, regardless of the license terms of these independent * modules, and to copy and distribute the resulting executable under terms * of your choice, provided that you also meet, for each linked independent * module, the terms and conditions of the license of that module. * An independent module is a module which is not derived from or based * on this library. If you modify this library, you may extend this exception * to your version of the library, but you are not obliged to do so. * If you do not wish to do so, delete this exception statement from your * version. * */ package com.himamis.retex.renderer.share; import com.himamis.retex.renderer.share.exception.InvalidUnitException; /** * An atom representing a fraction. */ public class FractionAtom extends Atom { // whether the default thickness should not be used for the fraction line private boolean noDefault = false; // unit used for the thickness of the fraction line private int unit; // alignment settings for the numerator and denominator private int numAlign = TeXConstants.ALIGN_CENTER, denomAlign = TeXConstants.ALIGN_CENTER; // the atoms representing the numerator and denominator private Atom numerator, denominator; // thickness of the fraction line private double thickness; // thickness of the fraction line relative to the default thickness private double defFactor; // whether the "defFactor" value should be used private boolean defFactorSet = false; @Override final public Atom duplicate() { return setFields(new FractionAtom(numerator, denominator, unit, thickness, numAlign, denomAlign, defFactor, defFactorSet, noDefault)); } /** * Uses the default thickness for the fraction line * * @param num the numerator * @param den the denominator */ public FractionAtom(Atom num, Atom den) { this(num, den, true); } /** * Uses the default thickness for the fraction line * * @param num the numerator * @param den the denominator * @param rule whether the fraction line should be drawn */ public FractionAtom(Atom num, Atom den, boolean rule) { this(num, den, !rule, TeXConstants.UNIT_PIXEL, 0f); } /** * Depending on noDef, the given thickness and unit will be used (<-> the default thickness). * * @param num the numerator * @param den the denominator * @param noDef whether the default thickness should not be used for the fraction line * @param unit a unit constant for the line thickness * @param t the thickness of the fraction line (in the given unit) * @throws InvalidUnitException if the given integer is not a valid unit constant */ public FractionAtom(Atom num, Atom den, boolean noDef, int unit, double t) throws InvalidUnitException { // check unit SpaceAtom.checkUnit(unit); // unit ok numerator = num; denominator = den; noDefault = noDef; thickness = t; this.unit = unit; type = TeXConstants.TYPE_INNER; } /** * Uses the default thickness for the fraction line. * * @param num the numerator * @param den the denominator * @param rule whether the fraction line should be drawn * @param numAlign alignment of the numerator * @param denomAlign alignment of the denominator */ public FractionAtom(Atom num, Atom den, boolean rule, int numAlign, int denomAlign) { this(num, den, rule); this.numAlign = checkAlignment(numAlign); this.denomAlign = checkAlignment(denomAlign); } /** * The thickness of the fraction line will be "defFactor" times the default thickness. * * @param num the numerator * @param den the denominator * @param defFactor the thickness of the fraction line relative to the default thickness * @param numAlign alignment of the numerator * @param denomAlign alignment of the denominator */ public FractionAtom(Atom num, Atom den, double defFactor, int numAlign, int denomAlign) { this(num, den, true, numAlign, denomAlign); this.defFactor = defFactor; defFactorSet = true; } /** * The thickness of the fraction line is determined by the given value "t" in the given unit. * * @param num the numerator * @param den the denominator * @param unit a unit constant for the line thickness * @param t the thickness of the fraction line (in the given unit) * @param numAlign alignment of the numerator * @param denomAlign alignment of the denominator */ public FractionAtom(Atom num, Atom den, int unit, double t, int numAlign, int denomAlign) { this(num, den, unit, t); this.numAlign = checkAlignment(numAlign); this.denomAlign = checkAlignment(denomAlign); } /** * The thickness of the fraction line is determined by the given value "t" in the given unit. * * @param num the numerator * @param den the denominator * @param unit a unit constant for the line thickness * @param t the thickness of the fraction line (in the given unit) */ public FractionAtom(Atom num, Atom den, int unit, double t) { this(num, den, true, unit, t); } private FractionAtom(Atom numerator, Atom denominator, int unit, double thickness, int numAlign, int denomAlign, double defFactor, boolean defFactorSet, boolean noDefault) { this(numerator, denominator, unit, thickness, numAlign, denomAlign); this.defFactor = defFactor; this.defFactorSet = defFactorSet; this.noDefault = noDefault; } // Checks if the alignment constant is valid. // If not, a default value will be used. private static int checkAlignment(int align) { if (align == TeXConstants.ALIGN_LEFT || align == TeXConstants.ALIGN_RIGHT) { return align; } return TeXConstants.ALIGN_CENTER; } @Override public Box createBox(TeXEnvironment env) { TeXFont tf = env.getTeXFont(); int style = env.getStyle(); // set thickness to default if default value should be used double drt = tf.getDefaultRuleThickness(style); if (noDefault) { // convert the thickness to pixels thickness *= SpaceAtom.getFactor(unit, env); } else { thickness = (defFactorSet ? defFactor * drt : drt); } // create equal width boxes (in appropriate styles) Box num = (numerator == null ? new StrutBox(0, 0, 0, 0) : numerator.createBox(env.numStyle())); Box denom = (denominator == null ? new StrutBox(0, 0, 0, 0) : denominator.createBox(env.denomStyle())); if (num.getWidth() < denom.getWidth()) { num = new HorizontalBox(num, denom.getWidth(), numAlign); } else { denom = new HorizontalBox(denom, num.getWidth(), denomAlign); } // calculate default shift amounts double shiftUp, shiftDown; if (style < TeXConstants.STYLE_TEXT) { shiftUp = tf.getNum1(style); shiftDown = tf.getDenom1(style); } else { shiftDown = tf.getDenom2(style); if (thickness > 0) { shiftUp = tf.getNum2(style); } else { shiftUp = tf.getNum3(style); } } // upper part of vertical box = numerator VerticalBox vBox = new VerticalBox(); vBox.add(num); // calculate clearance clr, adjust shift amounts and create vertical box double clr, delta, axis = tf.getAxisHeight(style); if (thickness > 0) { // WITH fraction rule // clearance clr if (style < TeXConstants.STYLE_TEXT) { clr = 3 * thickness; } else { clr = thickness; } // adjust shift amounts delta = thickness / 2; double kern1 = shiftUp - num.getDepth() - (axis + delta), kern2 = axis - delta - (denom.getHeight() - shiftDown); double delta1 = clr - kern1, delta2 = clr - kern2; if (delta1 > 0) { shiftUp += delta1; kern1 += delta1; } if (delta2 > 0) { shiftDown += delta2; kern2 += delta2; } // fill vertical box vBox.add(new StrutBox(0, kern1, 0, 0)); vBox.add(new HorizontalRule(thickness, num.getWidth(), 0)); vBox.add(new StrutBox(0, kern2, 0, 0)); } else { // WITHOUT fraction rule // clearance clr if (style < TeXConstants.STYLE_TEXT) { clr = 7 * drt; } else { clr = 3 * drt; } // adjust shift amounts double kern = shiftUp - num.getDepth() - (denom.getHeight() - shiftDown); delta = (clr - kern) / 2; if (delta > 0) { shiftUp += delta; shiftDown += delta; kern += 2 * delta; } // fill vertical box vBox.add(new StrutBox(0, kern, 0, 0)); } // finish vertical box vBox.add(denom); vBox.setHeight(shiftUp + num.getHeight()); vBox.setDepth(shiftDown + denom.getDepth()); // \nulldelimiterspace is set by default to 1.2pt = 0.12em) double f = new SpaceAtom(TeXConstants.UNIT_EM, 0.12f, 0, 0).createBox(env).getWidth(); return new HorizontalBox(vBox, vBox.getWidth() + 2 * f, TeXConstants.ALIGN_CENTER); } public Atom getNumerator() { return numerator; } public Atom getDenominator() { return denominator; } }