Search code examples
asp.netloadcontrol

Create Instance Aspx Page of Ascx Control In a Back End Class without Loading FilePath


Question: Is it possible in back end code (not in the code behind but in an actual back end class) to load and render a page or control defined in a .aspx or .ascx without having to use Load(path) and instead just create an instance of the page/control class?

I want to be able to do this (from a back end class NOT a code behind):

MyControl myCtl = new MyApp.Views.Shared.MyControl();
String html = Util.ControlToString(myCtl); //I get an empty string & hidden errors

instead of this

public static string ControlToString(string path)
{
    Page pageHolder = new Page();
    MyControl myCtl = (MyControl)pageHolder.LoadControl(path);
    pageHolder.Controls.Add(myCtl);
    StringWriter output = new StringWriter();
    HttpContext.Current.Server.Execute(pageHolder, output, false);
    return output.ToString();
}

Details: In a Asp.net WebApp I occasionally need to render a user control (.ascx) or page (.aspx) as a HTML string. When a page or control inherits from a code behind, its class shows up in intellisense in my back end code and I can create an instance and set properties without getting compile time or run time errors. However, when I try to render the page or control I always get an empty string and upon inspection the page or control shows suppressed internal rendering errors unless I load the page or control using its physical file path.

I think the key issue has to do with when & how the .aspx / .ascx files are runtime compiled. I don't want to create a pre compiled class library of user controls because that would make the design process awkward and I really like the designer features offered by the .aspx / .ascx pages and so I'd love to find a way to make the pages compile in the solution so that they are usable like any other back end class but can still be created using the designer. I want the best of both worlds (1) to be able to edit pages and controls in the designer and (2) create instances and set their properties using back end classes.


Solution

  • Here is an approach that may help in situations like this.

    The "back-end" code may not know where the user control is located, but the User Control does know where it is.

    So, in the User Control, add a static method like this:

    public partial class MyControl : UserControl
    {
      ...
      public static MyControl LoadControl(CustomDto initialData)
      {
        var myControl = 
            (MyControl) 
            ((Page) HttpContext.Current.Handler)
            .LoadControl("~\\UserControlsSecretPath\\MyControl.ascx");
        myControl._initialData = initialData;
        return myControl;
      }
      ...
      private CustomDto _initialData;
    }
    

    (The CustomDto is included to illustrate how initial data can be passed to the User Control. If you don't need to do that, take it out!)

    With this, the code that loads the user control does not need to know the path to where the user control is physically located. If that location ever changes, then update this one location. All other code that uses this UserControl is unchanged.

    In your back-end code, or anywhere else, you can do something this:

    var control = MyControl.LoadControl(customDto);
    PlaceHolder1.Controls.Add(control);