public
static
void
SignPdf(
string
fileName,
string
certThumbprint)
{
var
store =
new
X509Store(StoreName.My, StoreLocation.CurrentUser);
store.Open(OpenFlags.ReadOnly);
var
certs = store.Certificates.Find(X509FindType.FindByThumbprint, GetCertThumbprint(certThumbprint),
true
);
store.Close();
if
(certs.Count == 0)
{
MessageBox.Show(
"Nelze najít určený certifikát v Current user certificate store!"
,
"Sign PDF"
, MessageBoxButtons.OK, MessageBoxIcon.Warning);
return
;
}
byte
[] pdfData = File.ReadAllBytes(fileName);
byte
[] signedData = SignDocument(pdfData, certs[0]);
File.WriteAllBytes(fileName, signedData);
}
private
static
string
GetCertThumbprint(
string
certThumbprint)
{
string
thumbprint = certThumbprint.Replace(
" "
,
""
).ToUpperInvariant();
if
(thumbprint[0] == 8206)
{
thumbprint = thumbprint.Substring(1);
}
return
thumbprint;
}
private
static
byte
[] SignDocument(
byte
[] pdfData, X509Certificate2 cert)
{
using
(MemoryStream stream =
new
MemoryStream())
{
var
reader =
new
PdfReader(pdfData);
var
stp = PdfStamper.CreateSignature(reader, stream,
'\0'
);
var
sap = stp.SignatureAppearance;
stp.SetEncryption(
null
,
Guid.NewGuid().ToByteArray(),
PdfWriter.ALLOW_PRINTING | PdfWriter.ALLOW_COPY | PdfWriter.ALLOW_SCREENREADERS,
PdfWriter.ENCRYPTION_AES_256);
var
cp =
new
Org.BouncyCastle.X509.X509CertificateParser();
var
certChain =
new
Org.BouncyCastle.X509.X509Certificate[] { cp.ReadCertificate(cert.RawData) };
sap.SetCrypto(
null
, certChain,
null
, PdfSignatureAppearance.WINCER_SIGNED);
BaseFont helvetica = BaseFont.CreateFont(BaseFont.HELVETICA, BaseFont.CP1250, BaseFont.EMBEDDED);
Font font =
new
Font(helvetica, 12, iTextSharp.text.Font.NORMAL);
sap.Layer2Font = font;
sap.SetVisibleSignature(
new
iTextSharp.text.Rectangle(415, 100, 585, 40), 1,
null
);
var
dic =
new
PdfSignature(PdfName.ADOBE_PPKMS, PdfName.ADBE_PKCS7_SHA1);
dic.Date =
new
PdfDate(sap.SignDate);
dic.Name = cert.Subject;
if
(sap.Reason !=
null
)
{
dic.Reason = sap.Reason;
}
if
(sap.Location !=
null
)
{
dic.Location = sap.Location;
}
sap.CryptoDictionary = dic;
int
csize = 4096;
var
reservedSpace =
new
Dictionary<PdfName,
int
>();
reservedSpace[PdfName.CONTENTS] = csize * 2 + 2;
sap.PreClose(reservedSpace);
HashAlgorithm sha =
new
SHA1CryptoServiceProvider();
var
sapStream = sap.RangeStream;
int
read = 0;
byte
[] buff =
new
byte
[8192];
while
((read = sapStream.Read(buff, 0, 8192)) > 0)
{
sha.TransformBlock(buff, 0, read, buff, 0);
}
sha.TransformFinalBlock(buff, 0, 0);
byte
[] pk = SignMsg(sha.Hash, cert,
false
);
byte
[] outc =
new
byte
[csize];
Array.Copy(pk, 0, outc, 0, pk.Length);
PdfDictionary certificateDictionary =
new
PdfDictionary();
certificateDictionary.Put(PdfName.CONTENTS,
new
PdfString(outc).SetHexWriting(
true
));
sap.Close(certificateDictionary);
stp.Close();
reader.Close();
return
stream.GetBuffer();
}
}
private
static
byte
[] SignMsg(Byte[] msg, X509Certificate2 cert,
bool
detached)
{
ContentInfo contentInfo =
new
ContentInfo(msg);
SignedCms signedCms =
new
SignedCms(contentInfo, detached);
CmsSigner cmsSigner =
new
CmsSigner(cert);
cmsSigner.IncludeOption = X509IncludeOption.ExcludeRoot;
signedCms.ComputeSignature(cmsSigner,
false
);
return
signedCms.Encode();
}