2
votes

I am following this example

enter link description here

But when I submit, it says can't convert 'false' to 'int'

as I assume its the false or true that is getting passed, not the actual value

what am I doing wrong?

My model is

  public IEnumerable<AllocateRequirementViewModel> Requirements { get; set; }

  public List<int> RequirementIds { get; set; }

then my razor is

  <div id="RequirementsContainer">
            @foreach (var requirement in Model.Requirements)
            {
                <div class="row">
                    <input id="@requirement.Id" type="checkbox" name="RequirementIds" value="@requirement.Id" /> @requirement.Description
                </div>
            }
        </div>
1

1 Answers

1
votes

your model is going to need to contain both the checked values and all the possible values.

public class TestViewModel
{
    public Guid Id { get; set; }
    public IDictionary<Guid, String> AllCheckboxOptions { get; set; }
    public Guid[] CheckedOptions { get; set; }
}

I use Guids as PKs in my database so these are the IDs from the database

I then created a number of extension methods to generate the html

    public static MvcHtmlString CheckBoxList(this HtmlHelper htmlHelper, String name, List<SelectListItem> listInfo,
        IDictionary<String, Object> htmlAttributes, Int32 columns)
    {
        if (String.IsNullOrEmpty(name))
            throw new ArgumentException("The argument must have a value", "name");
        if (listInfo == null)
            throw new ArgumentNullException("listInfo");
        if (!listInfo.Any())
            return new MvcHtmlString(String.Empty);

        var outerSb = new StringBuilder();

        for (Int32 i = 0; i < columns; i++)
        {
            var listBuilder = new TagBuilder("ul");
            if (columns > 1)
                listBuilder.MergeAttribute("class", "column");

            var innerSb = new StringBuilder();

            var take = listInfo.Count % columns == 0
                ? listInfo.Count / columns
                : (listInfo.Count / columns) + 1;

            var items = listInfo.Skip(i * take).Take(take);

            foreach (var info in items)
            {
                var inputBuilder = new TagBuilder("input");
                if (info.Selected) inputBuilder.MergeAttribute("checked", "checked");
                inputBuilder.MergeAttribute("type", "checkbox");
                inputBuilder.MergeAttribute("value", info.Value);
                inputBuilder.MergeAttribute("id", info.Value);
                inputBuilder.MergeAttribute("name", info.Value);

                var labelBuilder = new TagBuilder("label");
                labelBuilder.MergeAttribute("for", @info.Value);
                labelBuilder.InnerHtml = info.Text;

                var listItemWrapper = new TagBuilder("li");
                //may have to encode here. 
                listItemWrapper.InnerHtml = inputBuilder.ToString(TagRenderMode.SelfClosing) + labelBuilder.ToString(TagRenderMode.Normal);

                innerSb.Append(listItemWrapper.ToString(TagRenderMode.Normal));
            }

            listBuilder.InnerHtml = innerSb.ToString();
            outerSb.Append(listBuilder.ToString(TagRenderMode.Normal));
        }

        return new MvcHtmlString(outerSb.ToString());
    }

then in the view you can call your extension method

    @Html.CheckBoxList("yourtypes", (from o in Model.AllCheckboxOptions
                                        select new SelectListItem
                                        {
                                            Text = o.Value,
                                            Selected = Model.CheckedOptions.Contains(o.Key),
                                            Value = o.Key.ToString()
                                        }).ToList())

I've used this technique with both MVC3 and MVC4 with success.