π Multiple Controllers
So far we've been using a single controller to handle our routes. However, larger projects often use many controllers and it's common to create a controller for each model in our application. In this lesson, we'll walk through the correct way to implement more than one controller.
Project Structureβ
Let's alter our To Do List to use multiple controllers. Our HomeController will handle a central homepage to greet users. It will be handled by a HomeController. Then, all logic, routes, and views meant to work with To Do List Items will be handled by a new ItemsController.
- Create an ItemsController.csfile in the existingToDoList/Controllersdirectory.
- Create a corresponding Itemssubdirectory in theToDoList/Viewsdirectory.
- Move the existing Views/Home/Index.cshtmlfile into theViews/Itemssubdirectory.
- Move our existing Views/Home/CreateForm.cshtmlfile into theViews/Itemssubdirectory.
- Add a new, empty Index.cshtmlfile into theViews/Homedirectory.
The resulting project structure should look like this:
ToDoList.Solution
βββ ToDoList
β   βββ Controllers
βΒ Β  βΒ Β  βββ HomeController.cs
βΒ Β  βΒ Β  βββ ItemsController.cs
βΒ Β  βββ Models
βΒ Β  βΒ Β  βββ Item.cs
βΒ Β  βββ Program.cs
βΒ Β  βββ ToDoList.csproj
βΒ Β  βββ Views
βΒ Β      βββ Home
βΒ Β      βΒ Β  βββ Index.cshtml
βΒ Β      βββ Items
βΒ Β          βββ CreateForm.cshtml
βΒ Β          βββ Index.cshtml
βββ ToDoList.Tests
ItemsControllerβ
We'll cut the contents of HomeController.cs and paste them into ItemsController.cs. This controller will be responsible for handling the routes that pertain to Items. By convention, we pluralize the class name, which is why it's ItemsController not ItemController.
Then we'll update the class name to state ItemsController instead of HomeController:
...
namespace ToDoList.Controllers
{
    public class ItemsController : Controller
    {
      ...
   ...
...
Next, we'll need to alter the path of our Index() route:
...
    [HttpGet("/items")]
    public ActionResult Index()
    {
      List<Item> allItems = Item.GetAll();
      return View(allItems);
    }
...
Here we simply change the path in the route's decorator to read "/items" instead of "/".  The entire updated ItemsController.cs file should look like this:
using Microsoft.AspNetCore.Mvc;
using ToDoList.Models;
using System.Collections.Generic;
namespace ToDoList.Controllers
{
  public class ItemsController : Controller
  {
    [HttpGet("/items")]
    public ActionResult Index()
    {
      List<Item> allItems = Item.GetAll();
      return View(allItems);
    }
    [HttpGet("/items/new")]
    public ActionResult CreateForm()
    {
      return View();
    }
    [HttpPost("/items")]
    public ActionResult Create(string description)
    {
      Item myItem = new Item(description);
      return RedirectToAction("Index");
    }
  }
}
Note that we now have two routes that use the "/items" path. We can do this because GET and POST are two different requests. We use
[HttpGet()] for the GET request and [HttpPost()] for the POST request. This way the server can easily tell them apart even if they use the same path.
Views for ItemsController Routesβ
Letβs revisit the views we've created for this controller. .NET will automatically look in a Views subdirectory that has the same name as the controller. This means our new ItemsController's views will reside in the Views/Items directory we just created.
We've already created Index.cshtml and CreateForm.cshtml and then moved them to the new Views/Items subdirectory so our views require no further changes.
HomeControllerβ
Let's build our HomeController now. It should look like this:
using Microsoft.AspNetCore.Mvc;
namespace ToDoList.Controllers
{
    public class HomeController : Controller
    {
      [HttpGet("/")]
      public ActionResult Index()
      {
        return View();
      }
    }
}
Views for HomeController Routesβ
Finally, we need to create a new Index view in the Home folder inside of the Views folder:
<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8">
    <title>My To-Do List!</title>
    <link 
      href="https://cdn.jsdelivr.net/npm/bootstrap@5.2.3/dist/css/bootstrap.min.css" 
      rel="stylesheet" 
      integrity="sha384-rbsA2VBKQhggwzxH7pPCaAqO46MgnOM80zW1RWuH61DGLwZJEdK2Kadq2F9CUG65" 
      crossorigin="anonymous">
  </head>
  <body>
    <h1>Welcome to the To Do List!</h1>
    <a href='/items/new'>Add a new item</a>
    <br />
    <a href='/items'>See list of items</a>
  </body>
</html>
We've successfully separated our code so that it uses two controllers. One handles Items while the other handles the home page.
Repository Referenceβ
Follow the link below to view how a sample version of the project should look at this point. Note that this is a link to a specific branch in the repository.