Package org.pdfclown.documents.contents.objects

Source Code of org.pdfclown.documents.contents.objects.ShowText$IScanner

/*
  Copyright 2007-2010 Stefano Chizzolini. http://www.pdfclown.org

  Contributors:
    * Stefano Chizzolini (original code developer, http://www.stefanochizzolini.it)

  This file should be part of the source code distribution of "PDF Clown library"
  (the Program): see the accompanying README files for more info.

  This Program is free software; you can redistribute it and/or modify it under the terms
  of the GNU Lesser General Public License as published by the Free Software Foundation;
  either version 3 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,
  either expressed or implied; without even the implied warranty of MERCHANTABILITY or
  FITNESS FOR A PARTICULAR PURPOSE. See the License for more details.

  You should have received a copy of the GNU Lesser General Public License along with this
  Program (see README files); if not, go to the GNU website (http://www.gnu.org/licenses/).

  Redistribution and use, with or without modification, are permitted provided that such
  redistributions retain the above copyright notice, license and disclaimer, along with
  this list of conditions.
*/

package org.pdfclown.documents.contents.objects;

import java.awt.geom.AffineTransform;
import java.awt.geom.Rectangle2D;
import java.util.Arrays;
import java.util.List;

import org.pdfclown.PDF;
import org.pdfclown.VersionEnum;
import org.pdfclown.documents.contents.ContentScanner;
import org.pdfclown.documents.contents.ContentScanner.GraphicsState;
import org.pdfclown.documents.contents.IContentContext;
import org.pdfclown.documents.contents.fonts.Font;
import org.pdfclown.objects.PdfDirectObject;

/**
  Abstract 'show a text string' operation [PDF:1.6:5.3.2].

  @author Stefano Chizzolini (http://www.stefanochizzolini.it)
  @since 0.0.8
  @version 0.1.0
*/
@PDF(VersionEnum.PDF10)
public abstract class ShowText
  extends Operation
{
  // <class>
  // <interfaces>
  public interface IScanner
  {
    /**
      Notifies the scanner about a text character.

      @param textChar Scanned character.
      @param textCharBox Bounding box of the scanned character.
    */
    void scanChar(
      char textChar,
      Rectangle2D textCharBox
      );
  }
  // </interfaces>

  // <dynamic>
  // <constructors>
  protected ShowText(
    String operator
    )
  {super(operator);}

  protected ShowText(
    String operator,
    PdfDirectObject... operands
    )
  {super(operator, operands);}

  protected ShowText(
    String operator,
    List<PdfDirectObject> operands
    )
  {super(operator, operands);}
  // </constructors>

  // <interface>
  // <public>
  /**
    Gets the encoded text.
    <h3>Remarks</h3>
    <p>Text is expressed in native encoding: to resolve it to Unicode, pass it
    to the decode method of the corresponding font.</p>
  */
  public abstract byte[] getText(
    );

  /**
    Gets the encoded text elements along with their adjustments.
    <h3>Remarks</h3>
    <p>Text is expressed in native encoding: to resolve it to Unicode, pass it
    to the decode method of the corresponding font.</p>

    @return Each element can be either a byte array or a number:
      if it's a byte array (encoded text), the operator shows text glyphs;
      if it's a number (glyph adjustment), the operator inversely adjusts the next glyph position by that amount
      (that is: a positive value reduces the distance between consecutive glyphs).
  */
  public List<Object> getValue(
    )
  {return Arrays.asList((Object)getText());}

  @Override
  public void scan(
    GraphicsState state
    )
  {scan(state, null);}

  /**
    Executes scanning on this operation.

    @param state Graphics state context.
    @param textScanner Scanner to be notified about text contents.
      In case it's null, the operation is applied to the graphics state context.
  */
  public void scan(
    ContentScanner.GraphicsState state,
    IScanner textScanner
    )
  {
    /*
      TODO: I really dislike this solution -- it's a temporary hack until the event-driven
      parsing mechanism is implemented...
     */
    /*
      TODO: support to vertical writing mode.
    */

    IContentContext context = state.getScanner().getContentContext();
    double contextHeight = context.getBox().getHeight();
    Font font = state.getFont();
    float fontSize = state.getFontSize();
    float scale = state.getScale() / 100;
    float scaledFactor = Font.getScalingFactor(fontSize) * scale;
    float wordSpace = state.getWordSpace() * scale;
    float charSpace = state.getCharSpace() * scale;
    AffineTransform ctm = (AffineTransform)state.getCtm().clone();
    AffineTransform tm;
    if(this instanceof ShowTextToNextLine)
    {
      ShowTextToNextLine showTextToNextLine = (ShowTextToNextLine)this;
      Float newWordSpace = showTextToNextLine.getWordSpace();
      if(newWordSpace != null)
      {
        if(textScanner == null)
        {state.setWordSpace(newWordSpace);}
        wordSpace = newWordSpace * scale;
      }
      Float newCharSpace = showTextToNextLine.getCharSpace();
      if(newCharSpace != null)
      {
        if(textScanner == null)
        {state.setCharSpace(newCharSpace);}
        charSpace = newCharSpace * scale;
      }
      tm = (AffineTransform)state.getTlm().clone();
      tm.translate(0, state.getLead());
    }
    else
    {tm = (AffineTransform)state.getTm().clone();}

    for(Object textElement : getValue())
    {
      if(textElement instanceof byte[]) // Text string.
      {
        String textString = font.decode((byte[])textElement);
        for(char textChar : textString.toCharArray())
        {
          float charWidth = font.getWidth(textChar) * scaledFactor;

          if(textScanner != null)
          {
            /*
              NOTE: The text rendering matrix is recomputed before each glyph is painted
              during a text-showing operation.
            */
            AffineTransform trm = (AffineTransform)ctm.clone(); trm.concatenate(tm);
            float charHeight = font.getHeight(textChar,fontSize);
            Rectangle2D charBox = new Rectangle2D.Double(
              trm.getTranslateX(),
              contextHeight - trm.getTranslateY() - font.getAscent(fontSize) * tm.getScaleY(),
              charWidth * tm.getScaleX(),
              charHeight * tm.getScaleY()
              );
            textScanner.scanChar(textChar,charBox);
          }

          /*
            NOTE: After the glyph is painted, the text matrix is updated
            according to the glyph displacement and any applicable spacing parameter.
          */
          tm.translate(charWidth + charSpace + (textChar == ' ' ? wordSpace : 0), 0);
        }
      }
      else // Text position adjustment.
      {tm.translate(-((Number)textElement).floatValue() * scaledFactor, 0);}
    }

    if(textScanner == null)
    {
      state.setTm(tm);

      if(this instanceof ShowTextToNextLine)
      {state.setTlm((AffineTransform)tm.clone());}
    }
  }

  /**
    @see #getText()
  */
  public abstract void setText(
    byte[] value
    );

  /**
    @see #getValue()
  */
  public void setValue(
    List<Object> value
    )
  {setText((byte[])value.get(0));}
  // </public>
  // </interface>
  // </dynamic>
  // </class>
}
TOP

Related Classes of org.pdfclown.documents.contents.objects.ShowText$IScanner

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.