5

これは、MVC3Razorhttppostが複雑なオブジェクトの子コレクションを返すことからのフォローアップの質問です。

私が挙げた例は非常に単純でした。子コレクションは、実際にはすべて抽象基本クラスからのオブジェクトのコレクションです。したがって、コレクションには基本クラスのリストがあります。

派生クラスごとにテンプレートを作成し、子がタイプの場合はテンプレート名を文字列として使用してみました。テンプレートはビューにレンダリングされますが、ポストバックには入力されません。

テンプレートでeditorforビットを使用して正しいテンプレートを選択し、親コンテナー内の子オブジェクトにマーシャリングする情報を取得する方法がわかりません。

4

1 に答える 1

8

カスタムモデルバインダーを使用できます。例を見てみましょう。

モデル:

public class MyViewModel
{
    public IList<BaseClass> Children { get; set; }
}

public abstract class BaseClass
{
    public int Id { get; set; }

    [HiddenInput(DisplayValue = false)]
    public string ModelType
    {
        get { return GetType().FullName; }
    }
}

public class Derived1 : BaseClass
{
    public string Derived1Property { get; set; }
}

public class Derived2 : BaseClass
{
    public string Derived2Property { get; set; }
}

コントローラ:

public class HomeController : Controller
{
    public ActionResult Index()
    {
        var model = new MyViewModel
        {
            Children = new BaseClass[]
            {
                new Derived1 { Id = 1, Derived1Property = "prop1" },
                new Derived2 { Id = 2, Derived2Property = "prop2" },
            }
        };
        return View(model);
    }

    [HttpPost]
    public ActionResult Index(MyViewModel model)
    {
        // everything will be fine and dandy here
        ...
    }
}

ビュー(~/Views/Home/Index.cshtml):

@model MyViewModel

@using (Html.BeginForm())
{
    for (int i = 0; i < Model.Children.Count; i++)
    {
        @Html.EditorFor(x => x.Children[i].ModelType)
        <div>
            @Html.EditorFor(x => x.Children[i].Id)
            @Html.EditorFor(x => x.Children[i])    
        </div>
    }

    <button type="submit">OK</button>
}

Dervied1タイプ( )のエディターテンプレート~/Views/Home/EditorTemplates/Derived1.cshtml

@model Derived1
@Html.EditorFor(x => x.Derived1Property)

Dervied2およびタイプ( )のエディターテンプレート~/Views/Home/EditorTemplates/Derived2.cshtml

@model Derived2
@Html.EditorFor(x => x.Derived2Property)

残っているのは、隠しフィールドの値を使用してコレクション内の適切なタイプをインスタンス化するカスタムモデルバインダーだけです。

public class BaseClassModelBinder : DefaultModelBinder
{
    protected override object CreateModel(ControllerContext controllerContext, ModelBindingContext bindingContext, Type modelType)
    {
        var typeValue = bindingContext.ValueProvider.GetValue(bindingContext.ModelName + ".ModelType");
        var type = Type.GetType(
            (string)typeValue.ConvertTo(typeof(string)),
            true
        );
        var model = Activator.CreateInstance(type);
        bindingContext.ModelMetadata = ModelMetadataProviders.Current.GetMetadataForType(() => model, type);
        return model;
    }
}

に登録されApplication_Startます:

ModelBinders.Binders.Add(typeof(BaseClass), new BaseClassModelBinder());
于 2012-05-11T17:15:03.933 に答える