我一直在研究模型绑定以解决我遇到的特定问题。我尝试了各种博客和stackoverflow答案中描述的一些方法,但是我并没有真正到达那里。

首先,我将介绍我的模型:

public class CampaignModel
{
    [Required]
    [StringLength(24)]
    [Display(Name = "CampaignModel_Name", Prompt = "CampaignModel_Name", ResourceType = typeof(CampaignResources))]
    public string Name { get; set; }

    [StringLength(255)]
    [Display(Name = "CampaignModel_Description", Prompt = "CampaignModel_Description", ResourceType = typeof(CampaignResources))]
    public string Description { get; set; }

    [Required]
    [DataType(DataType.Date)]
    [Display(Name = "CampaignModel_StartDate", ResourceType = typeof(CampaignResources))]
    public DateTime StartDate { get; set; }

    [Required]
    [DataType(DataType.Date)]
    [Display(Name = "CampaignModel_EndDate", ResourceType = typeof(CampaignResources))]
    public DateTime EndDate { get; set; }

    [Display(Name = "CampaignModel_Tags", Prompt = "CampaignModel_Tags", ResourceType = typeof(CampaignResources))]
    public TagList Tags { get; set; }
}


因此,这是一个相当基本的模型,除了最后一个属性TagList。现在,将来我的模型将分配一个TagList,所以这对我来说非常重要。 TagList就是这样:

public class TagList : List<Tag>
{
}


我创建此类是为了能够轻松为其创建EditorTemplate,而无需放置UIHint属性。现在,我在标签列表编辑器中使用了select2.js库,该库可处理现有标签中的Ajax搜索等。这里的问题是Select2绑定到一个隐藏字段,在其中它用a分隔各种标记值,并且根据它是现有标记还是新标记,它使用文本或id生成类似1,tag,34,my new tag的列表。我想将此输入转换为TagList。

因此,具体问题是:
我该如何将这个隐藏的输入模型绑定到我的模型的TagList属性中,并能够轻松地对我的所有模型重用此行为?

编辑=>添加基于Andrei的答案创建的EditorTemplate和ModelBinder的代码

我的EditorTemplate(省略的js)

<div class="form-group">
    @Html.Label(string.Empty, ViewData.ModelMetadata.DisplayName, new { @class = "col-md-3 control-label" })
    <div class="col-md-9">
        @Html.Hidden(string.Empty, ViewData.TemplateInfo.FormattedModelValue, new { @class = "form-control select2" })
    </div>
</div>


我的ModelBinder(在global.asax中设置为DefaultModelBinder)

public class TagListModelBinder : DefaultModelBinder
{
    protected override void BindProperty(
        ControllerContext controllerContext,
        ModelBindingContext bindingContext,
        PropertyDescriptor propertyDescriptor)
    {
        if (propertyDescriptor.PropertyType == typeof(TagList))
        {
            ValueProviderResult value = bindingContext.ValueProvider.GetValue(propertyDescriptor.Name);
            string[] rawTags = value.AttemptedValue.Split(',');
            List<long> tagIds = new List<long>();

            TagList tags = new TagList();
            foreach (string rawTag in rawTags)
            {
                long id;
                if (long.TryParse(rawTag, out id))
                {
                    // Existing tags need to be retrieved from DB
                    tagIds.Add(id);
                }
                else
                {
                    // New tags can simply be added without ID
                    tags.Add(new Tag { Text = rawTag });
                }
            }

            if (tagIds.Count > 0)
            {
                using (TagServiceClient client = new TagServiceClient())
                {
                    List<Services.TagService.Tag> existingTags = client.GetTagsByIds(tagIds);
                    tags.AddRange(existingTags.Select(t => new Tag { Id = t.id, Text = t.text }));
                }
            }

            propertyDescriptor.SetValue(bindingContext.Model, tags);
        }

        base.BindProperty(controllerContext, bindingContext, propertyDescriptor);
    }

最佳答案

最好的方法是从DefaultModelBinder继承并检查您要处理的属性。如果它是TagList类型-继续并应用所需的任何逻辑。否则,让默认实现完成工作。

public class TagListModelBinder : DefaultModelBinder
{
    protected override void BindProperty(ControllerContext controllerContext, ModelBindingContext bindingContext, System.ComponentModel.PropertyDescriptor propertyDescriptor )
    {
        if (propertyDescriptor.PropertyType == typeof(TagList))
        {
            ValueProviderResult value = bindingContext.ValueProvider.GetValue(propertyDescriptor.Name);
            string[] rawTags = value.ToString().Split(',');

            TagList tags = new TagList();
            foreach (string rawTag in rawTags)
            {
                // for numbers - get them from DB
                // for strings - create new and store in DB
                // then add them to tags
            }

            propertyDescriptor.SetValue(bindingContext.Model, tags);
        }
        else
        {
            base.BindProperty(controllerContext, bindingContext, propertyDescriptor)
        }
    }
}

10-08 06:12