180

I have a strongly-typed Partial View that takes a ProductImage and when it is rendered I would also like to provide it with some additional ViewData which I create dynamically in the containing page. How can I pass both my strongly typed object and my custom ViewData to the partial view with the RenderPartial call?

var index = 0;
foreach (var image in Model.Images.OrderBy(p => p.Order))
{
  Html.RenderPartial("ProductImageForm", image); // < Pass 'index' to partial
  index++;
}
John Saunders
  • 157,405
  • 24
  • 229
  • 388
Nathan Taylor
  • 23,720
  • 17
  • 90
  • 152

9 Answers9

259

RenderPartial takes another parameter that is simply a ViewDataDictionary. You're almost there, just call it like this:

Html.RenderPartial(
      "ProductImageForm", 
       image, 
       new ViewDataDictionary { { "index", index } }
); 

Note that this will override the default ViewData that all your other Views have by default. If you are adding anything to ViewData, it will not be in this new dictionary that you're passing to your partial view.

womp
  • 111,854
  • 24
  • 229
  • 262
166

To extend on what womp posted, you can pass new View Data while retaining the existing View Data if you use the constructor overload of the ViewDataDictionary like so:

Html.RenderPartial(
      "ProductImageForm", 
       image, 
       new ViewDataDictionary(this.ViewData) { { "index", index } }
); 
ctorx
  • 6,373
  • 7
  • 36
  • 52
  • 3
    this can be very useful if you need to retain `TemplateInfo.HtmlFieldPrefix` in the child control which will be reset otherwise – Simon_Weaver Jul 06 '13 at 10:44
  • 12
    This is definitely the better solution. Losing the ViewData means that you lose the ModelState and all validations. +1 – Kevin Farrugia Feb 07 '15 at 11:08
  • This seems to work only if you do not pass a TemplateInfo.HtmlFieldPrefix in your ViewDataDictionary. I can only pass a variable as ilustrated on the answer, or pass a TemplateInfo.HtmlFieldPrefix but not both. – Ferox Nov 24 '16 at 13:42
47
@Html.Partial("_Header", new ViewDataDictionary { { "HeaderName", "User Management" }, { "TitleName", "List Of Users" } })
or
@{Html.RenderPartial("_Header", new ViewDataDictionary { { "HeaderName", "User Management" }, { "TitleName", "List Of Users" } });}

Partial Page(_Header):

<div class="row titleBlock">
    <h1>@ViewData["HeaderName"].ToString()</h1>
    <h5>@ViewData["TitleName"].ToString()</h5>
</div>
muthuvel
  • 1,014
  • 10
  • 16
10

I think this should work no?

ViewData["currentIndex"] = index;
Joel Martinez
  • 44,051
  • 25
  • 123
  • 182
6

Create another class which contains your strongly typed class.

Add your new stuff to the class and return it in the view.

Then in the view, ensure you inherit your new class and change the bits of code that will now be in error. namely the references to your fields.

Hope this helps. If not then let me know and I'll post specific code.

griegs
  • 22,002
  • 28
  • 113
  • 201
4

The easiest way to pass additional data is to add the data to the existing ViewData for the view as @Joel Martinez notes. However, if you don't want to pollute your ViewData, RenderPartial has a method that takes three arguments as well as the two-argument version you show. The third argument is a ViewDataDictionary. You can construct a separate ViewDataDictionary just for your partial containing just the extra data that you want to pass in.

tvanfosson
  • 490,224
  • 93
  • 683
  • 780
4

I know this is an old post but I came across it when faced with a similar issue using core 3.0, hope it helps someone.

@{
Layout = null;
ViewData["SampleString"] = "some string need in the partial";
}

<partial name="_Partial" for="PartialViewModel" view-data="ViewData" />
Drytin
  • 326
  • 1
  • 8
2

This should also work.

this.ViewData.Add("index", index);

Html.RenderPartial(
      "ProductImageForm", 
       image, 
       this.ViewData
); 
Tony Abrams
  • 4,224
  • 3
  • 20
  • 31
0

You can use the dynamic variable ViewBag

ViewBag.AnotherValue = valueToView;
Jack1987
  • 679
  • 1
  • 14
  • 26