MVC 3.0 编辑可变长度列表并使用 PRG 模式

发布于 2024-11-14 08:59:06 字数 6106 浏览 5 评论 0 原文

我创建了一个具有可变长度列表的视图,如下所述: http://blog.stevensanderson.com/2010/01/28/editing-a-variable-length-list-aspnet-mvc-2-style/

我尝试将 PRG 模式与操作过滤器一起使用,如第 13 点所述:http://weblogs.asp.net/rashid/archive/2009/04/01/asp-net-mvc-best-practices-part-1.aspx

我有一个编辑操作:

    [HttpGet, ImportModelStateFromTempData]
    public ActionResult Edit(int id)
    {
    }

和后操作:

    [HttpPost, ExportModelStateToTempData]
    public ActionResult Edit(int id, FormCollection formCollection)
    {
        if (!TryUpdateModel<CategoryEntity>(category, formCollection))
        {
            return RedirectToAction("Edit", new { id = id });
        }

        // succes, no problem processing this...
        return RedirectToAction("Edit", new { id = id });
    }

一切正常,包括验证和错误消息。

我遇到的唯一问题是重定向后不会保留新添加的项目和删除的项目(客户端删除/添加)。我正在尝试找到一种方法在使用新项目重定向后更新我的模型。我更改了 ImportModelStateFromTempData 属性以使用 OnActionExecuting 覆盖而不是 OnActionExecuted 覆盖,以使 ModelState 在操作中可用,但我没有看到从传入的 ModelState 更新模型的干净方法。

更改了 ImportModelStateFromTempData:

public class ImportModelStateFromTempData : ModelStateTempDataTransfer
{
    public override void OnActionExecuting(ActionExecutingContext filterContext)
    {
        ModelStateDictionary modelState = filterContext.Controller.TempData[Key] as ModelStateDictionary;

        if (modelState != null)
        {
            filterContext.Controller.ViewData.ModelState.Merge(modelState);
        }
        base.OnActionExecuting(filterContext);
    }

    public override void OnActionExecuted(ActionExecutedContext filterContext)
    {
        //ModelStateDictionary modelState = filterContext.Controller.TempData[Key] as ModelStateDictionary;

        //if (modelState != null)
        //{
        //    //Only Import if we are viewing
        //    if (filterContext.Result is ViewResult)
        //    {
        //        filterContext.Controller.ViewData.ModelState.Merge(modelState);
        //    }
        //    else
        //    {
        //        //Otherwise remove it.
        //        filterContext.Controller.TempData.Remove(Key);
        //    }
        //}
        base.OnActionExecuted(filterContext);
    }
}

非常感谢对此的任何输入,谢谢。

Harmen

更新:我想我可能会添加更多的(伪)代码以使其更清晰:

public class CategoryEntity
{
    public int Id;
    public string Name;
    public IEnumerable<CategoryLocEntity> Localized;
}

public class CategoryLocEntity
{
    public int CategoryId;
    public int LanguageId;
    public string LanguageName;
    public string Name;
}

我的编辑视图:

@model CategoryEntity

@{
    ViewBag.Title = Views.Category.Edit;
}

<h2>@Views.Category.Edit</h2>

<script src="@Url.Content("~/Scripts/jquery.validate.min.js")" type="text/javascript"></script>
<script src="@Url.Content("~/Scripts/jquery.validate.unobtrusive.min.js")" type="text/javascript"></script>
<script type="text/javascript"><!--

    $(document).ready(function () {
        $('#addItem').click(function () {
            var languageId = $('#languageId').val();
            var index = $('#editor-rows').children().size() - 1;
            $.ajax({
                url: this.href + '?languageId=' + languageId + '&index=' + index,
                cache: false,
                error: function (xhr, status, error) {
                    alert(error);
                },
                success: function (html) {
                    $('#editor-rows').append(html);
                }
            });
            return false;
        });

        $("a.removeItem").live("click", function () {
            $(this).parents("div.editor-row:first").remove();
            return false;
        });
    });

--></script>

