通过ViewModel将值从多个DropDown列表传递给控制器​​时出错

时间:2013-06-25 23:34:35

标签: asp.net asp.net-mvc asp.net-mvc-4 viewmodel

我正在尝试创建由 SelectList 创建的表单DropDownLists,这些表单基于与学生用户关联的不同[Bank]帐户生成。到目前为止,我已经能够为所有帐户成功创建两个下拉列表,但是当我提交表单时,我收到错误:没有为此对象定义无参数构造函数。

我不确定这个错误的原因是什么我认为我设置了我的控制器并正确查看以处理不同的值,但我不确定我传入的ViewModel是否正确。

帐户模型(转移ActionResult中使用的银行帐户)

    public class Account
{
    [Key]
    [Editable(false)]
    [DatabaseGeneratedAttribute(DatabaseGeneratedOption.Identity)]
    public virtual int AccountID { get; set; }

    [Required(ErrorMessage = "Required")]
    [MaxLength(50)]
    public virtual string AccountName { get; set; }

    [DisplayFormat(DataFormatString = "{0:c}")]
    [Column(TypeName = "money")]
    public virtual decimal AccountTotal { get; set; }

    [ForeignKey("Student")]
    public virtual int StudentID { get; set; }

    public virtual Student Student { get; set; }

}

TransferFundsViewModel (这是我的强类型视图构建的类,我试图将其传递给我的控制器。)

    public class TransferFundsViewModel
{
    public SelectList ListOfSourceAccounts { get; set; }
    public int SelectedSourceAccountId { get; set; }

    public SelectList ListOfDestinationAccounts { get; set; }
    public int SelectedDestinationAccountId { get; set; }

    [Required]
    [DisplayFormat(DataFormatString = "{0:c}")]
    public decimal TransferAmount { get; set; }
}

我的控制器内的TransferFunds方法

[HttpGet]
    public ActionResult TransferFunds()
    {
        //Get logged in user
        var studentProfile = db.UserProfiles.Local.SingleOrDefault(u => u.UserName == User.Identity.Name)
            ?? db.UserProfiles.SingleOrDefault(u => u.UserName == User.Identity.Name);
        //Get student account associated with logged in user
        var student = db.Students.Find(studentProfile.UserId);
        //Get accounts associated with student account
        var accounts = db.Accounts.Where(x => x.StudentID == student.UserId);

        var selectionList = new SelectList(accounts, "AccountID", "AccountName");

        var vm = new TransferFundsViewModel { ListOfSourceAccounts = selectionList, ListOfDestinationAccounts = selectionList };

        return View(vm);
    }

    [HttpPost]
    public ActionResult TransferFunds(TransferFundsViewModel viewModel)
    {
        if (ModelState.IsValid)
        {
            try
            {
                AccountManager manager = new AccountManager();
                Account srcAccount = db.Accounts.Find(viewModel.SelectedSourceAccountId);
                Account destAccount = db.Accounts.Find(viewModel.SelectedDestinationAccountId);

                manager.TransferMoney(srcAccount, destAccount, viewModel.TransferAmount); //From Account Manager
                db.SaveChanges();
            }
            catch (DataException)
            {
                ModelState.AddModelError("", "Something went wrong, transfer could not be completed.");
            }
        }
        return RedirectToAction("AccountSummary");
    }

AccountManager (这是掌握实际转移资金的逻辑的类。)

    public class AccountManager
{
    public void TransferMoney(Account srcAccount, Account destAccount, decimal transferAmount)
    {
        if (srcAccount.AccountTotal >= transferAmount)
        {
            srcAccount.AccountTotal -= transferAmount;
            destAccount.AccountTotal += transferAmount;
        }
    }
}

最后,转移资金视图

@model eBank.ViewModels.TransferFundsViewModel
@using (Html.BeginForm()) {
    <fieldset>
        <legend>Transfer Funds</legend>

        <div class="editor-label">
            Source Account
        </div>
        <div class="editor-field">
            @Html.DropDownListFor(viewModel => viewModel.ListOfSourceAccounts, Model.ListOfSourceAccounts, "Choose an Account")
        </div>

        <div class="editor-label">
            Destination Account
        </div>
        <div class="editor-field">
            @Html.DropDownListFor(viewModel => viewModel.ListOfDestinationAccounts, Model.ListOfDestinationAccounts, "Choose an Account")
        </div>

        <div class="editor-label">
            Transfer Amount
        </div>
        <div class="editor-field">
            @Html.EditorFor(viewModel => viewModel.TransferAmount)
        </div>

        <p>
            <input type="submit" value="Transfer" />
        </p>
    </fieldset>
}

1 个答案:

答案 0 :(得分:2)

创建下拉列表时,您需要更改第一个参数以使用SelectedSourceAccountId。

更改

@Html.DropDownListFor(viewModel => viewModel.ListOfSourceAccounts, Model.ListOfSourceAccounts, "Choose an Account")

有关

@Html.DropDownListFor(viewModel => viewModel.SelectedSourceAccountId, Model.ListOfSourceAccounts, "Choose an Account")

与其他Dropdownlist相同。

现在你的模型应该有价值,srcAccount和destAccount对象不会为空。