Return several checkbox values of Html.CheckBoxFor to the controller

梦想与她 提交于 2020-07-18 07:53:08

问题


I have a view where the user should select several listed items by checkboxes, which should then be worked within the controller.

However, I am not getting the values passed to the POST method of the controller; the List comes in null when debugging.

While searching for answers, I was not able to find one matching with C# and the Html.CheckboxFor for multiple values.

I feel that this may be just a minor thing to fix, but I cannot figure it out for several days now.

Here is my model:

public class ShownListofTools// This List is given to the View to contain all Tools
{
    public IEnumerable<CheckboxItem> ToolsInShownList { get; set; } = new List<CheckboxItem>();         
}

public  class CheckboxItem //This Model is used to bring the Tools to the View
{
    public int ShownToolID { get; set; }
    public bool IsChecked { get; set; }
    public string DisplayName { get; set; }
    public int RemainingWorkTime { get; set; }
}

This is the controller:

[HttpGet]
public ActionResult IndexforLend()
{
        // 1)
        var notwornouttools = db.Tools.Where(t => t.Max_Worktime - t.CurrentWorktime > 0); // List Tools that are not worn out yet
        var tools = notwornouttools.Where(t => t.UserID == null);  //refine Filter to all tools that are currently not lent by other users                                                                          
        // 2)
        List<CheckboxItem> SelectListTools = new List<CheckboxItem>(); //temporary list, to gather all needed items

        // 3)
        foreach(var item in tools)//work down the "tool" list from above
        {
            SelectListTools.Add(
                new CheckboxItem()//Add a new Checkbox Item
                {
                     DisplayName = item.Tooltype + " " + item.Diameter+"mm", //Transfer the Attributes from the "tool" list
                     ShownToolID = item.ID,
                    IsChecked = false,
                    RemainingWorkTime = item.Max_Worktime - item.CurrentWorktime,
                }
             );
        }

        //Checkboxlist.ToolsInShownList = SelectListTools; //The Checkbox Items are Added to the List "Checkboxlist".
        return View(SelectListTools); // The List "Checkboxlist" is transferred to the View.
}               

//POST: Add Tools to a User
[HttpPost]
public ActionResult IndexforLend([Bind(Include = "IsChecked,ShownToolID")] List<CheckboxItem> chklist)  //import from View, but make it numerable         
{            
    foreach(var item in chklist) //go through the entire Checkboxlist
    {
        if (item.IsChecked == true) // if the Checkbox was Checked - not sure if only checked objects are returned
        {
            var id = item.ShownToolID; // take the Tool-ID to the variable "id"

            foreach(var realtool in db.Tools) //go through all Tools in Database
            {
                if (realtool.ID == id) // if the Id from the Checkbox is found
                {
                    realtool.UserID = LoggedInUser.ID; // set the Foreign Key UserID to the ID of the Logged in User
                    db.Entry(realtool).State = EntityState.Modified;
                    db.SaveChanges();
                }
            }
        };
    }

    return RedirectToAction("Index");
}

The view:

@model  List<WZ_Web2.Models.CheckboxItem>
   <p></p> 
<p></p>
<div>Please select the Tools you want to lend:</div>

@using (Html.BeginForm())
{
    @Html.AntiForgeryToken()
       <table class="table">
           @**@
        <tr>
            <th>
                Select
            </th>
            <th>
                Tool Name
            </th>
            <th>
                Remaining Work Time
            </th>                      
        </tr>

        @foreach (var item in Model)
        {
           <tr>
               <td>
                   @Html.CheckBoxFor(modelItem => item.IsChecked)
                   @Html.HiddenFor(modelItem => item.ShownToolID)

               </td>
               <td>
                   @Html.DisplayFor(modelItem => item.DisplayName)
               </td>              
               <td>
                   @Html.DisplayFor(modelItem => item.RemainingWorkTime)
               </td>

           </tr>
        }
    </table>
    <div class="col-md-offset-2 col-md-10">
        <input type="submit" value="submit" class="btn btn-default" />        
    </div>
}

回答1:


Using foreach that way it's not generating indexed name list in the post. As you iterate by item and set modelItem, it generates something like this in the post:

item.IsChecked=true&item.ShownToolID=1&item.IsChecked=false&item.ShownToolID=2...

But the controller is expecting an indexed list e.g.:

[0].IsChecked=true&[0].ShownToolID=1&[1].IsChecked=false&[1].ShownToolID=2...

As a result, mapper is unable to bind the data in your post. I've figured out by inspecting the post. Try using for instead...

In your view:

@for (int i = 0; i < Model.Count; i++ )
{
    <tr>
        <td>
            @Html.CheckBoxFor(item => Model[i].IsChecked)
            @Html.HiddenFor(item => Model[i].ShownToolID)

        </td>
        <td>
            @Html.DisplayFor(item => Model[i].DisplayName)
        </td>
        <td>
            @Html.DisplayFor(item => Model[i].RemainingWorkTime)
        </td>

    </tr>
}

In your controller:

[HttpPost]
public ActionResult IndexforLend(List<CheckboxItem> chklist)


来源:https://stackoverflow.com/questions/60360959/return-several-checkbox-values-of-html-checkboxfor-to-the-controller

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!