需求
按数据层级生成PDF文件,要有目录,目录里要有真实的页码,附件内容用表格显示,每页要有页码,大标题 做为封面当独显示一页,
PDF内容
大标题,
目录(里有对应的页码)
正文 里有 表格
每页还有页码
/// <summary>
/// 生成 pdf 文件
/// </summary>
/// <param name="model">实体对象</param>
/// <param name="caseDetailsViewModels">数据LIST</param>
/// <param name="TargetPath">目标文件路径</param>
/// <param name="NewFileName">文件名</param>
/// <returns></returns>
private static string createPDF(ViewModel model, List<DetailsViewModel> DetailsViewModels, string TargetPath, string NewFileName)
{
string outMergeFile = TargetPath + NewFileName;
if (Directory.Exists(outMergeFile))
{
Directory.Delete(outMergeFile);
}
var pageSize = PageSize.A4;
iTextSharp.text.Document document = new iTextSharp.text.Document(pageSize);
PdfWriter writer = PdfWriter.GetInstance(document, new FileStream(outMergeFile, FileMode.Create));
try
{
document.Open();
#region 字体
Font BF_Light = FontFactory.GetFont(@"C:\Windows\Fonts\simsun.ttc,0", BaseFont.IDENTITY_H, BaseFont.EMBEDDED);
Font bigTitle_Light = FontFactory.GetFont(@"C:\Windows\Fonts\simsun.ttc,0", BaseFont.IDENTITY_H, BaseFont.EMBEDDED);
bigTitle_Light.Size = 20;
Font title_Light = FontFactory.GetFont(@"C:\Windows\Fonts\simsun.ttc,0", BaseFont.IDENTITY_H, BaseFont.EMBEDDED);
title_Light.SetColor(0, 0, 255);
title_Light.Size = 16;
Font body_Light = FontFactory.GetFont(@"C:\Windows\Fonts\simsun.ttc,0", BaseFont.IDENTITY_H, BaseFont.EMBEDDED);
body_Light.Size = 12;
//var titleFont = GetFont();
//titleFont.SetStyle(Font.BOLD);
//titleFont.Color = BaseColor.BLACK;
//titleFont.Size = 16;
//new Font(baseFont2, 20f)
#endregion
//第一页
//段落 大标题
#region 大标题
var titleParagraph = new Paragraph(new Phrase(model.TitleModel.F_ContentStr, bigTitle_Light));
float x = 300;
float y = document.PageSize.Height / 2;
ColumnText.ShowTextAligned(writer.DirectContent, Element.ALIGN_CENTER, titleParagraph, x, y, 0);
#endregion
//第二页
#region 增加目录页
document.NewPage();
var catalogueP = new Paragraph(new Phrase("目录", body_Light));
//catalogueP.IndentationLeft = 160;
//document.Add(catalogueP);
//页头显示的位置
ColumnText.ShowTextAligned(writer.DirectContent, Element.ALIGN_CENTER, catalogueP, document.PageSize.Width / 2, document.Top, 0);
foreach (var item in model.TwoTitleList)
{
Chunk leader = new Chunk(new DottedLineSeparator(), 400);//400
var F_ContentStrTxt = item.F_ContentStr.Trim();
Paragraph pp = new Paragraph(new Phrase(" " + F_ContentStrTxt, BF_Light));
pp.Add(leader);
pp.Add(new Phrase(!string.IsNullOrEmpty(item.F_Page) ? item.F_Page : "1~1", BF_Light));
document.Add(pp);
}
#endregion
//正文
//第三页
#region 正文
int iPageNum = 3;
int beginPageNum = 3;
BaseFont baseFont2 = BaseFont.CreateFont(@"C:\Windows\Fonts\simsun.ttc,0", BaseFont.IDENTITY_H, BaseFont.NOT_EMBEDDED);
document.NewPage();
foreach (var item in model.TwoTitleList)
{
iPageNum = writer.PageNumber;
//标题
document.Add(new Paragraph(new Phrase(item.F_ContentStr, title_Light)));
foreach (var itemThree in item.ThreeContentList)
{
itemThree.F_ContentStr = itemThree.F_ContentStr.Replace("\r\n", string.Empty);
//内容
if (itemThree.F_ContentStr.Contains("</p>"))
{
var contentStr = itemThree.F_ContentStr.Replace("<p>", string.Empty);
var contentList = contentStr.Split("</p>");
foreach (var itemCon in contentList)
{
var itemConTxt = itemCon.Replace("\n", string.Empty).Trim();
var bydyPhrase = new Paragraph(new Phrase(" " + itemConTxt, body_Light));
document.Add(bydyPhrase);
if (writer.PageNumber > iPageNum)
{
writePageNumber(document, writer, BF_Light);
}
}
}
else
{
var txt = itemThree.F_ContentStr.Replace("\n", string.Empty).Trim();
document.Add(new Paragraph(new Phrase(" " + txt, body_Light)));
}
}
//if (writer.PageNumber > iPageNum)
//{
item.F_Page = string.Format("{0}~{1}", iPageNum, writer.PageNumber);
iPageNum = writer.PageNumber;
//}
}
#endregion
#region 附件
if (model.FileList != null && model.FileList.Count > 0)
{
var bydyPhrase1 = new Paragraph(new Phrase(" (一)检索结果分析列表", body_Light));
document.Add(bydyPhrase1);
document.Add(new Paragraph(new Phrase(" ")));
int groupNum = 1;
foreach (var item in model.FileList)
{
var keywordTxt = item.Keyword.Replace("\n", string.Empty).Trim();
var bydyPhrase = new Paragraph(new Phrase(" " + keywordTxt, body_Light));
document.Add(bydyPhrase);
document.Add(new Paragraph(new Phrase(" ")));
PdfPTable table = new PdfPTable(4);
int[] widths = { 40, 20, 20, 20 };
table.SetWidths(widths);//设置每列的宽度(求和百分比5,10=1,2)
//为pdfpTable的构造函数传入整数3,pdfpTable被初始化为一个三列的表格
//PdfPCell cell = new PdfPCell(new Phrase("第一行 跨3列", BF_Light));
//cell.Colspan = 6;//跨3列
//0=Left, 1=Centre, 2=Right
//table.AddCell(cell);
PdfPCell cell = new PdfPCell(new Phrase("标题", BF_Light));
cell.HorizontalAlignment = 1;
table.AddCell(cell);
PdfPCell cel2 = new PdfPCell(new Phrase("案号", BF_Light));
cel2.HorizontalAlignment = 1;
table.AddCell(cel2);
PdfPCell cel3 = new PdfPCell(new Phrase("是否加入检索报告", BF_Light));
cel3.HorizontalAlignment = 1;
table.AddCell(cel3);
PdfPCell cel4 = new PdfPCell(new Phrase("未加入检索报告原因", BF_Light));
cel4.HorizontalAlignment = 1;
table.AddCell(cel4);
foreach (var itemKey in item.KeywordFileList)
{
table.AddCell(new Phrase(itemKey.F_CaseTitle, BF_Light));
table.AddCell(new Phrase(itemKey.F_CaseCode.ToString(), BF_Light));
table.AddCell(new Phrase(itemKey.F_IsAdd ? "是" : "否", BF_Light));
table.AddCell(new Phrase(itemKey.F_NoAddCause, BF_Light));
groupNum++;
}
document.Add(table);
}
}
#endregion
#region 生成案例
if (caseDetailsViewModels != null && caseDetailsViewModels.Count > 0)
{
var bydyPhrase2 = new Paragraph(new Phrase(" (二)检索案例全文", body_Light));
document.Add(bydyPhrase2);
document.Add(new Paragraph(new Phrase(" ")));
foreach (var item in caseDetailsViewModels)
{
foreach (var itemCase in item.CaseItemList)
{
//标题
itemCase.F_Title = itemCase.F_Title.Replace("\r\n", string.Empty);
var F_TitleTxt = itemCase.F_Title.Trim();
document.Add(new Paragraph(new Phrase(" " + F_TitleTxt, body_Light)));
//内容
itemCase.F_Content = itemCase.F_Content.Replace("\r\n", string.Empty);
if (itemCase.F_Content.Contains("</p>"))
{
var contentStr = itemCase.F_Content.Replace("<p>", string.Empty);
var contentList = contentStr.Split("</p>");
foreach (var itemCon in contentList)
{
var itemConTxt = itemCon.Replace("\n", string.Empty).Trim();
var bydyPhrase = new Paragraph(new Phrase(" " + itemConTxt, body_Light));
document.Add(bydyPhrase);
if (writer.PageNumber > iPageNum)
{
writePageNumber(document, writer, BF_Light);
}
}
}
else
{
var F_ContentTxt = itemCase.F_Content.Replace("\n", string.Empty).Trim();
document.Add(new Paragraph(new Phrase(" " + F_ContentTxt, body_Light)));
}
}
}
}
#endregion
}
catch (Exception ex)
{
LogFactory.GetLogger("createReportDetailsPDFCatalogue").Info("createReportDetailsPDFCatalogue 报错" + ex.Message);
}
finally
{
if (document != null)
{
document.Close();
}
if (writer != null)
{
writer.Close();
}
}
return outMergeFile;
}
/// <summary>
/// 写页码
/// </summary>
/// <param name="document"></param>
/// <param name="writer"></param>
/// <param name="BF_Light"></param>
private static void writePageNumber(iTextSharp.text.Document document, PdfWriter writer, Font BF_Light)
{
Phrase header = new Phrase("第" + (writer.PageNumber).ToString() + "页", BF_Light);// - 1
//页脚显示的位置
ColumnText.ShowTextAligned(writer.DirectContent, Element.ALIGN_CENTER, header, document.PageSize.Width / 2, document.Bottom, 0);
}
调用,注意要调用两次,第一次生成时不知道目录里的内容在第几页,第一次在生成时收集页码。这时会生成一个临时文件。第二次生成的目录里的页码才是真实有效的数据。所以要生成两次。
就好比你在写WORD时也是不知道目录里的内容会写在第几次,都是一、二级标题和内容都写完后,再去生成目录的。这里的道理也是一样的。
var fileNameTempPath = createPDF(model, null, uploadPath, fileNameTemp);
var savePath = createPDF(model, caseDetailsViewModels, uploadPath, fileName);
如果有更好的方法请告诉我。