Easiest way to create a cascade dropdown in ASP.NET MVC 3 with C#
Asked Answered
R

1

68

I want to create two DropDownList in a cascade using MVC3 (preferably Razor) with C#.

I would like to have one dropdown where you can choose the year and another one where you can choose a specific set of months depending on the selected year.

Let's put it simple. When I choose the current year (i.e. 2011) in the dropdown list "year", the dropdown list "month" gets populated with the months until the current month (i.e. March). For the other cases (other years) no restriction is given. Moreover it would be nice to "block" the dropdown list "month" before any element in the dropdown list "year" is selected.

I already looked in the Internet for some solutions, using jQuery or even homemade approaches, but they all refer to past versions of MVC and some commands are deprecated in MVC3.

Rovner answered 31/3, 2011 at 9:10 Comment(0)
C
139

As always you start with a model:

public class MyViewModel
{
    public int? Year { get; set; }
    public int? Month { get; set; }

    public IEnumerable<SelectListItem> Years
    {
        get
        {
            return Enumerable.Range(2000, 12).Select(x => new SelectListItem
            {
                Value = x.ToString(),
                Text = x.ToString()
            });
        }
    }
}

then a controller:

public class HomeController : Controller
{
    public ActionResult Index()
    {
        var model = new MyViewModel();
        return View(model);
    }

    public ActionResult Months(int year)
    {
        if (year == 2011)
        {
            return Json(
                Enumerable.Range(1, 3).Select(x => new { value = x, text = x }), 
                JsonRequestBehavior.AllowGet
            );
        }
        return Json(
            Enumerable.Range(1, 12).Select(x => new { value = x, text = x }),
            JsonRequestBehavior.AllowGet
        );
    }
}

and finally a view:

@model AppName.Models.MyViewModel

@Html.DropDownListFor(
    x => x.Year, 
    new SelectList(Model.Years, "Value", "Text"),
    "-- select year --"
)

@Html.DropDownListFor(
    x => x.Month, 
    Enumerable.Empty<SelectListItem>(),
    "-- select month --"
)

<script type="text/javascript">
    $('#Year').change(function () {
        var selectedYear = $(this).val();
        if (selectedYear != null && selectedYear != '') {
            $.getJSON('@Url.Action("Months")', { year: selectedYear }, function (months) {
                var monthsSelect = $('#Month');
                monthsSelect.empty();
                $.each(months, function (index, month) {
                    monthsSelect.append($('<option/>', {
                        value: month.value,
                        text: month.text
                    }));
                });
            });
        }
    });
</script>

Obviously you will notice that in my example I have hardcoded all the values. You should improve this logic by using notions like current year, current month, probably even fetch those values from a repository, etc... but for the purpose of the demonstration this should be enough to put you on the right track.

Convalesce answered 31/3, 2011 at 9:21 Comment(9)
do I need to add any special AJAX references inside the program? ThanksRovner
@Francesco, yes, you need jquery.Convalesce
Whats with the <option/>? I don't get it.Norty
Roberto Bonini, <option /> is the selectlistitem. It is appended inside the list element.Brothel
This example produces empty <option/> elements. This is the only place I've seen this notation for the append() method. Does this require an additional jQuery plugin? Thanks!Customhouse
Thanks Darin, is it the shortest type of binding two dropdownlist? Can't we do anything special in the model code first, so that this applies automatically ?Spirogyra
Perfect! The only thing I needed to improve is to restore "-- select month --" label. It gets loaded initially, but then is removed in monthsSelect.empty();Aeronaut
Simple example that really works. Darin, can you explain why we have to add the $ in front of <option>. I know the $ makes it a jquery element.. I thought it should work without the $...Can you explain it a little bit..Thanks. monthsSelect.append($('<option/>', { value: month.value, text: month.text }));Holguin
Straight to the point answer. Upvoted, my friend! Correct me if I'm wrong but the action method should return a JsonResult, e.g. public JsonResult Months(int year)Secrete

© 2022 - 2024 — McMap. All rights reserved.