UPDATE: ho fatto un passo attraverso il codice MVC sorgente (in particolare DefaultModelBinder
classe) e qui è quello che ho trovato:
La classe determina che stiamo cercando di legare una collezione così chiama il metodo: UpdateCollection(...)
che crea un interno ModelBindingContext
che ha una proprietà null
Model
. Successivamente, tale contesto viene inviato al metodo BindComplexModel(...)
che controlla la proprietà Model
per null
e crea un'istanza nuova del tipo di modello, se questo è il caso.
Ecco cosa provoca il ripristino dei valori.
E così, vengono popolati solo i valori che stanno arrivando attraverso i dati del form/query string/route, il resto rimane nel suo stato inizializzato.
Sono stato in grado di apportare pochissime modifiche a UpdateCollection(...)
per risolvere questo problema.
Ecco il metodo con le mie modifiche:
internal object UpdateCollection(ControllerContext controllerContext, ModelBindingContext bindingContext, Type elementType) {
IModelBinder elementBinder = Binders.GetBinder(elementType);
// build up a list of items from the request
List<object> modelList = new List<object>();
for (int currentIndex = 0; ; currentIndex++) {
string subIndexKey = CreateSubIndexName(bindingContext.ModelName, currentIndex);
if (!DictionaryHelpers.DoesAnyKeyHavePrefix(bindingContext.ValueProvider, subIndexKey)) {
// we ran out of elements to pull
break;
}
// **********************************************************
// The DefaultModelBinder shouldn't always create a new
// instance of elementType in the collection we are updating here.
// If an instance already exists, then we should update it, not create a new one.
// **********************************************************
IList containerModel = bindingContext.Model as IList;
object elementModel = null;
if (containerModel != null && currentIndex < containerModel.Count)
{
elementModel = containerModel[currentIndex];
}
//*****************************************************
ModelBindingContext innerContext = new ModelBindingContext() {
Model = elementModel, // assign the Model property
ModelName = subIndexKey,
ModelState = bindingContext.ModelState,
ModelType = elementType,
PropertyFilter = bindingContext.PropertyFilter,
ValueProvider = bindingContext.ValueProvider
};
object thisElement = elementBinder.BindModel(controllerContext, innerContext);
// we need to merge model errors up
VerifyValueUsability(controllerContext, bindingContext.ModelState, subIndexKey, elementType, thisElement);
modelList.Add(thisElement);
}
// if there weren't any elements at all in the request, just return
if (modelList.Count == 0) {
return null;
}
// replace the original collection
object collection = bindingContext.Model;
CollectionHelpers.ReplaceCollection(elementType, collection, modelList);
return collection;
}
fonte
2009-07-30 19:26:36
è vero anche per MVC 3? – Vidar
@Vidar Sì, ha paura. – nfplee