diff --git a/CutList.Web/Migrations/20260205214055_FixDimensionDiscriminatorValues.Designer.cs b/CutList.Web/Migrations/20260205214055_FixDimensionDiscriminatorValues.Designer.cs
new file mode 100644
index 0000000..0b44728
--- /dev/null
+++ b/CutList.Web/Migrations/20260205214055_FixDimensionDiscriminatorValues.Designer.cs
@@ -0,0 +1,823 @@
+//
+using System;
+using CutList.Web.Data;
+using Microsoft.EntityFrameworkCore;
+using Microsoft.EntityFrameworkCore.Infrastructure;
+using Microsoft.EntityFrameworkCore.Metadata;
+using Microsoft.EntityFrameworkCore.Migrations;
+using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
+
+#nullable disable
+
+namespace CutList.Web.Migrations
+{
+ [DbContext(typeof(ApplicationDbContext))]
+ [Migration("20260205214055_FixDimensionDiscriminatorValues")]
+ partial class FixDimensionDiscriminatorValues
+ {
+ ///
+ protected override void BuildTargetModel(ModelBuilder modelBuilder)
+ {
+#pragma warning disable 612, 618
+ modelBuilder
+ .HasAnnotation("ProductVersion", "8.0.11")
+ .HasAnnotation("Relational:MaxIdentifierLength", 128);
+
+ SqlServerModelBuilderExtensions.UseIdentityColumns(modelBuilder);
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.CuttingTool", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("IsActive")
+ .HasColumnType("bit");
+
+ b.Property("IsDefault")
+ .HasColumnType("bit");
+
+ b.Property("KerfInches")
+ .HasPrecision(6, 4)
+ .HasColumnType("decimal(6,4)");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasMaxLength(50)
+ .HasColumnType("nvarchar(50)");
+
+ b.HasKey("Id");
+
+ b.ToTable("CuttingTools");
+
+ b.HasData(
+ new
+ {
+ Id = 1,
+ IsActive = true,
+ IsDefault = true,
+ KerfInches = 0.0625m,
+ Name = "Bandsaw"
+ },
+ new
+ {
+ Id = 2,
+ IsActive = true,
+ IsDefault = false,
+ KerfInches = 0.125m,
+ Name = "Chop Saw"
+ },
+ new
+ {
+ Id = 3,
+ IsActive = true,
+ IsDefault = false,
+ KerfInches = 0.0625m,
+ Name = "Cold Cut Saw"
+ },
+ new
+ {
+ Id = 4,
+ IsActive = true,
+ IsDefault = false,
+ KerfInches = 0.0625m,
+ Name = "Hacksaw"
+ });
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.Job", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("CreatedAt")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("datetime2")
+ .HasDefaultValueSql("GETUTCDATE()");
+
+ b.Property("Customer")
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("CuttingToolId")
+ .HasColumnType("int");
+
+ b.Property("JobNumber")
+ .IsRequired()
+ .HasMaxLength(20)
+ .HasColumnType("nvarchar(20)");
+
+ b.Property("Name")
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("Notes")
+ .HasColumnType("nvarchar(max)");
+
+ b.Property("UpdatedAt")
+ .HasColumnType("datetime2");
+
+ b.HasKey("Id");
+
+ b.HasIndex("CuttingToolId");
+
+ b.HasIndex("JobNumber")
+ .IsUnique();
+
+ b.ToTable("Jobs");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.JobPart", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("JobId")
+ .HasColumnType("int");
+
+ b.Property("LengthInches")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.Property("MaterialId")
+ .HasColumnType("int");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("Quantity")
+ .HasColumnType("int");
+
+ b.Property("SortOrder")
+ .HasColumnType("int");
+
+ b.HasKey("Id");
+
+ b.HasIndex("JobId");
+
+ b.HasIndex("MaterialId");
+
+ b.ToTable("JobParts");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.JobStock", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("IsCustomLength")
+ .HasColumnType("bit");
+
+ b.Property("JobId")
+ .HasColumnType("int");
+
+ b.Property("LengthInches")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.Property("MaterialId")
+ .HasColumnType("int");
+
+ b.Property("Priority")
+ .HasColumnType("int");
+
+ b.Property("Quantity")
+ .HasColumnType("int");
+
+ b.Property("SortOrder")
+ .HasColumnType("int");
+
+ b.Property("StockItemId")
+ .HasColumnType("int");
+
+ b.HasKey("Id");
+
+ b.HasIndex("JobId");
+
+ b.HasIndex("MaterialId");
+
+ b.HasIndex("StockItemId");
+
+ b.ToTable("JobStocks");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.Material", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("CreatedAt")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("datetime2")
+ .HasDefaultValueSql("GETUTCDATE()");
+
+ b.Property("Description")
+ .HasMaxLength(255)
+ .HasColumnType("nvarchar(255)");
+
+ b.Property("Grade")
+ .HasMaxLength(50)
+ .HasColumnType("nvarchar(50)");
+
+ b.Property("IsActive")
+ .HasColumnType("bit");
+
+ b.Property("Shape")
+ .IsRequired()
+ .HasMaxLength(50)
+ .HasColumnType("nvarchar(50)");
+
+ b.Property("Size")
+ .IsRequired()
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("SortOrder")
+ .HasColumnType("int");
+
+ b.Property("Type")
+ .IsRequired()
+ .HasMaxLength(20)
+ .HasColumnType("nvarchar(20)");
+
+ b.Property("UpdatedAt")
+ .HasColumnType("datetime2");
+
+ b.HasKey("Id");
+
+ b.ToTable("Materials");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.MaterialDimensions", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("DimensionType")
+ .IsRequired()
+ .HasMaxLength(21)
+ .HasColumnType("nvarchar(21)");
+
+ b.Property("MaterialId")
+ .HasColumnType("int");
+
+ b.HasKey("Id");
+
+ b.HasIndex("MaterialId")
+ .IsUnique();
+
+ b.ToTable("MaterialDimensions");
+
+ b.HasDiscriminator("DimensionType").HasValue("MaterialDimensions");
+
+ b.UseTphMappingStrategy();
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.StockItem", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("CreatedAt")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("datetime2")
+ .HasDefaultValueSql("GETUTCDATE()");
+
+ b.Property("IsActive")
+ .HasColumnType("bit");
+
+ b.Property("LengthInches")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.Property("MaterialId")
+ .HasColumnType("int");
+
+ b.Property("Name")
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("Notes")
+ .HasMaxLength(255)
+ .HasColumnType("nvarchar(255)");
+
+ b.Property("QuantityOnHand")
+ .HasColumnType("int");
+
+ b.Property("UpdatedAt")
+ .HasColumnType("datetime2");
+
+ b.HasKey("Id");
+
+ b.HasIndex("MaterialId", "LengthInches")
+ .IsUnique();
+
+ b.ToTable("StockItems");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.StockTransaction", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("CreatedAt")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("datetime2")
+ .HasDefaultValueSql("GETUTCDATE()");
+
+ b.Property("JobId")
+ .HasColumnType("int");
+
+ b.Property("Notes")
+ .HasMaxLength(500)
+ .HasColumnType("nvarchar(500)");
+
+ b.Property("Quantity")
+ .HasColumnType("int");
+
+ b.Property("StockItemId")
+ .HasColumnType("int");
+
+ b.Property("SupplierId")
+ .HasColumnType("int");
+
+ b.Property("Type")
+ .HasColumnType("int");
+
+ b.Property("UnitPrice")
+ .HasPrecision(10, 2)
+ .HasColumnType("decimal(10,2)");
+
+ b.HasKey("Id");
+
+ b.HasIndex("JobId");
+
+ b.HasIndex("StockItemId");
+
+ b.HasIndex("SupplierId");
+
+ b.ToTable("StockTransactions");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.Supplier", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("ContactInfo")
+ .HasMaxLength(500)
+ .HasColumnType("nvarchar(500)");
+
+ b.Property("CreatedAt")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("datetime2")
+ .HasDefaultValueSql("GETUTCDATE()");
+
+ b.Property("IsActive")
+ .HasColumnType("bit");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("Notes")
+ .HasColumnType("nvarchar(max)");
+
+ b.HasKey("Id");
+
+ b.ToTable("Suppliers");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.SupplierOffering", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("IsActive")
+ .HasColumnType("bit");
+
+ b.Property("Notes")
+ .HasMaxLength(255)
+ .HasColumnType("nvarchar(255)");
+
+ b.Property("PartNumber")
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("Price")
+ .HasPrecision(10, 2)
+ .HasColumnType("decimal(10,2)");
+
+ b.Property("StockItemId")
+ .HasColumnType("int");
+
+ b.Property("SupplierDescription")
+ .HasMaxLength(255)
+ .HasColumnType("nvarchar(255)");
+
+ b.Property("SupplierId")
+ .HasColumnType("int");
+
+ b.HasKey("Id");
+
+ b.HasIndex("StockItemId");
+
+ b.HasIndex("SupplierId", "StockItemId")
+ .IsUnique();
+
+ b.ToTable("SupplierOfferings");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.AngleDimensions", b =>
+ {
+ b.HasBaseType("CutList.Web.Data.Entities.MaterialDimensions");
+
+ b.Property("Leg1")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.Property("Leg2")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.Property("Thickness")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Thickness");
+
+ b.HasIndex("Leg1");
+
+ b.HasDiscriminator().HasValue("Angle");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.ChannelDimensions", b =>
+ {
+ b.HasBaseType("CutList.Web.Data.Entities.MaterialDimensions");
+
+ b.Property("Flange")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.Property("Height")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Height");
+
+ b.Property("Web")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.HasIndex("Height");
+
+ b.HasDiscriminator().HasValue("Channel");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.FlatBarDimensions", b =>
+ {
+ b.HasBaseType("CutList.Web.Data.Entities.MaterialDimensions");
+
+ b.Property("Thickness")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Thickness");
+
+ b.Property("Width")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Width");
+
+ b.HasIndex("Width");
+
+ b.HasDiscriminator().HasValue("FlatBar");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.IBeamDimensions", b =>
+ {
+ b.HasBaseType("CutList.Web.Data.Entities.MaterialDimensions");
+
+ b.Property("Height")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Height");
+
+ b.Property("WeightPerFoot")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.HasIndex("Height");
+
+ b.HasDiscriminator().HasValue("IBeam");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.PipeDimensions", b =>
+ {
+ b.HasBaseType("CutList.Web.Data.Entities.MaterialDimensions");
+
+ b.Property("NominalSize")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.Property("Schedule")
+ .HasMaxLength(20)
+ .HasColumnType("nvarchar(20)");
+
+ b.Property("Wall")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Wall");
+
+ b.HasIndex("NominalSize");
+
+ b.HasDiscriminator().HasValue("Pipe");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.RectangularTubeDimensions", b =>
+ {
+ b.HasBaseType("CutList.Web.Data.Entities.MaterialDimensions");
+
+ b.Property("Height")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Height");
+
+ b.Property("Wall")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Wall");
+
+ b.Property("Width")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Width");
+
+ b.HasIndex("Width");
+
+ b.HasDiscriminator().HasValue("RectangularTube");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.RoundBarDimensions", b =>
+ {
+ b.HasBaseType("CutList.Web.Data.Entities.MaterialDimensions");
+
+ b.Property("Diameter")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.HasIndex("Diameter");
+
+ b.HasDiscriminator().HasValue("RoundBar");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.RoundTubeDimensions", b =>
+ {
+ b.HasBaseType("CutList.Web.Data.Entities.MaterialDimensions");
+
+ b.Property("OuterDiameter")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.Property("Wall")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Wall");
+
+ b.HasIndex("OuterDiameter");
+
+ b.HasDiscriminator().HasValue("RoundTube");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.SquareBarDimensions", b =>
+ {
+ b.HasBaseType("CutList.Web.Data.Entities.MaterialDimensions");
+
+ b.Property("Size")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Size");
+
+ b.HasIndex("Size");
+
+ b.HasDiscriminator().HasValue("SquareBar");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.SquareTubeDimensions", b =>
+ {
+ b.HasBaseType("CutList.Web.Data.Entities.MaterialDimensions");
+
+ b.Property("Size")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Size");
+
+ b.Property("Wall")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Wall");
+
+ b.HasIndex("Size");
+
+ b.HasDiscriminator().HasValue("SquareTube");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.Job", b =>
+ {
+ b.HasOne("CutList.Web.Data.Entities.CuttingTool", "CuttingTool")
+ .WithMany("Jobs")
+ .HasForeignKey("CuttingToolId")
+ .OnDelete(DeleteBehavior.SetNull);
+
+ b.Navigation("CuttingTool");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.JobPart", b =>
+ {
+ b.HasOne("CutList.Web.Data.Entities.Job", "Job")
+ .WithMany("Parts")
+ .HasForeignKey("JobId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.HasOne("CutList.Web.Data.Entities.Material", "Material")
+ .WithMany("JobParts")
+ .HasForeignKey("MaterialId")
+ .OnDelete(DeleteBehavior.Restrict)
+ .IsRequired();
+
+ b.Navigation("Job");
+
+ b.Navigation("Material");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.JobStock", b =>
+ {
+ b.HasOne("CutList.Web.Data.Entities.Job", "Job")
+ .WithMany("Stock")
+ .HasForeignKey("JobId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.HasOne("CutList.Web.Data.Entities.Material", "Material")
+ .WithMany()
+ .HasForeignKey("MaterialId")
+ .OnDelete(DeleteBehavior.Restrict)
+ .IsRequired();
+
+ b.HasOne("CutList.Web.Data.Entities.StockItem", "StockItem")
+ .WithMany()
+ .HasForeignKey("StockItemId")
+ .OnDelete(DeleteBehavior.SetNull);
+
+ b.Navigation("Job");
+
+ b.Navigation("Material");
+
+ b.Navigation("StockItem");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.MaterialDimensions", b =>
+ {
+ b.HasOne("CutList.Web.Data.Entities.Material", "Material")
+ .WithOne("Dimensions")
+ .HasForeignKey("CutList.Web.Data.Entities.MaterialDimensions", "MaterialId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.Navigation("Material");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.StockItem", b =>
+ {
+ b.HasOne("CutList.Web.Data.Entities.Material", "Material")
+ .WithMany("StockItems")
+ .HasForeignKey("MaterialId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.Navigation("Material");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.StockTransaction", b =>
+ {
+ b.HasOne("CutList.Web.Data.Entities.Job", "Job")
+ .WithMany()
+ .HasForeignKey("JobId")
+ .OnDelete(DeleteBehavior.SetNull);
+
+ b.HasOne("CutList.Web.Data.Entities.StockItem", "StockItem")
+ .WithMany("Transactions")
+ .HasForeignKey("StockItemId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.HasOne("CutList.Web.Data.Entities.Supplier", "Supplier")
+ .WithMany()
+ .HasForeignKey("SupplierId")
+ .OnDelete(DeleteBehavior.SetNull);
+
+ b.Navigation("Job");
+
+ b.Navigation("StockItem");
+
+ b.Navigation("Supplier");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.SupplierOffering", b =>
+ {
+ b.HasOne("CutList.Web.Data.Entities.StockItem", "StockItem")
+ .WithMany("SupplierOfferings")
+ .HasForeignKey("StockItemId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.HasOne("CutList.Web.Data.Entities.Supplier", "Supplier")
+ .WithMany("Offerings")
+ .HasForeignKey("SupplierId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.Navigation("StockItem");
+
+ b.Navigation("Supplier");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.CuttingTool", b =>
+ {
+ b.Navigation("Jobs");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.Job", b =>
+ {
+ b.Navigation("Parts");
+
+ b.Navigation("Stock");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.Material", b =>
+ {
+ b.Navigation("Dimensions");
+
+ b.Navigation("JobParts");
+
+ b.Navigation("StockItems");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.StockItem", b =>
+ {
+ b.Navigation("SupplierOfferings");
+
+ b.Navigation("Transactions");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.Supplier", b =>
+ {
+ b.Navigation("Offerings");
+ });
+#pragma warning restore 612, 618
+ }
+ }
+}
diff --git a/CutList.Web/Migrations/20260205214055_FixDimensionDiscriminatorValues.cs b/CutList.Web/Migrations/20260205214055_FixDimensionDiscriminatorValues.cs
new file mode 100644
index 0000000..7b2a439
--- /dev/null
+++ b/CutList.Web/Migrations/20260205214055_FixDimensionDiscriminatorValues.cs
@@ -0,0 +1,47 @@
+using Microsoft.EntityFrameworkCore.Migrations;
+
+#nullable disable
+
+namespace CutList.Web.Migrations
+{
+ ///
+ public partial class FixDimensionDiscriminatorValues : Migration
+ {
+ ///
+ protected override void Up(MigrationBuilder migrationBuilder)
+ {
+ // Fix discriminator values: database has class names (e.g. 'AngleDimensions')
+ // but EF Core config expects short names (e.g. 'Angle')
+ migrationBuilder.Sql(@"
+ UPDATE MaterialDimensions SET DimensionType = 'Angle' WHERE DimensionType = 'AngleDimensions';
+ UPDATE MaterialDimensions SET DimensionType = 'Channel' WHERE DimensionType = 'ChannelDimensions';
+ UPDATE MaterialDimensions SET DimensionType = 'FlatBar' WHERE DimensionType = 'FlatBarDimensions';
+ UPDATE MaterialDimensions SET DimensionType = 'IBeam' WHERE DimensionType = 'IBeamDimensions';
+ UPDATE MaterialDimensions SET DimensionType = 'Pipe' WHERE DimensionType = 'PipeDimensions';
+ UPDATE MaterialDimensions SET DimensionType = 'RectangularTube' WHERE DimensionType = 'RectangularTubeDimensions';
+ UPDATE MaterialDimensions SET DimensionType = 'RoundBar' WHERE DimensionType = 'RoundBarDimensions';
+ UPDATE MaterialDimensions SET DimensionType = 'RoundTube' WHERE DimensionType = 'RoundTubeDimensions';
+ UPDATE MaterialDimensions SET DimensionType = 'SquareBar' WHERE DimensionType = 'SquareBarDimensions';
+ UPDATE MaterialDimensions SET DimensionType = 'SquareTube' WHERE DimensionType = 'SquareTubeDimensions';
+ ");
+ }
+
+ ///
+ protected override void Down(MigrationBuilder migrationBuilder)
+ {
+ // Revert to class name discriminator values
+ migrationBuilder.Sql(@"
+ UPDATE MaterialDimensions SET DimensionType = 'AngleDimensions' WHERE DimensionType = 'Angle';
+ UPDATE MaterialDimensions SET DimensionType = 'ChannelDimensions' WHERE DimensionType = 'Channel';
+ UPDATE MaterialDimensions SET DimensionType = 'FlatBarDimensions' WHERE DimensionType = 'FlatBar';
+ UPDATE MaterialDimensions SET DimensionType = 'IBeamDimensions' WHERE DimensionType = 'IBeam';
+ UPDATE MaterialDimensions SET DimensionType = 'PipeDimensions' WHERE DimensionType = 'Pipe';
+ UPDATE MaterialDimensions SET DimensionType = 'RectangularTubeDimensions' WHERE DimensionType = 'RectangularTube';
+ UPDATE MaterialDimensions SET DimensionType = 'RoundBarDimensions' WHERE DimensionType = 'RoundBar';
+ UPDATE MaterialDimensions SET DimensionType = 'RoundTubeDimensions' WHERE DimensionType = 'RoundTube';
+ UPDATE MaterialDimensions SET DimensionType = 'SquareBarDimensions' WHERE DimensionType = 'SquareBar';
+ UPDATE MaterialDimensions SET DimensionType = 'SquareTubeDimensions' WHERE DimensionType = 'SquareTube';
+ ");
+ }
+ }
+}
diff --git a/CutList.Web/Migrations/20260205214339_FixEmptyMaterialType.Designer.cs b/CutList.Web/Migrations/20260205214339_FixEmptyMaterialType.Designer.cs
new file mode 100644
index 0000000..6bbb546
--- /dev/null
+++ b/CutList.Web/Migrations/20260205214339_FixEmptyMaterialType.Designer.cs
@@ -0,0 +1,823 @@
+//
+using System;
+using CutList.Web.Data;
+using Microsoft.EntityFrameworkCore;
+using Microsoft.EntityFrameworkCore.Infrastructure;
+using Microsoft.EntityFrameworkCore.Metadata;
+using Microsoft.EntityFrameworkCore.Migrations;
+using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
+
+#nullable disable
+
+namespace CutList.Web.Migrations
+{
+ [DbContext(typeof(ApplicationDbContext))]
+ [Migration("20260205214339_FixEmptyMaterialType")]
+ partial class FixEmptyMaterialType
+ {
+ ///
+ protected override void BuildTargetModel(ModelBuilder modelBuilder)
+ {
+#pragma warning disable 612, 618
+ modelBuilder
+ .HasAnnotation("ProductVersion", "8.0.11")
+ .HasAnnotation("Relational:MaxIdentifierLength", 128);
+
+ SqlServerModelBuilderExtensions.UseIdentityColumns(modelBuilder);
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.CuttingTool", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("IsActive")
+ .HasColumnType("bit");
+
+ b.Property("IsDefault")
+ .HasColumnType("bit");
+
+ b.Property("KerfInches")
+ .HasPrecision(6, 4)
+ .HasColumnType("decimal(6,4)");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasMaxLength(50)
+ .HasColumnType("nvarchar(50)");
+
+ b.HasKey("Id");
+
+ b.ToTable("CuttingTools");
+
+ b.HasData(
+ new
+ {
+ Id = 1,
+ IsActive = true,
+ IsDefault = true,
+ KerfInches = 0.0625m,
+ Name = "Bandsaw"
+ },
+ new
+ {
+ Id = 2,
+ IsActive = true,
+ IsDefault = false,
+ KerfInches = 0.125m,
+ Name = "Chop Saw"
+ },
+ new
+ {
+ Id = 3,
+ IsActive = true,
+ IsDefault = false,
+ KerfInches = 0.0625m,
+ Name = "Cold Cut Saw"
+ },
+ new
+ {
+ Id = 4,
+ IsActive = true,
+ IsDefault = false,
+ KerfInches = 0.0625m,
+ Name = "Hacksaw"
+ });
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.Job", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("CreatedAt")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("datetime2")
+ .HasDefaultValueSql("GETUTCDATE()");
+
+ b.Property("Customer")
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("CuttingToolId")
+ .HasColumnType("int");
+
+ b.Property("JobNumber")
+ .IsRequired()
+ .HasMaxLength(20)
+ .HasColumnType("nvarchar(20)");
+
+ b.Property("Name")
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("Notes")
+ .HasColumnType("nvarchar(max)");
+
+ b.Property("UpdatedAt")
+ .HasColumnType("datetime2");
+
+ b.HasKey("Id");
+
+ b.HasIndex("CuttingToolId");
+
+ b.HasIndex("JobNumber")
+ .IsUnique();
+
+ b.ToTable("Jobs");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.JobPart", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("JobId")
+ .HasColumnType("int");
+
+ b.Property("LengthInches")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.Property("MaterialId")
+ .HasColumnType("int");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("Quantity")
+ .HasColumnType("int");
+
+ b.Property("SortOrder")
+ .HasColumnType("int");
+
+ b.HasKey("Id");
+
+ b.HasIndex("JobId");
+
+ b.HasIndex("MaterialId");
+
+ b.ToTable("JobParts");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.JobStock", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("IsCustomLength")
+ .HasColumnType("bit");
+
+ b.Property("JobId")
+ .HasColumnType("int");
+
+ b.Property("LengthInches")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.Property("MaterialId")
+ .HasColumnType("int");
+
+ b.Property("Priority")
+ .HasColumnType("int");
+
+ b.Property("Quantity")
+ .HasColumnType("int");
+
+ b.Property("SortOrder")
+ .HasColumnType("int");
+
+ b.Property("StockItemId")
+ .HasColumnType("int");
+
+ b.HasKey("Id");
+
+ b.HasIndex("JobId");
+
+ b.HasIndex("MaterialId");
+
+ b.HasIndex("StockItemId");
+
+ b.ToTable("JobStocks");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.Material", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("CreatedAt")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("datetime2")
+ .HasDefaultValueSql("GETUTCDATE()");
+
+ b.Property("Description")
+ .HasMaxLength(255)
+ .HasColumnType("nvarchar(255)");
+
+ b.Property("Grade")
+ .HasMaxLength(50)
+ .HasColumnType("nvarchar(50)");
+
+ b.Property("IsActive")
+ .HasColumnType("bit");
+
+ b.Property("Shape")
+ .IsRequired()
+ .HasMaxLength(50)
+ .HasColumnType("nvarchar(50)");
+
+ b.Property("Size")
+ .IsRequired()
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("SortOrder")
+ .HasColumnType("int");
+
+ b.Property("Type")
+ .IsRequired()
+ .HasMaxLength(20)
+ .HasColumnType("nvarchar(20)");
+
+ b.Property("UpdatedAt")
+ .HasColumnType("datetime2");
+
+ b.HasKey("Id");
+
+ b.ToTable("Materials");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.MaterialDimensions", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("DimensionType")
+ .IsRequired()
+ .HasMaxLength(21)
+ .HasColumnType("nvarchar(21)");
+
+ b.Property("MaterialId")
+ .HasColumnType("int");
+
+ b.HasKey("Id");
+
+ b.HasIndex("MaterialId")
+ .IsUnique();
+
+ b.ToTable("MaterialDimensions");
+
+ b.HasDiscriminator("DimensionType").HasValue("MaterialDimensions");
+
+ b.UseTphMappingStrategy();
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.StockItem", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("CreatedAt")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("datetime2")
+ .HasDefaultValueSql("GETUTCDATE()");
+
+ b.Property("IsActive")
+ .HasColumnType("bit");
+
+ b.Property("LengthInches")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.Property("MaterialId")
+ .HasColumnType("int");
+
+ b.Property("Name")
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("Notes")
+ .HasMaxLength(255)
+ .HasColumnType("nvarchar(255)");
+
+ b.Property("QuantityOnHand")
+ .HasColumnType("int");
+
+ b.Property("UpdatedAt")
+ .HasColumnType("datetime2");
+
+ b.HasKey("Id");
+
+ b.HasIndex("MaterialId", "LengthInches")
+ .IsUnique();
+
+ b.ToTable("StockItems");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.StockTransaction", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("CreatedAt")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("datetime2")
+ .HasDefaultValueSql("GETUTCDATE()");
+
+ b.Property("JobId")
+ .HasColumnType("int");
+
+ b.Property("Notes")
+ .HasMaxLength(500)
+ .HasColumnType("nvarchar(500)");
+
+ b.Property("Quantity")
+ .HasColumnType("int");
+
+ b.Property("StockItemId")
+ .HasColumnType("int");
+
+ b.Property("SupplierId")
+ .HasColumnType("int");
+
+ b.Property("Type")
+ .HasColumnType("int");
+
+ b.Property("UnitPrice")
+ .HasPrecision(10, 2)
+ .HasColumnType("decimal(10,2)");
+
+ b.HasKey("Id");
+
+ b.HasIndex("JobId");
+
+ b.HasIndex("StockItemId");
+
+ b.HasIndex("SupplierId");
+
+ b.ToTable("StockTransactions");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.Supplier", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("ContactInfo")
+ .HasMaxLength(500)
+ .HasColumnType("nvarchar(500)");
+
+ b.Property("CreatedAt")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("datetime2")
+ .HasDefaultValueSql("GETUTCDATE()");
+
+ b.Property("IsActive")
+ .HasColumnType("bit");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("Notes")
+ .HasColumnType("nvarchar(max)");
+
+ b.HasKey("Id");
+
+ b.ToTable("Suppliers");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.SupplierOffering", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("IsActive")
+ .HasColumnType("bit");
+
+ b.Property("Notes")
+ .HasMaxLength(255)
+ .HasColumnType("nvarchar(255)");
+
+ b.Property("PartNumber")
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("Price")
+ .HasPrecision(10, 2)
+ .HasColumnType("decimal(10,2)");
+
+ b.Property("StockItemId")
+ .HasColumnType("int");
+
+ b.Property("SupplierDescription")
+ .HasMaxLength(255)
+ .HasColumnType("nvarchar(255)");
+
+ b.Property("SupplierId")
+ .HasColumnType("int");
+
+ b.HasKey("Id");
+
+ b.HasIndex("StockItemId");
+
+ b.HasIndex("SupplierId", "StockItemId")
+ .IsUnique();
+
+ b.ToTable("SupplierOfferings");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.AngleDimensions", b =>
+ {
+ b.HasBaseType("CutList.Web.Data.Entities.MaterialDimensions");
+
+ b.Property("Leg1")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.Property("Leg2")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.Property("Thickness")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Thickness");
+
+ b.HasIndex("Leg1");
+
+ b.HasDiscriminator().HasValue("Angle");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.ChannelDimensions", b =>
+ {
+ b.HasBaseType("CutList.Web.Data.Entities.MaterialDimensions");
+
+ b.Property("Flange")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.Property("Height")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Height");
+
+ b.Property("Web")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.HasIndex("Height");
+
+ b.HasDiscriminator().HasValue("Channel");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.FlatBarDimensions", b =>
+ {
+ b.HasBaseType("CutList.Web.Data.Entities.MaterialDimensions");
+
+ b.Property("Thickness")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Thickness");
+
+ b.Property("Width")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Width");
+
+ b.HasIndex("Width");
+
+ b.HasDiscriminator().HasValue("FlatBar");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.IBeamDimensions", b =>
+ {
+ b.HasBaseType("CutList.Web.Data.Entities.MaterialDimensions");
+
+ b.Property("Height")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Height");
+
+ b.Property("WeightPerFoot")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.HasIndex("Height");
+
+ b.HasDiscriminator().HasValue("IBeam");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.PipeDimensions", b =>
+ {
+ b.HasBaseType("CutList.Web.Data.Entities.MaterialDimensions");
+
+ b.Property("NominalSize")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.Property("Schedule")
+ .HasMaxLength(20)
+ .HasColumnType("nvarchar(20)");
+
+ b.Property("Wall")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Wall");
+
+ b.HasIndex("NominalSize");
+
+ b.HasDiscriminator().HasValue("Pipe");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.RectangularTubeDimensions", b =>
+ {
+ b.HasBaseType("CutList.Web.Data.Entities.MaterialDimensions");
+
+ b.Property("Height")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Height");
+
+ b.Property("Wall")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Wall");
+
+ b.Property("Width")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Width");
+
+ b.HasIndex("Width");
+
+ b.HasDiscriminator().HasValue("RectangularTube");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.RoundBarDimensions", b =>
+ {
+ b.HasBaseType("CutList.Web.Data.Entities.MaterialDimensions");
+
+ b.Property("Diameter")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.HasIndex("Diameter");
+
+ b.HasDiscriminator().HasValue("RoundBar");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.RoundTubeDimensions", b =>
+ {
+ b.HasBaseType("CutList.Web.Data.Entities.MaterialDimensions");
+
+ b.Property("OuterDiameter")
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)");
+
+ b.Property("Wall")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Wall");
+
+ b.HasIndex("OuterDiameter");
+
+ b.HasDiscriminator().HasValue("RoundTube");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.SquareBarDimensions", b =>
+ {
+ b.HasBaseType("CutList.Web.Data.Entities.MaterialDimensions");
+
+ b.Property("Size")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Size");
+
+ b.HasIndex("Size");
+
+ b.HasDiscriminator().HasValue("SquareBar");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.SquareTubeDimensions", b =>
+ {
+ b.HasBaseType("CutList.Web.Data.Entities.MaterialDimensions");
+
+ b.Property("Size")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Size");
+
+ b.Property("Wall")
+ .ValueGeneratedOnUpdateSometimes()
+ .HasPrecision(10, 4)
+ .HasColumnType("decimal(10,4)")
+ .HasColumnName("Wall");
+
+ b.HasIndex("Size");
+
+ b.HasDiscriminator().HasValue("SquareTube");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.Job", b =>
+ {
+ b.HasOne("CutList.Web.Data.Entities.CuttingTool", "CuttingTool")
+ .WithMany("Jobs")
+ .HasForeignKey("CuttingToolId")
+ .OnDelete(DeleteBehavior.SetNull);
+
+ b.Navigation("CuttingTool");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.JobPart", b =>
+ {
+ b.HasOne("CutList.Web.Data.Entities.Job", "Job")
+ .WithMany("Parts")
+ .HasForeignKey("JobId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.HasOne("CutList.Web.Data.Entities.Material", "Material")
+ .WithMany("JobParts")
+ .HasForeignKey("MaterialId")
+ .OnDelete(DeleteBehavior.Restrict)
+ .IsRequired();
+
+ b.Navigation("Job");
+
+ b.Navigation("Material");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.JobStock", b =>
+ {
+ b.HasOne("CutList.Web.Data.Entities.Job", "Job")
+ .WithMany("Stock")
+ .HasForeignKey("JobId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.HasOne("CutList.Web.Data.Entities.Material", "Material")
+ .WithMany()
+ .HasForeignKey("MaterialId")
+ .OnDelete(DeleteBehavior.Restrict)
+ .IsRequired();
+
+ b.HasOne("CutList.Web.Data.Entities.StockItem", "StockItem")
+ .WithMany()
+ .HasForeignKey("StockItemId")
+ .OnDelete(DeleteBehavior.SetNull);
+
+ b.Navigation("Job");
+
+ b.Navigation("Material");
+
+ b.Navigation("StockItem");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.MaterialDimensions", b =>
+ {
+ b.HasOne("CutList.Web.Data.Entities.Material", "Material")
+ .WithOne("Dimensions")
+ .HasForeignKey("CutList.Web.Data.Entities.MaterialDimensions", "MaterialId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.Navigation("Material");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.StockItem", b =>
+ {
+ b.HasOne("CutList.Web.Data.Entities.Material", "Material")
+ .WithMany("StockItems")
+ .HasForeignKey("MaterialId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.Navigation("Material");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.StockTransaction", b =>
+ {
+ b.HasOne("CutList.Web.Data.Entities.Job", "Job")
+ .WithMany()
+ .HasForeignKey("JobId")
+ .OnDelete(DeleteBehavior.SetNull);
+
+ b.HasOne("CutList.Web.Data.Entities.StockItem", "StockItem")
+ .WithMany("Transactions")
+ .HasForeignKey("StockItemId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.HasOne("CutList.Web.Data.Entities.Supplier", "Supplier")
+ .WithMany()
+ .HasForeignKey("SupplierId")
+ .OnDelete(DeleteBehavior.SetNull);
+
+ b.Navigation("Job");
+
+ b.Navigation("StockItem");
+
+ b.Navigation("Supplier");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.SupplierOffering", b =>
+ {
+ b.HasOne("CutList.Web.Data.Entities.StockItem", "StockItem")
+ .WithMany("SupplierOfferings")
+ .HasForeignKey("StockItemId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.HasOne("CutList.Web.Data.Entities.Supplier", "Supplier")
+ .WithMany("Offerings")
+ .HasForeignKey("SupplierId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.Navigation("StockItem");
+
+ b.Navigation("Supplier");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.CuttingTool", b =>
+ {
+ b.Navigation("Jobs");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.Job", b =>
+ {
+ b.Navigation("Parts");
+
+ b.Navigation("Stock");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.Material", b =>
+ {
+ b.Navigation("Dimensions");
+
+ b.Navigation("JobParts");
+
+ b.Navigation("StockItems");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.StockItem", b =>
+ {
+ b.Navigation("SupplierOfferings");
+
+ b.Navigation("Transactions");
+ });
+
+ modelBuilder.Entity("CutList.Web.Data.Entities.Supplier", b =>
+ {
+ b.Navigation("Offerings");
+ });
+#pragma warning restore 612, 618
+ }
+ }
+}
diff --git a/CutList.Web/Migrations/20260205214339_FixEmptyMaterialType.cs b/CutList.Web/Migrations/20260205214339_FixEmptyMaterialType.cs
new file mode 100644
index 0000000..3d2f240
--- /dev/null
+++ b/CutList.Web/Migrations/20260205214339_FixEmptyMaterialType.cs
@@ -0,0 +1,38 @@
+using Microsoft.EntityFrameworkCore.Migrations;
+
+#nullable disable
+
+namespace CutList.Web.Migrations
+{
+ ///
+ public partial class FixEmptyMaterialType : Migration
+ {
+ ///
+ protected override void Up(MigrationBuilder migrationBuilder)
+ {
+ // Fix empty Type values from when the column was added with defaultValue: ""
+ migrationBuilder.Sql("UPDATE Materials SET Type = 'Steel' WHERE Type = '' OR Type IS NULL;");
+
+ // Change the default so new rows also get 'Steel'
+ migrationBuilder.AlterColumn(
+ name: "Type",
+ table: "Materials",
+ type: "nvarchar(20)",
+ maxLength: 20,
+ nullable: false,
+ defaultValue: "Steel");
+ }
+
+ ///
+ protected override void Down(MigrationBuilder migrationBuilder)
+ {
+ migrationBuilder.AlterColumn(
+ name: "Type",
+ table: "Materials",
+ type: "nvarchar(20)",
+ maxLength: 20,
+ nullable: false,
+ defaultValue: "");
+ }
+ }
+}