VisualSignature.cs
// 
// このコードは、DioDocs for PDF のサンプルの一部として提供されています。
// Copyright (c) GrapeCity inc. All rights reserved.
// 
using System;
using System.IO;
using System.Drawing;
using GrapeCity.Documents.Pdf;
using GrapeCity.Documents.Pdf.AcroForms;
using GrapeCity.Documents.Text;
using GrapeCity.Documents.Drawing;
using System.Security.Cryptography.X509Certificates;
using GrapeCity.Documents.Pdf.Annotations;

namespace GcPdfWeb.Samples
{
    // このサンプルは、SignatureField と署名画像を使用して、PDF を作成し 
    // .pfx ファイルで署名する方法を示します。
    // また、サンプルは署名されたファイルを別の GcPdfDocument インスタンスに読み込み、
    // 署名を検証します。
    // このサンプルは SignDoc サンプルと同じですが、署名を表す画像を追加します。
    public class VisualSignature
    {
        public int CreatePDF(Stream stream)
        {
            GcPdfDocument doc = new GcPdfDocument();
            Page page = doc.NewPage();
            TextFormat tf = new TextFormat() { Font = StandardFonts.Times, FontSize = 14 };
            page.Graphics.DrawString(
                "電子署名のサンプルです\r\nこのPDFは表示している印影画像に署名されています。",
                tf, new PointF(72, 72));
            page.Graphics.DrawString(
                "注意:署名に使用している証明書、印影はテスト用です。\r\nブラウザのビルトインビューアによっては署名が表示されない場合があります。",
                new TextFormat(tf) { FontSize = 10 }, new PointF(72, 72 * 2));

            // テスト証明書を初期化します。
            var pfxPath = Path.Combine("Resources", "Misc", "GcPdfTest.pfx");
            X509Certificate2 cert = new X509Certificate2(File.ReadAllBytes(pfxPath), "qq",
                X509KeyStorageFlags.MachineKeySet | X509KeyStorageFlags.PersistKeySet | X509KeyStorageFlags.Exportable);
            SignatureProperties sp = new SignatureProperties();
            sp.Certificate = cert;
            sp.Location = "DioDocs Sample Browser";
            sp.SignerName = "株式会社ディオドック";

            // 署名を表す画像を追加します。
            sp.SignatureAppearance.Image = Image.FromFile(Path.Combine("Resources", "ImagesBis", "kaku_in.png"));
            sp.SignatureAppearance.CaptionImageRelation = GrapeCity.Documents.Pdf.Annotations.CaptionImageRelation.ImageOnly;

            // 署名を保持する署名フィールドを初期化します。
            SignatureField sf = new SignatureField();
            sf.Widget.Rect = new RectangleF(72 * 2, 72 * 2.5f, 72 * 2, 72 * 2);
            sf.Widget.Page = page;
            sf.Widget.BackColor = Color.LightSeaGreen;
            sf.Widget.TextFormat.Font = Common.Util.getFont();
            sf.Widget.ButtonAppearance.Caption = $"署名者: {sp.SignerName}\r\n場所: {sp.Location}";
            // 文書に署名フィールドを追加します。
            doc.AcroForm.Fields.Add(sf);
            // 署名フィールドと署名プロパティを結びつけます。
            sp.SignatureField = sf;

            // 署名して文書を保存します。
            // 注意:
            // - 署名と保存は一連の操作であり、2つは分離できません。
            // - Sign() メソッドに渡されたストリームは読み込み可能でなければなりません。
            doc.Sign(sp, stream);

            // ストリームを巻き戻して、作成した文書を別の GcPdfDocument 
            // オブジェクトに読み込み、署名を確認します。
            stream.Seek(0, SeekOrigin.Begin);
            GcPdfDocument doc2 = new GcPdfDocument();
            doc2.Load(stream);
            SignatureField sf2 = (SignatureField)doc2.AcroForm.Fields[0];
            if (!sf2.Value.VerifySignature())
                throw new Exception("Failed to verify the signature");

            // 終了(生成および署名された文書はすでに 'stream' に保存されています)。
            return doc.Pages.Count;
        }
    }
}