I wrote custom VirtualFile and VirtualPathProvider implementations that are successfully obtaining embedded resources that are Partial Views.
However, when I attempt to render them it produces this error:
The view at '~/Succeed.Web/Succeed.Web.Controls.SImporter._SImporter.cshtml' must derive from WebViewPage, or WebViewPage<TModel>.
When rendering the partial view, inside of a regular View, it looks like the following:
Html.RenderPartial("~/Succeed.Web/Succeed.Web.Controls.SImporter._SImporter.cshtml");
What is causing it to believe this isn't a partial view?
EDIT: I Posted my code for both the virtual file & virtual file provider implementations for anyone who stumbles upon this looking for solution on getting that component working. This question will also serve well for those based upon the question title.
ere is the VirtualFile implementation for reference:
public class SVirtualFile : VirtualFile
{
private string m_path;
public SVirtualFile(string virtualPath)
: base(virtualPath)
{
m_path = VirtualPathUtility.ToAppRelative(virtualPath);
}
public override System.IO.Stream Open()
{
var parts = m_path.Split('/');
var assemblyName = parts[1];
var resourceName = parts[2];
assemblyName = Path.Combine(HttpRuntime.BinDirectory, assemblyName);
var assembly = System.Reflection.Assembly.LoadFile(assemblyName + ".dll");
if (assembly != null)
{
return assembly.GetManifestResourceStream(resourceName);
}
return null;
}
}
VirtualPathProvider:
public class SVirtualPathProvider : VirtualPathProvider
{
public SVirtualPathProvider()
{
}
private bool IsEmbeddedResourcePath(string virtualPath)
{
var checkPath = VirtualPathUtility.ToAppRelative(virtualPath);
return checkPath.StartsWith("~/Succeed.Web/", StringComparison.InvariantCultureIgnoreCase);
}
public override bool FileExists(string virtualPath)
{
return IsEmbeddedResourcePath(virtualPath) || base.FileExists(virtualPath);
}
public override VirtualFile GetFile(string virtualPath)
{
if (IsEmbeddedResourcePath(virtualPath))
{
return new SVirtualFile(virtualPath);
}
else
{
return base.GetFile(virtualPath);
}
}
public override CacheDependency GetCacheDependency( string virtualPath, IEnumerable virtualPathDependencies, DateTime utcStart)
{
if (IsEmbeddedResourcePath(virtualPath))
{
return null;
}
else
{
return base.GetCacheDependency(virtualPath, virtualPathDependencies, utcStart);
}
}
}
And of course, don't forget to register this new provider in the Global.asax file of your project in the Application_Start() event
System.Web.Hosting.HostingEnvironment.RegisterVirtualPathProvider(new SVirtualPathProvider());
I used the OPs answer as a base but expanded on it a bit and incorporated the answer to the question in my solution.
This seems like a somewhat commonly asked question here on SO and I haven't seen a complete answer so I thought it might be helpful to share my working solution.
I load my resources from a database and I have them cached in the default Cache (System.Web.Caching.Cache).
What I ended up doing was creating a custom CacheDependency on the KEY that I am using to lookup the resource. That way, whenever my other code invalidates that cache (on an edit, etc.) the cache dependency on that key is removed and the VirtualPathProvider in turn invalidates its cache and the VirtualFile gets reloaded.
I also changed the code so that it automatically prepends inherits statement so that it doesn't need to be stored in my database resource and I also automatically prepend a few default using statements as this "view" is not loaded via the normal channels, so anything default includes in your web.config or viewstart are not usable.
CustomVirtualFile:
CustomVirtualPathProvider:
Hope this helps!
I leaned heavily on the information in the OP as well as Darin Dimitrov's answer to create a simple prototype for sharing MVC components across projects. While those were very helpful, I still ran into a few additional barriers that are addressed in the prototype like using shared views with @model's.
Because now you are serving your views from some unknown location there is no longer the
~/Views/web.config
file which applies and indicates the base class for your razor views (<pages pageBaseType="System.Web.Mvc.WebViewPage">
). So you could add an @inherits directive at the top of each embedded view to indicate the base class.