Come faccio invoke UpdateModel dall'interno di un ModelBinder personalizzato? (MVC)
-
02-10-2019 - |
Domanda
Sto creando un paio di raccoglitori personalizzati per i tipi complessi nel mio modello. Il mio modello è composto da oggetti che hanno i propri raccoglitori separati. Voglio che l'oggetto di base di fare il suo lavoro sporco e quindi popolare l'oggetto complesso racchiude spacciando per l'instradamento ModelBinder standard. Come posso fare questo?
A scopo illustrativo ho creato un esempio molto semplice.
Di 'il mio modello contiene questi oggetti.
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;}
}
e ho il seguente Leganti per ciascuno di questi modelli. Non che il PersonBinder ha bisogno di popolare il PhoneNumber ma non vuole duplicare il codice nel legante PhoneNumber. Come delegare a tornare al stardard Binder di routing?
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;
}
}
E, naturalmente, ho registrato il mio ModelBinders nel file Global.asax.cs.
protected void Application_Start()
{
AreaRegistration.RegisterAllAreas();
RegisterRoutes(RouteTable.Routes);
ModelBinders.Binders[typeof(Person)] = new PersonBinder();
ModelBinders.Binders[typeof(PhoneNumber)] = new PhoneNumberBinder();
}
Grazie,
Justin
Soluzione
Bene sono riuscito a trovare una soluzione. Non esitate a commentare sulla sua validità.
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;
}
Ecco un riassunto di quello che ho fatto.
- Ricerca del ModelBinder utilizzando l'insieme ModelBinders.Binders globalmente accessibile (fallback al default se non è registrato)
- Creare il ModelMetadataProvider per il modello che sto legame.
- Impostare la proprietà ModelName del BindingContext alla proprietà modello che sto cercando di popolare ( "PhoneNumber").
Altri suggerimenti
Invece di scrivere un legante si potrebbe utente automapper e gestire la costruzione del modello complesso in azione.