动态扩展应用程序的功能?

时间:2012-06-13 21:43:55

标签: c# asp.net asp.net-mvc asp.net-mvc-4

在使用ASP.NET MVC3 / 4时,我最近遇到了way来开发可插拔的应用程序模块,我喜欢这种方法的简单性。目前,我的应用程序结构如下:

enter image description here

因此,任何想要为我的应用程序开发扩展的人都遵循上述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>

为简单起见,假设区域名称是唯一的。有关如何做到这一点的任何建议吗?

5 个答案:

答案 0 :(得分:10)

如何将区域分配为可插拔模块?

不要在托管网络应用程序中创建区域,而是创建单独的项目,编译以分隔dll。将dll复制到您要使用它的任何Web应用程序。请记住将所有静态文件设置为“EmbeddedResource”。

如何更改以下代码,以便当有人将新区域放入bin文件夹时,应用程序会自动选择它并创建链接?插件作者应该怎么做才能启用它?

您可以使用MvcContrib PortableAreaRegistration的“总线”将便携式区域的消息和命令发送给“总线”上的任何人。这可以是托管网络应用程序,或理论上独立的区域可以相互发送消息。

在github上创建了两个粗略但功能性的演示代码:

MVC3版本:

https://github.com/AkosLukacs/PluggablePortableAreas

MVC4 RC版本:

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我觉得它会对你有很多帮助。