UPDATE
Основываясь на сообщении в блоге Jeroen (см. Его ответ ниже, со ссылкой), а также на флеш-памяти, которая была у меня после повторного просмотра кода, я обновил ExtendedJsonValueProviderFactory , чтобы он всегда правильно создать BackingStore для словаря верхнего уровня, переданного через JSON.
Код доступен на GitHub на https://github.com/counsellorben/ASP.NET-MVC-JsonDictionaryBinding,, а рабочий пример на http://oss.form.vu/json-dictionary-example/.
Удалив текущий JsonValueProviderFactory
и заменив тот, который может обрабатывать создание словаря, вы можете связать свой словарь. Во-первых, как указал Кейт, в вашем Javascript обязательно закройте свой словарь внутри «filterItem», так как это имя переменной модели в действии вашего контроллера, а для JSON - имя переменной в действии контроллера должен совпадать с именем возвращаемого элемента Json. Кроме того, при передаче класса все вложенные элементы должны соответствовать именам свойств в классе.
Затем создайте класс ExtendedJsonValueProviderFactory
следующим образом:
using System;
using System.Collections;
using System.Collections.Generic;
using System.Collections.Specialized;
using System.Globalization;
using System.IO;
using System.Web.Script.Serialization;
public sealed class ExtendedJsonValueProviderFactory : ValueProviderFactory
{
private void AddToBackingStore(Dictionary<string, object> backingStore, string prefix, object value)
{
IDictionary<string, object> d = value as IDictionary<string, object>;
if (d != null)
{
foreach (KeyValuePair<string, object> entry in d)
{
if (entry.Key.EndsWith("Dictionary", StringComparison.CurrentCulture))
CreateDictionary(backingStore, entry);
else
AddToBackingStore(backingStore, MakePropertyKey(prefix, entry.Key), entry.Value);
}
return;
}
IList l = value as IList;
if (l != null)
{
for (int i = 0; i < l.Count; i++)
{
AddToBackingStore(backingStore, MakeArrayKey(prefix, i), l[i]);
}
return;
}
// primitive
backingStore[prefix] = value;
}
private void CreateDictionary(Dictionary<string, object> backingStore, KeyValuePair<string, object> source)
{
var d = source.Value as IDictionary<string, object>;
var dictionary = new Dictionary<string, string>();
foreach (KeyValuePair<string, object> entry in d)
dictionary.Add(entry.Key, entry.Value.ToString());
AddToBackingStore(backingStore, source.Key, dictionary);
return;
}
private static object GetDeserializedObject(ControllerContext controllerContext)
{
if (!controllerContext.HttpContext.Request.ContentType.StartsWith("application/json", StringComparison.OrdinalIgnoreCase))
{
// not JSON request
return null;
}
StreamReader reader = new StreamReader(controllerContext.HttpContext.Request.InputStream);
string bodyText = reader.ReadToEnd();
if (String.IsNullOrEmpty(bodyText))
{
// no JSON data
return null;
}
JavaScriptSerializer serializer = new JavaScriptSerializer();
object jsonData = serializer.DeserializeObject(bodyText);
return jsonData;
}
public override IValueProvider GetValueProvider(ControllerContext controllerContext)
{
if (controllerContext == null)
{
throw new ArgumentNullException("controllerContext");
}
object jsonData = GetDeserializedObject(controllerContext);
if (jsonData == null)
{
return null;
}
Dictionary<string, object> backingStore = new Dictionary<string, object>(StringComparer.OrdinalIgnoreCase);
AddToBackingStore(backingStore, String.Empty, jsonData);
return new DictionaryValueProvider<object>(backingStore, CultureInfo.CurrentCulture);
}
private static string MakeArrayKey(string prefix, int index)
{
return prefix + "[" + index.ToString(CultureInfo.InvariantCulture) + "]";
}
private static string MakePropertyKey(string prefix, string propertyName)
{
return (String.IsNullOrEmpty(prefix)) ? propertyName : prefix + "." + propertyName;
}
}
Вы можете заметить, что этот класс практически идентичен стандартному классу JsonValueProviderFactory, за исключением расширения для создания записи в DictionaryValueProvider типа Dictionary<string,string>
. Вы также должны заметить, что для того, чтобы обрабатываться как словарь, элемент должен иметь имя, оканчивающееся на «Словарь» (и хотя я думаю, что это значительный запах кода, я не могу придумать другую альтернативу в это время ... я открыт для предложений).
Затем добавьте следующее к Application_Start
в Global.asax.cs
:
var j = ValueProviderFactories.Factories.FirstOrDefault(f => f.GetType().Equals(typeof(JsonValueProviderFactory)));
if (j != null)
ValueProviderFactories.Factories.Remove(j);
ValueProviderFactories.Factories.Add(new ExtendedJsonValueProviderFactory());
Это удалит стандартную JsonValueProviderFactory и заменит ее нашим расширенным классом.
Последний шаг: наслаждайся совершенством.