I have tried to simplify this problem as much as possible and cannot find a solution or fix.
I have followed the second approach ("Add metadata class") in this Microsoft tutorial https://learn.microsoft.com/en-us/aspnet/mvc/overview/getting-started/database-first-development/enhancing-data-validation?fbclid=IwAR0_eqraghf-faEqw1d0Fijw9D_su5BGIMMrwQQj4DgYwgiDht5NHLgx3BI
I am using data first with a Razor project. The database scaffolds without problem to create the dbContext. I want to annotate columns eg with Display(Name=""). This cannot go in the scaffolding generated files because they may get overwritten so the tutorial says to create metadata classes in a separate file and then a partial class in another separate file to merge the metadata into the scaffold generated classes. I understand all of this but it doesn't work.
To make it extremely simply I created a table Customer and after running scaffolding I edited the scaffold generated file so that everything was in the one file (so no potential problems with namespace - even though everything was in the same namespace anyway...).
So here is all of the code:
CREATE TABLE [dbo].[Customers] (
[Id] INT NOT NULL,
[Name] VARCHAR (30) NOT NULL,
[Details] VARCHAR (MAX) NOT NULL,
PRIMARY KEY CLUSTERED ([Id] ASC)
);
I then edited the Customer.cs class generated by scaffolding to include the metadata class and MetadataType property. The file is:
using System;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations;
namespace MyProject.Models
{
public partial class Customer
{
public int Id { get; set; }
// [Display(Name ="Customer name")]
// if I put the annotation here it works. If I comment this out and put it in the metadata class it doesn't work
public string Name { get; set; } = null!;
public string Details { get; set; } = null!;
}
[MetadataType(typeof(CustomerMetaData))]
public partial class Customer
{
}
public class CustomerMetaData
{
[Display(Name ="Customer name")]
public string Name";
}
}
So once I've done that I simply created a folder in the Pages folder and right clicked and Add > Razor Page > Razor Pages with EF (CRUD) to generate the default CRUD pages for the Customer class in the dbContext.
So that's it. Nothing complicated. Back in the Customer.cs class file if I put the [Display(Name="Customer Name"] in the Scaffolding generated class it works and the pages display "Customer name" instead of the column name "Name". However, comment out the annotation in the Customer class and put the MetaDataType attribute on a second partial class Customer and have a CustomerMetaData class that includes the annotation and it doesn't work.
I can not get it to work.
The problem is compounded because you cannot see the annotations if you "watch" a data object during debug
I'm experiencing the same issue. I'm in .Net 6, ASP.NET, MVC, with EF Core.
The only way I'm able to get my display name is directly in the view. I can live with that but what's killing me is I can't control the validation stuff that way.
Update:
I somewhat have a hack working. I used a subclass of the model and put the annotations on that. The subclass includes only a copy of the fields I wanted to annotate. I did have to use the [new] keyword but those properties are throw-away anyway. I then adjust the view to use that subclass as the @model.
That's all I changed. I leave the EF context pointing to the original class. The same with all the controllers. The dependency injection gnomes look at what the controller is expecting (the original class) and pipe it in that way.
I'm not 100% happy with this, though. Even though the code impact is the same thing as the metadata class and attribute combo, it's not visibly understandable as to what or why it's doing it. And there is the hidden dependency on dependency injection working correctly.
Update #2:
I changed all of this around to use the data annotation applied to an interface instead. I was able to apply the interface to "user safe" side of the model partial classes, just a stub really, and changed the @model tag on the views to use the interface. I think it's much cleaner and there isn't some hidden context switching between classes.