@using (Html.BeginForm()) 
{
    @Html.ValidationSummary(false)
    <fieldset>
        <legend>@Views.Shared.Category</legend>
        @Html.HiddenFor(model => model.Id)
        <div id="editor-rows">
            <div class="editor-row">
                <div class="editor-label">
                    @Html.LabelFor(model => model.Name, Views.Shared.NameEnglish)
                </div>
                <div class="editor-field">
                    @Html.EditorFor(model => model.Name)
                    @Html.ValidationMessageFor(model => model.Name)
                </div>
            </div>

            @for (int i = 0; i < Model.Localized.Count; i++)
            {
                @Html.EditorFor(m => m.Localized[i], "_CategoryLoc", null, null)
            }
        </div>

        <div class="editor-label"></div>
        <div class="editor-field">
            @Html.DropDownList("languageId", (IEnumerable<SelectListItem>)ViewBag.LanguageSelectList)
            @Html.ActionLink(Views.Category.AddNewLanguage, "AddNewLanguage", null, new { id = "addItem" })
        </div>

        <p class="clear">
            <input type="submit" value="@Views.Shared.Save" />
        </p>
    </fieldset>
}

<div>
    @Html.ActionLink(Views.Shared.BackToList, "Index")
</div>

CategoryLocEntity 的编辑器模板:

@model CategoryLocEntity

<div class="editor-row">
    @Html.HiddenFor(model => model.Id)
    @Html.HiddenFor(model => model.LanguageId)
    <div class="editor-label">
        @Html.LabelFor(model => model.LanguageName, Model.LanguageName)
    </div>
    <div class="editor-field">
        @Html.EditorFor(model => model.Name)
        <a href="#" class="removeItem">@Views.Shared.Remove</a>
        @Html.ValidationMessageFor(model => model.Name)
    </div>
</div>

I created a view with a variable length list as described here: http://blog.stevensanderson.com/2010/01/28/editing-a-variable-length-list-aspnet-mvc-2-style/.

I am trying to use the PRG pattern with action filters as described at point 13 here: http://weblogs.asp.net/rashid/archive/2009/04/01/asp-net-mvc-best-practices-part-1.aspx.

I have an Edit action:

    [HttpGet, ImportModelStateFromTempData]
    public ActionResult Edit(int id)
    {
    }

And the post action:

    [HttpPost, ExportModelStateToTempData]
    public ActionResult Edit(int id, FormCollection formCollection)
    {
        if (!TryUpdateModel<CategoryEntity>(category, formCollection))
        {
            return RedirectToAction("Edit", new { id = id });
        }

        // succes, no problem processing this...
        return RedirectToAction("Edit", new { id = id });
    }

All works fine including validation and error messages.

The only problem I have is that newly added items and deleted items (client side deleted/added) are not preserved after the redirect. I am trying to find a way to update my model after the redirect with the new items. I changed the ImportModelStateFromTempData attribute to use the OnActionExecuting override instead of the OnActionExecuted override to have the ModelState available in the action but I don't see a clean way to update my model from the passed in ModelState.

Changed ImportModelStateFromTempData:

public class ImportModelStateFromTempData : ModelStateTempDataTransfer
{
    public override void OnActionExecuting(ActionExecutingContext filterContext)
    {
        ModelStateDictionary modelState = filterContext.Controller.TempData[Key] as ModelStateDictionary;

        if (modelState != null)
        {
            filterContext.Controller.ViewData.ModelState.Merge(modelState);
        }
        base.OnActionExecuting(filterContext);
    }

    public override void OnActionExecuted(ActionExecutedContext filterContext)
    {
        //ModelStateDictionary modelState = filterContext.Controller.TempData[Key] as ModelStateDictionary;

        //if (modelState != null)
        //{
        //    //Only Import if we are viewing
        //    if (filterContext.Result is ViewResult)
        //    {
        //        filterContext.Controller.ViewData.ModelState.Merge(modelState);
        //    }
        //    else
        //    {
        //        //Otherwise remove it.
        //        filterContext.Controller.TempData.Remove(Key);
        //    }
        //}
        base.OnActionExecuted(filterContext);
    }
}

Any input on this is much appreciated, thanks.

