2014-02-17 149 views
5

有没有人有任何示例代码获取Umbraco MVC与Castle Windsor依赖注入框架一起工作?我遇到的问题是让我的表面控制器使用可注射的参数化构造函数。我知道我做错了什么,但不知道是什么。Umbraco MVC与城堡温莎

我也跟着(非一把umbraco)这里的教程 - http://docs.castleproject.org/Windsor.Windsor-tutorial-part-four-putting-it-all-together.ashx - 这对App_Start基本上意味着我运行这段代码:

var container = new WindsorContainer().Install(FromAssembly.This()); 
var controllerFactory = new MyCustomControllerFactory(container.Kernel); 
ControllerBuilder.Current.SetControllerFactory(controllerFactory); 

代码MyCustomControllerFactory如下。

另外,我实现IWindsorInstaller包含以下内容:

container.Register(Classes.FromThisAssembly() 
    .BasedOn<SurfaceController>() 
    .LifestyleTransient()); 

我得到的例外是“支持Umbraco.Web.Mvc.RenderMvcController发现服务的成分,不”,由GetControllerInstance抛出

public class TestSurfaceController : SurfaceController 
{ 
    public TestSurfaceController(INameService nameService) 
    { 
     .... 
    } 
} 

如果任何人有一些示例代码工作我会很感激:方法时,我称之为地面控制器与parametised构造如下。我之前用Umbraco连接了Ninject,但没有遇到任何问题,但是在这个项目中,我被绑定到温莎城堡并且无处不在!提前致谢。

MyCustomControllerFactory.cs:

public class MyCustomControllerFactory : DefaultControllerFactory 
{ 
    private readonly IKernel kernel; 

    public FastStartControllerFactory(IKernel kernel) 
    { 
     this.kernel = kernel; 
    } 

    public override void ReleaseController(IController controller) 
    { 
     kernel.ReleaseComponent(controller); 
    } 

    protected override IController GetControllerInstance(RequestContext requestContext, Type controllerType) 
    { 
     if (controllerType == null) 
     { 
      throw new HttpException(404, string.Format("The controller for path '{0}' could not be found.", requestContext.HttpContext.Request.Path)); 
     } 
     return (IController)kernel.Resolve(controllerType); 
    } 
} 

回答

7

我相信你的问题是在这里:

ControllerBuilder.Current.SetControllerFactory(controllerFactory); 

这是更换控制器工厂所有控制器,包括RenderMVCController,和城堡找不到匹配该类型的组件。

诀窍是使用FilteredControllerFactoryResolver,它允许Umbraco根据您提供的一些标准(在这种情况下,您的容器是否可以解析控制器类型)决定使用哪个控制器。构图并不像直MVC应用程序(恕我直言)那样干净,但它的工作原理。

下面是一个(一把umbraco 7.x中)经滤波的控制器,实现了IFilteredControllerFactory界面的示例:(使用ApplicationEventHandler

public class FilteredControllerFactory : ControllerFactory, IFilteredControllerFactory 
{ 
    public bool CanHandle(RequestContext request) 
    { 
     Type controllerType = GetControllerType(request, request.RouteData.Values["controller"].ToString()); 
     return ApplicationStartup.Container.Kernel.HasComponent(controllerType); 
    } 
} 

以及相应的代码来设置组合物:

public class ApplicationStartup : ApplicationEventHandler 
{ 
    internal static IWindsorContainer Container; 

    protected override void ApplicationStarting(UmbracoApplicationBase umbracoApplication, ApplicationContext applicationContext) 
    { 
     base.ApplicationStarting(umbracoApplication, applicationContext); 

     Container = new WindsorContainer() 
      .Install(Configuration.FromAppConfig()) 
      .Register(Classes.FromThisAssembly().BasedOn<IController>().LifestyleTransient()); 

     FilteredControllerFactoriesResolver.Current.InsertType<FilteredControllerFactory>(0); 
    } 
} 

这种方法应该适用于路由劫持和地面控制器。

最后,请注意,如果您还想支持注入API控制器,则需要单独连线。例如:

GlobalConfiguration.Configuration.Services.Replace(typeof(IHttpControllerActivator), new CompositionRoot(Container.Kernel)) 

其中CompositionRoot是您自己的Windsor组合根类。

The Gist here也可能证明有用。

+0

上谢谢你这么多的宝贵时间这个。我几乎没有时间来测试它,但这一切都非常有意义。一旦我尝试过,我一定会报告回来。 – getsetcode

+0

你的要点中的对象工厂事情复杂一点,但它的工作。干杯! –

0

我读过Kristopher的回答,我发现它很有趣,因为我不知道IFilteredControllerFactory及其用法。感谢分享。

无论如何,通常在我的项目我有很多包含每个自己的控制器的dll,所以我更喜欢一个更一般的方式来注册的所有控制器:

container.Register(
      Classes 
      .FromAssemblyInDirectory(new AssemblyFilter(AssemblyDirectory)) 
      .BasedOn<IController>() 
      .LifestyleTransient()); 

其中

/// <summary> 
    /// Local Directory where are present all the assemblies 
    /// </summary> 
    static public string AssemblyDirectory 
    { 
     //Snippet code from: https://gist.github.com/iamkoch/2344638 
     get 
     { 
      var codeBase = Assembly.GetExecutingAssembly().CodeBase; 
      var uri = new UriBuilder(codeBase); 
      var path = Uri.UnescapeDataString(uri.Path); 
      return Path.GetDirectoryName(path); 
     } 
    } 

通过这种方式,Umbraco的RenderMVCController也将被映射并正确解析。

最近我写了几个关于DI的文章在一把umbraco应用:

希望它可以帮助