using System;
using System.Collections;
using System.util;
using iTextSharp.text;
/*
* $Id: PdfAnnotation.cs,v 1.12 2008/05/24 18:41:23 psoares33 Exp $
*
*
* Copyright 1999, 2000, 2001, 2002 Bruno Lowagie
*
* The contents of this file are subject to the Mozilla Public License Version 1.1
* (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.mozilla.org/MPL/
*
* Software distributed under the License is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
* for the specific language governing rights and limitations under the License.
*
* The Original Code is 'iText, a free JAVA-PDF library'.
*
* The Initial Developer of the Original Code is Bruno Lowagie. Portions created by
* the Initial Developer are Copyright (C) 1999, 2000, 2001, 2002 by Bruno Lowagie.
* All Rights Reserved.
* Co-Developer of the code is Paulo Soares. Portions created by the Co-Developer
* are Copyright (C) 2000, 2001, 2002 by Paulo Soares. All Rights Reserved.
*
* Contributor(s): all the names of the contributors are added in the source code
* where applicable.
*
* Alternatively, the contents of this file may be used under the terms of the
* LGPL license (the "GNU LIBRARY GENERAL PUBLIC LICENSE"), in which case the
* provisions of LGPL are applicable instead of those above. If you wish to
* allow use of your version of this file only under the terms of the LGPL
* License and not to allow others to use your version of this file under
* the MPL, indicate your decision by deleting the provisions above and
* replace them with the notice and other provisions required by the LGPL.
* If you do not delete the provisions above, a recipient may use your version
* of this file under either the MPL or the GNU LIBRARY GENERAL PUBLIC LICENSE.
*
* This library is free software; you can redistribute it and/or modify it
* under the terms of the MPL as stated above or under the terms of the GNU
* Library General Public License as published by the Free Software Foundation;
* either version 2 of the License, or any later version.
*
* This library 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 Library general Public License for more
* details.
*
* If you didn't download this code from the following link, you should check if
* you aren't using an obsolete version:
* http://www.lowagie.com/iText/
*/
namespace iTextSharp.text.pdf {
/**
* A PdfAnnotation
is a note that is associated with a page.
*
* @see PdfDictionary
*/
public class PdfAnnotation : PdfDictionary {
public static readonly PdfName HIGHLIGHT_NONE = PdfName.N;
public static readonly PdfName HIGHLIGHT_INVERT = PdfName.I;
public static readonly PdfName HIGHLIGHT_OUTLINE = PdfName.O;
public static readonly PdfName HIGHLIGHT_PUSH = PdfName.P;
public static readonly PdfName HIGHLIGHT_TOGGLE = PdfName.T;
public const int FLAGS_INVISIBLE = 1;
public const int FLAGS_HIDDEN = 2;
public const int FLAGS_PRINT = 4;
public const int FLAGS_NOZOOM = 8;
public const int FLAGS_NOROTATE = 16;
public const int FLAGS_NOVIEW = 32;
public const int FLAGS_READONLY = 64;
public const int FLAGS_LOCKED = 128;
public const int FLAGS_TOGGLENOVIEW = 256;
public static readonly PdfName APPEARANCE_NORMAL = PdfName.N;
public static readonly PdfName APPEARANCE_ROLLOVER = PdfName.R;
public static readonly PdfName APPEARANCE_DOWN = PdfName.D;
public static readonly PdfName AA_ENTER = PdfName.E;
public static readonly PdfName AA_EXIT = PdfName.X;
public static readonly PdfName AA_DOWN = PdfName.D;
public static readonly PdfName AA_UP = PdfName.U;
public static readonly PdfName AA_FOCUS = PdfName.FO;
public static readonly PdfName AA_BLUR = PdfName.BL;
public static readonly PdfName AA_JS_KEY = PdfName.K;
public static readonly PdfName AA_JS_FORMAT = PdfName.F;
public static readonly PdfName AA_JS_CHANGE = PdfName.V;
public static readonly PdfName AA_JS_OTHER_CHANGE = PdfName.C;
public const int MARKUP_HIGHLIGHT = 0;
public const int MARKUP_UNDERLINE = 1;
public const int MARKUP_STRIKEOUT = 2;
/** attributevalue */
public const int MARKUP_SQUIGGLY = 3;
protected internal PdfWriter writer;
protected internal PdfIndirectReference reference;
protected internal Hashtable templates;
protected internal bool form = false;
protected internal bool annotation = true;
/** Holds value of property used. */
protected internal bool used = false;
/** Holds value of property placeInPage. */
private int placeInPage = -1;
// constructors
public PdfAnnotation(PdfWriter writer, Rectangle rect) {
this.writer = writer;
if (rect != null)
Put(PdfName.RECT, new PdfRectangle(rect));
}
/**
* Constructs a new PdfAnnotation
of subtype text.
*/
public PdfAnnotation(PdfWriter writer, float llx, float lly, float urx, float ury, PdfString title, PdfString content) {
this.writer = writer;
Put(PdfName.SUBTYPE, PdfName.TEXT);
Put(PdfName.T, title);
Put(PdfName.RECT, new PdfRectangle(llx, lly, urx, ury));
Put(PdfName.CONTENTS, content);
}
/**
* Constructs a new PdfAnnotation
of subtype link (Action).
*/
public PdfAnnotation(PdfWriter writer, float llx, float lly, float urx, float ury, PdfAction action) {
this.writer = writer;
Put(PdfName.SUBTYPE, PdfName.LINK);
Put(PdfName.RECT, new PdfRectangle(llx, lly, urx, ury));
Put(PdfName.A, action);
Put(PdfName.BORDER, new PdfBorderArray(0, 0, 0));
Put(PdfName.C, new PdfColor(0x00, 0x00, 0xFF));
}
/**
* Creates a screen PdfAnnotation
* @param writer
* @param rect
* @param clipTitle
* @param fs
* @param mimeType
* @param playOnDisplay
* @return a screen PdfAnnotation
* @throws IOException
*/
public static PdfAnnotation CreateScreen(PdfWriter writer, Rectangle rect, String clipTitle, PdfFileSpecification fs,
String mimeType, bool playOnDisplay) {
PdfAnnotation ann = new PdfAnnotation(writer, rect);
ann.Put(PdfName.SUBTYPE, PdfName.SCREEN);
ann.Put (PdfName.F, new PdfNumber(FLAGS_PRINT));
ann.Put(PdfName.TYPE, PdfName.ANNOT);
ann.SetPage();
PdfIndirectReference refi = ann.IndirectReference;
PdfAction action = PdfAction.Rendition(clipTitle,fs,mimeType, refi);
PdfIndirectReference actionRef = writer.AddToBody(action).IndirectReference;
// for play on display add trigger event
if (playOnDisplay)
{
PdfDictionary aa = new PdfDictionary();
aa.Put(new PdfName("PV"), actionRef);
ann.Put(PdfName.AA, aa);
}
ann.Put(PdfName.A, actionRef);
return ann;
}
public PdfIndirectReference IndirectReference {
get {
if (reference == null) {
reference = writer.PdfIndirectReference;
}
return reference;
}
}
public static PdfAnnotation CreateText(PdfWriter writer, Rectangle rect, string title, string contents, bool open, string icon) {
PdfAnnotation annot = new PdfAnnotation(writer, rect);
annot.Put(PdfName.SUBTYPE, PdfName.TEXT);
if (title != null)
annot.Put(PdfName.T, new PdfString(title, PdfObject.TEXT_UNICODE));
if (contents != null)
annot.Put(PdfName.CONTENTS, new PdfString(contents, PdfObject.TEXT_UNICODE));
if (open)
annot.Put(PdfName.OPEN, PdfBoolean.PDFTRUE);
if (icon != null) {
annot.Put(PdfName.NAME, new PdfName(icon));
}
return annot;
}
protected static PdfAnnotation CreateLink(PdfWriter writer, Rectangle rect, PdfName highlight) {
PdfAnnotation annot = new PdfAnnotation(writer, rect);
annot.Put(PdfName.SUBTYPE, PdfName.LINK);
if (!highlight.Equals(HIGHLIGHT_INVERT))
annot.Put(PdfName.H, highlight);
return annot;
}
public static PdfAnnotation CreateLink(PdfWriter writer, Rectangle rect, PdfName highlight, PdfAction action) {
PdfAnnotation annot = CreateLink(writer, rect, highlight);
annot.PutEx(PdfName.A, action);
return annot;
}
public static PdfAnnotation CreateLink(PdfWriter writer, Rectangle rect, PdfName highlight, string namedDestination) {
PdfAnnotation annot = CreateLink(writer, rect, highlight);
annot.Put(PdfName.DEST, new PdfString(namedDestination));
return annot;
}
public static PdfAnnotation CreateLink(PdfWriter writer, Rectangle rect, PdfName highlight, int page, PdfDestination dest) {
PdfAnnotation annot = CreateLink(writer, rect, highlight);
PdfIndirectReference piref = writer.GetPageReference(page);
dest.AddPage(piref);
annot.Put(PdfName.DEST, dest);
return annot;
}
public static PdfAnnotation CreateFreeText(PdfWriter writer, Rectangle rect, string contents, PdfContentByte defaultAppearance) {
PdfAnnotation annot = new PdfAnnotation(writer, rect);
annot.Put(PdfName.SUBTYPE, PdfName.FREETEXT);
annot.Put(PdfName.CONTENTS, new PdfString(contents, PdfObject.TEXT_UNICODE));
annot.DefaultAppearanceString = defaultAppearance;
return annot;
}
public static PdfAnnotation CreateLine(PdfWriter writer, Rectangle rect, string contents, float x1, float y1, float x2, float y2) {
PdfAnnotation annot = new PdfAnnotation(writer, rect);
annot.Put(PdfName.SUBTYPE, PdfName.LINE);
annot.Put(PdfName.CONTENTS, new PdfString(contents, PdfObject.TEXT_UNICODE));
PdfArray array = new PdfArray(new PdfNumber(x1));
array.Add(new PdfNumber(y1));
array.Add(new PdfNumber(x2));
array.Add(new PdfNumber(y2));
annot.Put(PdfName.L, array);
return annot;
}
public static PdfAnnotation CreateSquareCircle(PdfWriter writer, Rectangle rect, string contents, bool square) {
PdfAnnotation annot = new PdfAnnotation(writer, rect);
if (square)
annot.Put(PdfName.SUBTYPE, PdfName.SQUARE);
else
annot.Put(PdfName.SUBTYPE, PdfName.CIRCLE);
annot.Put(PdfName.CONTENTS, new PdfString(contents, PdfObject.TEXT_UNICODE));
return annot;
}
public static PdfAnnotation CreateMarkup(PdfWriter writer, Rectangle rect, string contents, int type, float[] quadPoints) {
PdfAnnotation annot = new PdfAnnotation(writer, rect);
PdfName name = PdfName.HIGHLIGHT;
switch (type) {
case MARKUP_UNDERLINE:
name = PdfName.UNDERLINE;
break;
case MARKUP_STRIKEOUT:
name = PdfName.STRIKEOUT;
break;
case MARKUP_SQUIGGLY:
name = PdfName.SQUIGGLY;
break;
}
annot.Put(PdfName.SUBTYPE, name);
annot.Put(PdfName.CONTENTS, new PdfString(contents, PdfObject.TEXT_UNICODE));
PdfArray array = new PdfArray();
for (int k = 0; k < quadPoints.Length; ++k)
array.Add(new PdfNumber(quadPoints[k]));
annot.Put(PdfName.QUADPOINTS, array);
return annot;
}
public static PdfAnnotation CreateStamp(PdfWriter writer, Rectangle rect, string contents, string name) {
PdfAnnotation annot = new PdfAnnotation(writer, rect);
annot.Put(PdfName.SUBTYPE, PdfName.STAMP);
annot.Put(PdfName.CONTENTS, new PdfString(contents, PdfObject.TEXT_UNICODE));
annot.Put(PdfName.NAME, new PdfName(name));
return annot;
}
public static PdfAnnotation CreateInk(PdfWriter writer, Rectangle rect, string contents, float[][] inkList) {
PdfAnnotation annot = new PdfAnnotation(writer, rect);
annot.Put(PdfName.SUBTYPE, PdfName.INK);
annot.Put(PdfName.CONTENTS, new PdfString(contents, PdfObject.TEXT_UNICODE));
PdfArray outer = new PdfArray();
for (int k = 0; k < inkList.Length; ++k) {
PdfArray inner = new PdfArray();
float[] deep = inkList[k];
for (int j = 0; j < deep.Length; ++j)
inner.Add(new PdfNumber(deep[j]));
outer.Add(inner);
}
annot.Put(PdfName.INKLIST, outer);
return annot;
}
/** Creates a file attachment annotation.
* @param writer the PdfWriter
* @param rect the dimensions in the page of the annotation
* @param contents the file description
* @param fileStore an array with the file. If it's null
* the file will be read from the disk
* @param file the path to the file. It will only be used if
* fileStore
is not null
* @param fileDisplay the actual file name stored in the pdf
* @throws IOException on error
* @return the annotation
*/
public static PdfAnnotation CreateFileAttachment(PdfWriter writer, Rectangle rect, String contents, byte[] fileStore, String file, String fileDisplay) {
return CreateFileAttachment(writer, rect, contents, PdfFileSpecification.FileEmbedded(writer, file, fileDisplay, fileStore));
}
/** Creates a file attachment annotation
* @param writer
* @param rect
* @param contents
* @param fs
* @return the annotation
* @throws IOException
*/
public static PdfAnnotation CreateFileAttachment(PdfWriter writer, Rectangle rect, String contents, PdfFileSpecification fs) {
PdfAnnotation annot = new PdfAnnotation(writer, rect);
annot.Put(PdfName.SUBTYPE, PdfName.FILEATTACHMENT);
if (contents != null)
annot.Put(PdfName.CONTENTS, new PdfString(contents, PdfObject.TEXT_UNICODE));
annot.Put(PdfName.FS, fs.Reference);
return annot;
}
public static PdfAnnotation CreatePopup(PdfWriter writer, Rectangle rect, string contents, bool open) {
PdfAnnotation annot = new PdfAnnotation(writer, rect);
annot.Put(PdfName.SUBTYPE, PdfName.POPUP);
if (contents != null)
annot.Put(PdfName.CONTENTS, new PdfString(contents, PdfObject.TEXT_UNICODE));
if (open)
annot.Put(PdfName.OPEN, PdfBoolean.PDFTRUE);
return annot;
}
public PdfContentByte DefaultAppearanceString {
set {
byte[] b = value.InternalBuffer.ToByteArray();
int len = b.Length;
for (int k = 0; k < len; ++k) {
if (b[k] == '\n')
b[k] = 32;
}
Put(PdfName.DA, new PdfString(b));
}
}
public int Flags {
set {
if (value == 0)
Remove(PdfName.F);
else
Put(PdfName.F, new PdfNumber(value));
}
}
public PdfBorderArray Border {
set {
Put(PdfName.BORDER, value);
}
}
public PdfBorderDictionary BorderStyle {
set {
Put(PdfName.BS, value);
}
}
/**
* Sets the annotation's highlighting mode. The values can be
* HIGHLIGHT_NONE
, HIGHLIGHT_INVERT
,
* HIGHLIGHT_OUTLINE
and HIGHLIGHT_PUSH
;
* @param highlight the annotation's highlighting mode
*/
public void SetHighlighting(PdfName highlight) {
if (highlight.Equals(HIGHLIGHT_INVERT))
Remove(PdfName.H);
else
Put(PdfName.H, highlight);
}
public void SetAppearance(PdfName ap, PdfTemplate template) {
PdfDictionary dic = (PdfDictionary)Get(PdfName.AP);
if (dic == null)
dic = new PdfDictionary();
dic.Put(ap, template.IndirectReference);
Put(PdfName.AP, dic);
if (!form)
return;
if (templates == null)
templates = new Hashtable();
templates[template] = null;
}
public void SetAppearance(PdfName ap, string state, PdfTemplate template) {
PdfDictionary dicAp = (PdfDictionary)Get(PdfName.AP);
if (dicAp == null)
dicAp = new PdfDictionary();
PdfDictionary dic;
PdfObject obj = dicAp.Get(ap);
if (obj != null && obj.IsDictionary())
dic = (PdfDictionary)obj;
else
dic = new PdfDictionary();
dic.Put(new PdfName(state), template.IndirectReference);
dicAp.Put(ap, dic);
Put(PdfName.AP, dicAp);
if (!form)
return;
if (templates == null)
templates = new Hashtable();
templates[template] = null;
}
public string AppearanceState {
set {
if (value == null) {
Remove(PdfName.AS);
return;
}
Put(PdfName.AS, new PdfName(value));
}
}
public Color Color {
set {
Put(PdfName.C, new PdfColor(value));
}
}
public string Title {
set {
if (value == null) {
Remove(PdfName.T);
return;
}
Put(PdfName.T, new PdfString(value, PdfObject.TEXT_UNICODE));
}
}
public PdfAnnotation Popup {
set {
Put(PdfName.POPUP, value.IndirectReference);
value.Put(PdfName.PARENT, this.IndirectReference);
}
}
public PdfAction Action {
set {
Put(PdfName.A, value);
}
}
public void SetAdditionalActions(PdfName key, PdfAction action) {
PdfDictionary dic;
PdfObject obj = Get(PdfName.AA);
if (obj != null && obj.IsDictionary())
dic = (PdfDictionary)obj;
else
dic = new PdfDictionary();
dic.Put(key, action);
Put(PdfName.AA, dic);
}
internal virtual bool IsUsed() {
return used;
}
public virtual void SetUsed() {
used = true;
}
public Hashtable Templates {
get {
return templates;
}
}
/** Getter for property form.
* @return Value of property form.
*/
public bool IsForm() {
return form;
}
/** Getter for property annotation.
* @return Value of property annotation.
*/
public bool IsAnnotation() {
return annotation;
}
public int Page {
set {
Put(PdfName.P, writer.GetPageReference(value));
}
}
public void SetPage() {
Put(PdfName.P, writer.CurrentPage);
}
/** Getter for property placeInPage.
* @return Value of property placeInPage.
*/
public int PlaceInPage {
get {
return placeInPage;
}
set {
this.placeInPage = value;
}
}
public static PdfAnnotation ShallowDuplicate(PdfAnnotation annot) {
PdfAnnotation dup;
if (annot.IsForm()) {
dup = new PdfFormField(annot.writer);
PdfFormField dupField = (PdfFormField)dup;
PdfFormField srcField = (PdfFormField)annot;
dupField.parent = srcField.parent;
dupField.kids = srcField.kids;
}
else
dup = new PdfAnnotation(annot.writer, null);
dup.Merge(annot);
dup.form = annot.form;
dup.annotation = annot.annotation;
dup.templates = annot.templates;
return dup;
}
public int Rotate {
set {
Put(PdfName.ROTATE, new PdfNumber(value));
}
}
internal PdfDictionary MK {
get {
PdfDictionary mk = (PdfDictionary)Get(PdfName.MK);
if (mk == null) {
mk = new PdfDictionary();
Put(PdfName.MK, mk);
}
return mk;
}
}
public int MKRotation {
set {
MK.Put(PdfName.R, new PdfNumber(value));
}
}
public static PdfArray GetMKColor(Color color) {
PdfArray array = new PdfArray();
int type = ExtendedColor.GetType(color);
switch (type) {
case ExtendedColor.TYPE_GRAY: {
array.Add(new PdfNumber(((GrayColor)color).Gray));
break;
}
case ExtendedColor.TYPE_CMYK: {
CMYKColor cmyk = (CMYKColor)color;
array.Add(new PdfNumber(cmyk.Cyan));
array.Add(new PdfNumber(cmyk.Magenta));
array.Add(new PdfNumber(cmyk.Yellow));
array.Add(new PdfNumber(cmyk.Black));
break;
}
case ExtendedColor.TYPE_SEPARATION:
case ExtendedColor.TYPE_PATTERN:
case ExtendedColor.TYPE_SHADING:
throw new Exception("Separations, patterns and shadings are not allowed in MK dictionary.");
default:
array.Add(new PdfNumber(color.R / 255f));
array.Add(new PdfNumber(color.G / 255f));
array.Add(new PdfNumber(color.B / 255f));
break;
}
return array;
}
public Color MKBorderColor {
set {
if (value == null)
MK.Remove(PdfName.BC);
else
MK.Put(PdfName.BC, GetMKColor(value));
}
}
public Color MKBackgroundColor {
set {
if (value == null)
MK.Remove(PdfName.BG);
else
MK.Put(PdfName.BG, GetMKColor(value));
}
}
public string MKNormalCaption {
set {
MK.Put(PdfName.CA, new PdfString(value, PdfObject.TEXT_UNICODE));
}
}
public string MKRolloverCaption {
set {
MK.Put(PdfName.RC, new PdfString(value, PdfObject.TEXT_UNICODE));
}
}
public string MKAlternateCaption {
set {
MK.Put(PdfName.AC, new PdfString(value, PdfObject.TEXT_UNICODE));
}
}
public PdfTemplate MKNormalIcon {
set {
MK.Put(PdfName.I, value.IndirectReference);
}
}
public PdfTemplate MKRolloverIcon {
set {
MK.Put(PdfName.RI, value.IndirectReference);
}
}
public PdfTemplate MKAlternateIcon {
set {
MK.Put(PdfName.IX, value.IndirectReference);
}
}
public void SetMKIconFit(PdfName scale, PdfName scalingType, float leftoverLeft, float leftoverBottom, bool fitInBounds) {
PdfDictionary dic = new PdfDictionary();
if (!scale.Equals(PdfName.A))
dic.Put(PdfName.SW, scale);
if (!scalingType.Equals(PdfName.P))
dic.Put(PdfName.S, scalingType);
if (leftoverLeft != 0.5f || leftoverBottom != 0.5f) {
PdfArray array = new PdfArray(new PdfNumber(leftoverLeft));
array.Add(new PdfNumber(leftoverBottom));
dic.Put(PdfName.A, array);
}
if (fitInBounds)
dic.Put(PdfName.FB, PdfBoolean.PDFTRUE);
MK.Put(PdfName.IF, dic);
}
public int MKTextPosition {
set {
MK.Put(PdfName.TP, new PdfNumber(value));
}
}
/**
* Sets the layer this annotation belongs to.
* @param layer the layer this annotation belongs to
*/
public IPdfOCG Layer {
set {
Put(PdfName.OC, value.Ref);
}
}
/**
* Sets the name of the annotation.
* With this name the annotation can be identified among
* all the annotations on a page (it has to be unique).
*/
public String Name {
set {
Put(PdfName.NM, new PdfString(value));
}
}
/**
* This class processes links from imported pages so that they may be active. The following example code reads a group
* of files and places them all on the output PDF, four pages in a single page, keeping the links active.
*
* String[] files = new String[] {"input1.pdf", "input2.pdf"}; * String outputFile = "output.pdf"; * int firstPage=1; * Document document = new Document(); * PdfWriter writer = PdfWriter.GetInstance(document, new FileOutputStream(outputFile)); * document.SetPageSize(PageSize.A4); * float W = PageSize.A4.GetWidth() / 2; * float H = PageSize.A4.GetHeight() / 2; * document.Open(); * PdfContentByte cb = writer.GetDirectContent(); * for (int i = 0; i < files.length; i++) { * PdfReader currentReader = new PdfReader(files[i]); * currentReader.ConsolidateNamedDestinations(); * for (int page = 1; page <= currentReader.GetNumberOfPages(); page++) { * PdfImportedPage importedPage = writer.GetImportedPage(currentReader, page); * float a = 0.5f; * float e = (page % 2 == 0) ? W : 0; * float f = (page % 4 == 1 || page % 4 == 2) ? H : 0; * ArrayList links = currentReader.GetLinks(page); * cb.AddTemplate(importedPage, a, 0, 0, a, e, f); * for (int j = 0; j < links.Size(); j++) { * PdfAnnotation.PdfImportedLink link = (PdfAnnotation.PdfImportedLink)links.Get(j); * if (link.IsInternal()) { * int dPage = link.GetDestinationPage(); * int newDestPage = (dPage-1)/4 + firstPage; * float ee = (dPage % 2 == 0) ? W : 0; * float ff = (dPage % 4 == 1 || dPage % 4 == 2) ? H : 0; * link.SetDestinationPage(newDestPage); * link.TransformDestination(a, 0, 0, a, ee, ff); * } * link.TransformRect(a, 0, 0, a, e, f); * writer.AddAnnotation(link.CreateAnnotation(writer)); * } * if (page % 4 == 0) * document.NewPage(); * } * if (i < files.length - 1) * document.NewPage(); * firstPage += (currentReader.GetNumberOfPages()+3)/4; * } * document.Close(); **/ public class PdfImportedLink { float llx, lly, urx, ury; Hashtable parameters; PdfArray destination = null; int newPage=0; internal PdfImportedLink(PdfDictionary annotation) { parameters = (Hashtable)annotation.hashMap.Clone(); try { destination = (PdfArray)parameters[PdfName.DEST]; parameters.Remove(PdfName.DEST); } catch (Exception) { throw new ArgumentException("You have to consolidate the named destinations of your reader."); } if (destination != null) { destination = new PdfArray(destination); } PdfArray rc = (PdfArray)parameters[PdfName.RECT]; parameters.Remove(PdfName.RECT); llx = rc.GetAsNumber(0).FloatValue; lly = rc.GetAsNumber(1).FloatValue; urx = rc.GetAsNumber(2).FloatValue; ury = rc.GetAsNumber(3).FloatValue; } public bool IsInternal() { return destination != null; } public int GetDestinationPage() { if (!IsInternal()) return 0; // here destination is something like // [132 0 R, /XYZ, 29.3898, 731.864502, null] PdfIndirectReference refi = destination.GetAsIndirectObject(0); PRIndirectReference pr = (PRIndirectReference) refi; PdfReader r = pr.Reader; for (int i = 1; i <= r.NumberOfPages; i++) { PRIndirectReference pp = r.GetPageOrigRef(i); if (pp.Generation == pr.Generation && pp.Number == pr.Number) return i; } throw new ArgumentException("Page not found."); } public void SetDestinationPage(int newPage) { if (!IsInternal()) throw new ArgumentException("Cannot change destination of external link"); this.newPage=newPage; } public void TransformDestination(float a, float b, float c, float d, float e, float f) { if (!IsInternal()) throw new ArgumentException("Cannot change destination of external link"); if (destination.GetAsName(1).Equals(PdfName.XYZ)) { float x = destination.GetAsNumber(2).FloatValue; float y = destination.GetAsNumber(3).FloatValue; float xx = x * a + y * c + e; float yy = x * b + y * d + f; destination.ArrayList[2] = new PdfNumber(xx); destination.ArrayList[3] = new PdfNumber(yy); } } public void TransformRect(float a, float b, float c, float d, float e, float f) { float x = llx * a + lly * c + e; float y = llx * b + lly * d + f; llx = x; lly = y; x = urx * a + ury * c + e; y = urx * b + ury * d + f; urx = x; ury = y; } public PdfAnnotation CreateAnnotation(PdfWriter writer) { PdfAnnotation annotation = new PdfAnnotation(writer, new Rectangle(llx, lly, urx, ury)); if (newPage != 0) { PdfIndirectReference refi = writer.GetPageReference(newPage); destination.ArrayList[0] = refi; } if (destination != null) annotation.Put(PdfName.DEST, destination); foreach (object key in parameters.Keys) annotation.hashMap[key] = parameters[key]; return annotation; } } } }