在使用ASP.NET MVC3 / 4时,我最近遇到了way来开发可插拔的应用程序模块,我喜欢这种方法的简单性。目前,我的应用程序结构如下:
因此,任何想要为我的应用程序开发扩展的人都遵循上述tutorial的方法,并创建一个保留在Areas
文件夹中的扩展名。我认为当添加新Areas
(作为新项目创建)时,会创建.pdb
个文件并将其放在bin
目录中。我的问题是:
Areas
作为可插拔模块进行分发?Area
放入bin
文件夹时,应用程序会自动选择并创建链接?插件作者应该怎么做才能启用它?在我的_Layout.cshtml
(全局共享布局)中,我执行以下操作来构建链接:
<ul>
<li>@Html.ActionLink("Area1", "Index", "Area1", new { Area = "Area1" }, null)</li>
<li>@Html.ActionLink("Area2", "Index", "Area2", new { Area = "Area2" }, null)</li>
<li>@Html.ActionLink("Area3", "Index", "Area3", new { Area = "Area3" }, null)</li>
</ul>
为简单起见,假设区域名称是唯一的。有关如何做到这一点的任何建议吗?
答案 0 :(得分:10)
不要在托管网络应用程序中创建区域,而是创建单独的项目,编译以分隔dll。将dll复制到您要使用它的任何Web应用程序。请记住将所有静态文件设置为“EmbeddedResource”。
您可以使用MvcContrib PortableAreaRegistration的“总线”将便携式区域的消息和命令发送给“总线”上的任何人。这可以是托管网络应用程序,或理论上独立的区域可以相互发送消息。
在github上创建了两个粗略但功能性的演示代码:
https://github.com/AkosLukacs/PluggablePortableAreas
https://github.com/AkosLukacs/PluggablePortableAreasMVC4
首先,您定义一条可以传递所需信息的消息。只是一个具有一些属性的POCO(PluggablePortableAreas.Common \ RegisterAreaMessage.cs):
public class RegisterAreaMessage : IEventMessage
{
public string LinkText { get; private set; }
public string AreaName { get; private set; }
public string ControllerName { get; private set; }
public string ActionName { get; private set; }
//...
}
为该消息类型创建一个处理程序(PluggablePortableAreas.Common \ RegisterAreaEventHandler.cs):
public class RegisterAreaEventHandler : MessageHandler<RegisterAreaMessage>{}
在这种情况下,MessageHandler只是将收到的消息添加到静态ConcurrentBag<RegisterAreaMessage>
。如果您愿意,可以使用DI,但希望保持简单。
您可以从便携式区域发送消息(Areas \ DemoArea1 \ Areas \ Demo1 \ Demo1AreaRegistration.cs):
//the portable area sends a message to the 'bus'
bus.Send(new RegisterAreaMessage("Link to the Demo area", AreaName, DefaultController, DefaultAction));
通过迭代消息集合(PluggablePortableAreas.Web \ Views \ Shared_Layout.cshtml)显示动态添加的链接:
@foreach(var sor in PluggablePortableAreas.Common.RegisterAreaEventHandler.RegisteredAreas) {
<li>@Html.ActionLink(sor.LinkText, sor.ActionName, sor.ControllerName, new{Area=sor.AreaName}, null)</li>
}
需要注意的另一件事:使用“完全限定”的区域名称。如果未明确指定区域名称,则MVC假定它是当前区域。没有区域没有问题,但第二个将指向"/path_to_your_app/CurrentArea/Home"
而不是"/path_to_your_app/Home"
。
<li>@Html.ActionLink("Home", "Index", "Home", new { Area = "" }, null)</li>
<li>@Html.ActionLink("I don't work from the portable area!", "Index", "Home")</li>
VS中的开发服务器感觉有点“不稳定”,有时便携式区域无法加载。在完整的IIS中可靠地运行......
答案 1 :(得分:2)
你正在做的事情与MVCContrib的便携式区域非常相似。
他们使用消息/应用程序总线模式在Web元素上动态添加新的小部件。 通过反射发现MessageHandler,并将每条消息传递给所有消息。因此,在您的情况下,插件作者应该只为标准消息实现处理程序(比如注册全局菜单链接)。
嵌入了可移植区域的资源,因此只能在bin文件夹中删除一个dll。为了自动提取并在您的应用中使用它,您必须通过FileSystemWatcher观看bin文件夹并重新启动您的应用程序(没有其他方法可以将新的.dll加载到asp中的AppDomain中。网络应用我想)。
您可以使用ASP.NET 4中的BuildManager功能从其他文件夹加载.dll。有关here的更多有用信息。
答案 2 :(得分:1)
这与Orchard CMS的模块类似。
看看他们的Gallery ......模块作为Nuget包发布,包含整个模块项目。
答案 3 :(得分:1)
您好,您可以在nopcommerce项目中寻找非常好的插件架构。
答案 4 :(得分:1)
看看这个tutorial我觉得它会对你有很多帮助。