Search code examples
c#asp.net-mvcasp.net-mvc-viewmodel

Having problem using the viewmodel in the view


So I am trying to use a viewmodel in one of my Views but I am getting a InvalidOperationException: The model item passed into the ViewDataDictionary is of type 'SchoolProject.Models.Student', but this ViewDataDictionary instance requires a model item of type 'SchoolProject.ViewModels.StudentViewModel'.

Here is my models that forms the view model :

Student.cs

public class Student {

    [Key]
    public int StudentID { get; set; }
    [Required]
    public string Name { get; set; }
    public string Surname { get; set; }
    public int? Age { get; set; }

    public ICollection<Enrollment> Enrollments { get; set; }



}

Course.cs

public class Course
{
    [DatabaseGenerated(DatabaseGeneratedOption.None)]
    public int CourseID { get; set; }
    public string? Title { get; set; }
    public int Credit { get; set; }

    public ICollection<Enrollment> Enrollments { get; set; }


}

And the StudentViewModel :

public class StudentViewModel
{
    public Student Student { get; set; }
    public List<Course> Courses { get; set; }

}

The action method :

public async Task<IActionResult> Details(int id)
    {
        
        var student = await _context.Student.FirstOrDefaultAsync(m => m.StudentID == id);
        return View(student);
    }

Lastly, The view :

@model SchoolProject.ViewModels.StudentViewModel
@{
IEnumerable<Course> courses = ViewData["Courses"] as IEnumerable<Course>;

}

<html>
<body>
    <h1 style="text-align : left" >Öğrenci Bilgileri</h1>
    <br></br>
    <br></br>
    <p class="bx" style=" display : inline-block">Öğrenci Adı : </p>
    <p class="bx1" style="display : inline-block"> @Html.DisplayFor(model => model.Student.Name)</p>
    <br></br>
    <p class="bx" style=" display : inline-block">Öğrenci Soyadı : </p>
    <p class="bx1" style="display : inline-block"> @Html.DisplayFor(model => model.Student.Surname)</p>
    <br></br>
    <p class="bx" style=" display : inline-block">Öğrenci Yaşı : </p>
    <p class="bx1" style="display : inline-block"> @Html.DisplayFor(model => model.Student.Age)</p>
    <h1 style="position : absolute; top : 80px; left: 700px"> Öğrencinin Aldığı dersler</h1>
    <div style="position : absolute; left : 600px; top: 100px" class="vl"></div>
@foreach(var ders in Model.Courses)
{
    @Html.DisplayFor(modelItem => ders.CourseID)
}
<table class="table">
        <thead>
            <tr>
                <td>
                    Dersler
                </td>
                <td>
                    Kredi
                </td>
            </tr>
        </thead>
        <tbody>
        @foreach(var course in Model.Courses)
        {
            <tr>
                <td>
                    @Html.DisplayFor(model => course.Title)
                </td>
                <td>
                    @Html.DisplayFor(model => course.Credit)
                </td>
            </tr>
        }
    </tbody>
    </table>
</body>

I know I do not even return a viewmodel instance in the action method, but I just simply do not know how can I do that. I am looking forward to learn how to implement viewmodels to view.


Solution

  • you should pass StudentViewModel into the view like this :

    var student = await _context.Student.FirstOrDefaultAsync(m => m.StudentID == id);
    var cousre = await _context.course(/*on related student column*/).ToListAsync();    
    return View(  new StudentViewModel{
    Student = student , 
    Courses  = cousre
    });