Harmen

UPDATE: Thought I might add some more of my (pseudo) code to make it more clear:

public class CategoryEntity
{
    public int Id;
    public string Name;
    public IEnumerable<CategoryLocEntity> Localized;
}

public class CategoryLocEntity
{
    public int CategoryId;
    public int LanguageId;
    public string LanguageName;
    public string Name;
}

My Edit view:

@model CategoryEntity

@{
    ViewBag.Title = Views.Category.Edit;
}

<h2>@Views.Category.Edit</h2>

<script src="@Url.Content("~/Scripts/jquery.validate.min.js")" type="text/javascript"></script>
<script src="@Url.Content("~/Scripts/jquery.validate.unobtrusive.min.js")" type="text/javascript"></script>
<script type="text/javascript"><!--

    $(document).ready(function () {
        $('#addItem').click(function () {
            var languageId = $('#languageId').val();
            var index = $('#editor-rows').children().size() - 1;
            $.ajax({
                url: this.href + '?languageId=' + languageId + '&index=' + index,
                cache: false,
                error: function (xhr, status, error) {
                    alert(error);
                },
                success: function (html) {
                    $('#editor-rows').append(html);
                }
            });
            return false;
        });

        $("a.removeItem").live("click", function () {
            $(this).parents("div.editor-row:first").remove();
            return false;
        });
    });

--></script>

@using (Html.BeginForm()) 
{
    @Html.ValidationSummary(false)
    <fieldset>
        <legend>@Views.Shared.Category</legend>
        @Html.HiddenFor(model => model.Id)
        <div id="editor-rows">
            <div class="editor-row">
                <div class="editor-label">
                    @Html.LabelFor(model => model.Name, Views.Shared.NameEnglish)
                </div>
                <div class="editor-field">
                    @Html.EditorFor(model => model.Name)
                    @Html.ValidationMessageFor(model => model.Name)
                </div>
            </div>

            @for (int i = 0; i < Model.Localized.Count; i++)
            {
                @Html.EditorFor(m => m.Localized[i], "_CategoryLoc", null, null)
            }
        </div>

        <div class="editor-label"></div>
        <div class="editor-field">
            @Html.DropDownList("languageId", (IEnumerable<SelectListItem>)ViewBag.LanguageSelectList)
            @Html.ActionLink(Views.Category.AddNewLanguage, "AddNewLanguage", null, new { id = "addItem" })
        </div>

        <p class="clear">
            <input type="submit" value="@Views.Shared.Save" />
        </p>
    </fieldset>
}

<div>
    @Html.ActionLink(Views.Shared.BackToList, "Index")
</div>

Editor template for the CategoryLocEntity:

@model CategoryLocEntity

<div class="editor-row">
    @Html.HiddenFor(model => model.Id)
    @Html.HiddenFor(model => model.LanguageId)
    <div class="editor-label">
        @Html.LabelFor(model => model.LanguageName, Model.LanguageName)
    </div>
    <div class="editor-field">
        @Html.EditorFor(model => model.Name)
        <a href="#" class="removeItem">@Views.Shared.Remove</a>
        @Html.ValidationMessageFor(model => model.Name)
    </div>
