まず、いくつかのコンテキスト:
- 言語 - C#
- プラットフォーム - .Net Framework 4.5
- プロジェクトの種類 - ASP.Net MVC 4
MVC プロジェクトのどのビューが次のメソッドへの明示的な呼び出しを処理しているかを判断しようとしています。メソッドの MSDN ドキュメントはこちら: http://msdn.microsoft.com/EN-US/library/dd492930.aspx
protected internal ViewResult View(
Object model
)
元の作成者はビューを使用して、サードパーティ ライブラリで PDF ファイルを生成しています。ビューを変更して追加情報を含める必要があります。
問題: 変更するビューを見つけるのに苦労しています。それらは何百もありますが、(IMHO)名前も整理も不十分です。PDF を生成する基本的なプロセスは次のようになります。ステップ3と4の間で混乱しています。
- Entity の ID が ActionResult に渡される
- エンティティはバッキング ストアから取得されます
- モデルは前述の Controller.View メソッドに渡さ
れます。
var xmlText = RenderActionResultToString( viewModel ); - 結果の ViewResult は ControllerContext のインスタンスと共に使用され、ブラウザーによって要求されたかのように HTML を生成します。
- 結果の HTML はサードパーティ ツールに渡され、PDF に変換されます。
私は他のすべてを非常に明確に理解しています。私が理解していないのは、View(model)への呼び出しが、ViewResult を返すときに使用するビュー ファイルを決定する方法です。どんな助けでも大歓迎です!
誰かが答えを判断するのに役立つ場合に備えて、以下のコードを含めています。
アクション結果:
public ActionResult ProposalPDF(String id, String location, bool hidePrices = false)
{
var proposal = _adc.Proposal.GetByKey(int.Parse(id));
var opportunity = _adc.Opportunity.GetByKey(proposal.FkOpportunityId.Value);
ViewData["AccountId"] = opportunity.FkAccountId;
ViewData["AccountType"] = opportunity.FkAccount.FkAccountTypeId;
ViewData["Location"] = location;
ViewData["HidePrices"] = hidePrices;
return ViewPdf(proposal);
}
ViewPDF メソッド:
protected ActionResult ViewPdf(object model)
{
// Create the iTextSharp document.
var document = new Document(PageSize.LETTER);
// Set the document to write to memory.
var memoryStream = new MemoryStream();
var pdfWriter = PdfWriter.GetInstance(document, memoryStream);
pdfWriter.CloseStream = false;
document.Open();
// Render the view xml to a string, then parse that string into an XML dom.
var viewModel = View(model);
var xmlText = RenderActionResultToString(viewModel);
var htmlPipelineContext = new HtmlPipelineContext();
htmlPipelineContext.SetTagFactory(Tags.GetHtmlTagProcessorFactory());
//CSS stuff
var cssResolver = XMLWorkerHelper.GetInstance().GetDefaultCssResolver(false);
var cssResolverPipeline = new CssResolverPipeline(cssResolver, new HtmlPipeline(htmlPipelineContext, new PdfWriterPipeline(document, pdfWriter)));
var xmlWorker = new XMLWorker(cssResolverPipeline, true);
var xmlParser = new XMLParser(xmlWorker);
xmlParser.Parse(new StringReader(xmlText));
// Close and get the resulted binary data.
document.Close();
var buffer = new byte[memoryStream.Position];
memoryStream.Position = 0;
memoryStream.Read(buffer, 0, buffer.Length);
// Send the binary data to the browser.
return new BinaryContentResult(buffer, "application/pdf");
}
RenderActionResultToString ヘルパー メソッド:
protected string RenderActionResultToString(ActionResult result)
{
// Create memory writer.
var sb = new StringBuilder();
var memWriter = new StringWriter(sb);
// Create fake http context to render the view.
var fakeResponse = new HttpResponse(memWriter);
var fakeContext = new HttpContext(System.Web.HttpContext.Current.Request, fakeResponse);
var fakeControllerContext = new ControllerContext(new HttpContextWrapper(fakeContext), this.ControllerContext.RouteData, this.ControllerContext.Controller);
var oldContext = System.Web.HttpContext.Current;
System.Web.HttpContext.Current = fakeContext;
// Render the view.
result.ExecuteResult(fakeControllerContext);
// Restore data.
System.Web.HttpContext.Current = oldContext;
// Flush memory and return output.
memWriter.Flush();
return sb.ToString();
}