Вопрос

Я создаю несколько пользовательских связующих для сложных типов в моей модели. Моя модель состоит из объектов, которые имеют свои отдельные связующие. Я хочу, чтобы базовый объект выполнять свою грязную работу, а затем заполнить комплексный объект, который он инкапсулирует, передавая стандартную маршруту ModelBinder. Как мне это сделать?

Для иллюстрации я создал очень простой пример.

Скажем, моя модель содержит эти объекты.

public class Person
{
    public string Name {get; set;}
    public PhoneNumber PhoneNumber {get; set;}
}

public class PhoneNumber
{
    public string AreaCode {get; set;}
    public string LocalNumber {get; set;}
}

И у меня есть следующие связки для каждой из этих моделей. Не то, чтобы прелисовестность должна заполнить фонеборум, но не хочет дублировать код в связующем со стокам. Как это делегирует вернуться к маршрутизации Binder Stardard?

public class PersonBinder: IModelBinder
{
    public object BindModel(ControllerContext controllerContext, ModelBindingContext     bindingContext)
    {
        Person person = new Person();
        person.Name = bindingContext.ValueProvider.GetValue(String.Format("{0}.{1}", bindingContext.ModelName, "Name")).AttemptedValue

        // This is where I'd like to have the PhoneNumber object use binding from another customer ModelBinder.
        // Of course the bindingContext.ModelName should be updated to its current value + "PhoneNumber"
        person.PhoneNumber = ???;  // I don't want to explicitly call the PhoneNumberBinder it should go through standard Binding routing.  (ie.  ModelBinders.Binders[typeof(PhoneNumber)] = new PhoneNumberBinder();)

        return person;       
    }
}

public class PhoneNumberBinder: IModelBinder
{
    public object BindModel(ControllerContext controllerContext, ModelBindingContext     bindingContext)
    {
        PhoneNumber phoneNumber = new PhoneNumber();
        phoneNumber.AreaCode = bindingContext.ValueProvider.GetValue(String.Format("{0}.{1}", bindingContext.ModelName, "AreaCode")).AttemptedValue
        phoneNumber.LocalNumber = bindingContext.ValueProvider.GetValue(String.Format("{0}.{1}", bindingContext.ModelName, "LocalNumber")).AttemptedValue

        return phoneNumber;
    }
}

И, конечно, я зарегистрировал мои модели в файле Global.Asax.cs.

protected void Application_Start()
{
    AreaRegistration.RegisterAllAreas();

    RegisterRoutes(RouteTable.Routes);

    ModelBinders.Binders[typeof(Person)] = new PersonBinder();
    ModelBinders.Binders[typeof(PhoneNumber)] = new PhoneNumberBinder();
}

Спасибо,

Джастин

Это было полезно?

Решение

Ну, мне удалось придумать решение. Пожалуйста, не стесняйтесь комментировать его действительность.

public object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
{
     Person person = new Person();
     person.Name = bindingContext.ValueProvider.GetValue("Name").AttemptedValue

     if (bindingContext.ModelName == String.Empty)
     {
         bindingContext.ModelName = "PhoneNumber";
     }
     else
     {
         bindingContext.ModelName = bindingContext.ModelName + ".PhoneNumber";
     }

     PhoneNumber phoneNumber = new PhoneNumber();
     bindingContext.ModelMetadata = ModelMetadataProviders.Current.GetMetadataForType(() => phoneNumber, phoneNumber.GetType());

     IModelBinder binder = ModelBinders.Binders[typeof(PhoneNumber)];
     if (binder == null)
     {
          binder = ModelBinders.Binders.DefaultBinder;
     }

     person.PhoneNumber = binder.BindModel(controllerContext, bindingContext) as PhoneNumber;

     return person;                         
}

Вот краткое изложение того, что я сделал.

  1. Найдите в том, что ModelBinder с использованием глобально доступных ModelBinders.binders Collection (Foleback к по умолчанию, если никто не зарегистрирован)
  2. Создайте ModelMetadatataProvider для модели, с которой я связываюсь.
  3. Установите свойство ModelName BindingContext на свойство модели, которое я пытаюсь заполнить («Phonenumber»).

Другие советы

Вместо того, чтобы написать связующее, вы могли бы пользователя автоматически и обрабатывать сложную конструкцию модели в действии.

Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top