</div>

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(1

捎一片雪花 2024-11-21 08:59:06

我找到了一个解决方案(可能不是最优雅的解决方案,但它对我有用)。我创建了自己的 ModelStateValueProvider 来与 UpdateModel 一起使用。该代码基于DictionaryValueProvider。请参阅http://www.java2s.com/Open-Source/CSharp/2.6.4-mono-.net-core/System.Web/System/Web/Mvc/DictionaryValueProvider%601.cs.htm

public class ModelStateValueProvider : IValueProvider
{
    HashSet<string> prefixes = new HashSet<string>(StringComparer.OrdinalIgnoreCase);
    ModelStateDictionary modelStateDictionary;

    public ModelStateValueProvider(ModelStateDictionary modelStateDictionary)
    {
        if (modelStateDictionary == null)
            throw new ArgumentNullException("modelStateDictionary");

        this.modelStateDictionary = modelStateDictionary;

        FindPrefixes();
    }

    private void FindPrefixes()
    {
        if (modelStateDictionary.Count > 0)
            prefixes.Add(string.Empty);

        foreach (var modelState in modelStateDictionary)
            prefixes.UnionWith(GetPrefixes(modelState.Key));
    }

    public bool ContainsPrefix(string prefix)
    {
        if (prefix == null)
        {
            throw new ArgumentNullException("prefix");
        }

        return prefixes.Contains(prefix);
    }

    public ValueProviderResult GetValue(string key)
    {
        if (key == null)
            throw new ArgumentNullException("key");

        return modelStateDictionary.ContainsKey(key) ? modelStateDictionary[key].Value : null;
    }

    static IEnumerable<string> GetPrefixes(string key)
    {
        yield return key;
        for (int i = key.Length - 1; i >= 0; i--)
        {
            switch (key[i])
            {
                case '.':
                case '[':
                    yield return key.Substring(0, i);
                    break;
            }
        }
    }
}

public class ModelStateValueProviderFactory : ValueProviderFactory
{
    public override IValueProvider GetValueProvider(ControllerContext controllerContext)
    {
        return new ModelStateValueProvider(controllerContext.Controller.ViewData.ModelState);
    }
}

我在编辑(获取)操作中使用它,例如:

[HttpGet, ImportModelStateFromTempData]
public ActionResult Edit(int id)
{
  var category = new CategoryEntity(id);
  if (!ModelState.IsValid)
  {    
     TryUpdateModel<CategoryEntity>(category, 
         new ModelStateValueProviderFactory().GetValueProvider(ControllerContext));
  }
  return View(category);
}

期待您的评论...

I found a solution (probably not the most elegant one but it works for me). I created my own ModelStateValueProvider to be used with UpdateModel. The code is based on the DictionaryValueProvider. See http://www.java2s.com/Open-Source/CSharp/2.6.4-mono-.net-core/System.Web/System/Web/Mvc/DictionaryValueProvider%601.cs.htm.

public class ModelStateValueProvider : IValueProvider
{
    HashSet<string> prefixes = new HashSet<string>(StringComparer.OrdinalIgnoreCase);
    ModelStateDictionary modelStateDictionary;

    public ModelStateValueProvider(ModelStateDictionary modelStateDictionary)
    {
        if (modelStateDictionary == null)
            throw new ArgumentNullException("modelStateDictionary");

        this.modelStateDictionary = modelStateDictionary;

        FindPrefixes();
    }

    private void FindPrefixes()
    {
        if (modelStateDictionary.Count > 0)
            prefixes.Add(string.Empty);

        foreach (var modelState in modelStateDictionary)
            prefixes.UnionWith(GetPrefixes(modelState.Key));
    }

    public bool ContainsPrefix(string prefix)
    {
        if (prefix == null)
        {
            throw new ArgumentNullException("prefix");
        }

        return prefixes.Contains(prefix);
    }

    public ValueProviderResult GetValue(string key)
    {
        if (key == null)
            throw new ArgumentNullException("key");

        return modelStateDictionary.ContainsKey(key) ? modelStateDictionary[key].Value : null;
    }

    static IEnumerable<string> GetPrefixes(string key)
    {
        yield return key;
        for (int i = key.Length - 1; i >= 0; i--)
        {
            switch (key[i])
            {
                case '.':
                case '[':
                    yield return key.Substring(0, i);
                    break;
            }
        }
    }
}

public class ModelStateValueProviderFactory : ValueProviderFactory
{
    public override IValueProvider GetValueProvider(ControllerContext controllerContext)
    {
        return new ModelStateValueProvider(controllerContext.Controller.ViewData.ModelState);
    }
}

I use it in de Edit (Get) action like:

[HttpGet, ImportModelStateFromTempData]
public ActionResult Edit(int id)
{
  var category = new CategoryEntity(id);
  if (!ModelState.IsValid)
  {    
     TryUpdateModel<CategoryEntity>(category, 
         new ModelStateValueProviderFactory().GetValueProvider(ControllerContext));
  }
  return View(category);
}

Looking forward to your comments...

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文