diff --git a/CutList.Web/Data/Entities/MaterialDimensions.cs b/CutList.Web/Data/Entities/MaterialDimensions.cs
new file mode 100644
index 0000000..5d42b1d
--- /dev/null
+++ b/CutList.Web/Data/Entities/MaterialDimensions.cs
@@ -0,0 +1,192 @@
+namespace CutList.Web.Data.Entities;
+
+///
+/// Base class for material dimensions. Each shape has its own derived class with specific properties.
+///
+public abstract class MaterialDimensions
+{
+ public int Id { get; set; }
+ public int MaterialId { get; set; }
+ public Material Material { get; set; } = null!;
+
+ ///
+ /// Generates a display string for the size based on the dimensions.
+ ///
+ public abstract string GenerateSizeString();
+
+ ///
+ /// Gets the primary dimension value for sorting (in thousandths of an inch).
+ ///
+ public abstract int GetSortOrder();
+}
+
+///
+/// Dimensions for Round Bar: solid round stock.
+///
+public class RoundBarDimensions : MaterialDimensions
+{
+ public decimal Diameter { get; set; }
+
+ public override string GenerateSizeString() =>
+ FormatDimension(Diameter);
+
+ public override int GetSortOrder() => (int)(Diameter * 1000);
+
+ private static string FormatDimension(decimal value) =>
+ CutList.Core.Formatting.ArchUnits.FormatFromInches((double)value);
+}
+
+///
+/// Dimensions for Round Tube: hollow round stock.
+///
+public class RoundTubeDimensions : MaterialDimensions
+{
+ public decimal OuterDiameter { get; set; }
+ public decimal Wall { get; set; }
+
+ public override string GenerateSizeString() =>
+ $"{FormatDimension(OuterDiameter)} OD x {FormatDimension(Wall)} wall";
+
+ public override int GetSortOrder() => (int)(OuterDiameter * 1000);
+
+ private static string FormatDimension(decimal value) =>
+ CutList.Core.Formatting.ArchUnits.FormatFromInches((double)value);
+}
+
+///
+/// Dimensions for Flat Bar: rectangular solid stock.
+///
+public class FlatBarDimensions : MaterialDimensions
+{
+ public decimal Width { get; set; }
+ public decimal Thickness { get; set; }
+
+ public override string GenerateSizeString() =>
+ $"{FormatDimension(Width)} x {FormatDimension(Thickness)}";
+
+ public override int GetSortOrder() => (int)(Width * 1000);
+
+ private static string FormatDimension(decimal value) =>
+ CutList.Core.Formatting.ArchUnits.FormatFromInches((double)value);
+}
+
+///
+/// Dimensions for Square Bar: solid square stock.
+///
+public class SquareBarDimensions : MaterialDimensions
+{
+ public decimal Size { get; set; }
+
+ public override string GenerateSizeString() =>
+ FormatDimension(Size);
+
+ public override int GetSortOrder() => (int)(Size * 1000);
+
+ private static string FormatDimension(decimal value) =>
+ CutList.Core.Formatting.ArchUnits.FormatFromInches((double)value);
+}
+
+///
+/// Dimensions for Square Tube: hollow square stock.
+///
+public class SquareTubeDimensions : MaterialDimensions
+{
+ public decimal Size { get; set; }
+ public decimal Wall { get; set; }
+
+ public override string GenerateSizeString() =>
+ $"{FormatDimension(Size)} x {FormatDimension(Wall)} wall";
+
+ public override int GetSortOrder() => (int)(Size * 1000);
+
+ private static string FormatDimension(decimal value) =>
+ CutList.Core.Formatting.ArchUnits.FormatFromInches((double)value);
+}
+
+///
+/// Dimensions for Rectangular Tube: hollow rectangular stock.
+///
+public class RectangularTubeDimensions : MaterialDimensions
+{
+ public decimal Width { get; set; }
+ public decimal Height { get; set; }
+ public decimal Wall { get; set; }
+
+ public override string GenerateSizeString() =>
+ $"{FormatDimension(Width)} x {FormatDimension(Height)} x {FormatDimension(Wall)} wall";
+
+ public override int GetSortOrder() => (int)(Width * 1000);
+
+ private static string FormatDimension(decimal value) =>
+ CutList.Core.Formatting.ArchUnits.FormatFromInches((double)value);
+}
+
+///
+/// Dimensions for Angle: L-shaped stock.
+///
+public class AngleDimensions : MaterialDimensions
+{
+ public decimal Leg1 { get; set; }
+ public decimal Leg2 { get; set; }
+ public decimal Thickness { get; set; }
+
+ public override string GenerateSizeString() =>
+ $"{FormatDimension(Leg1)} x {FormatDimension(Leg2)} x {FormatDimension(Thickness)}";
+
+ public override int GetSortOrder() => (int)(Leg1 * 1000);
+
+ private static string FormatDimension(decimal value) =>
+ CutList.Core.Formatting.ArchUnits.FormatFromInches((double)value);
+}
+
+///
+/// Dimensions for Channel: C-shaped stock.
+///
+public class ChannelDimensions : MaterialDimensions
+{
+ public decimal Height { get; set; }
+ public decimal Flange { get; set; }
+ public decimal Web { get; set; }
+
+ public override string GenerateSizeString() =>
+ $"{FormatDimension(Height)} x {FormatDimension(Flange)} x {FormatDimension(Web)}";
+
+ public override int GetSortOrder() => (int)(Height * 1000);
+
+ private static string FormatDimension(decimal value) =>
+ CutList.Core.Formatting.ArchUnits.FormatFromInches((double)value);
+}
+
+///
+/// Dimensions for I-Beam: wide flange beam.
+///
+public class IBeamDimensions : MaterialDimensions
+{
+ public decimal Height { get; set; }
+ public decimal WeightPerFoot { get; set; }
+
+ public override string GenerateSizeString() =>
+ $"W{Height:0.##} x {WeightPerFoot:0.##}";
+
+ public override int GetSortOrder() => (int)(Height * 1000);
+}
+
+///
+/// Dimensions for Pipe: nominal pipe size.
+///
+public class PipeDimensions : MaterialDimensions
+{
+ public decimal NominalSize { get; set; }
+ public decimal? Wall { get; set; }
+ public string? Schedule { get; set; }
+
+ public override string GenerateSizeString() =>
+ !string.IsNullOrEmpty(Schedule)
+ ? $"{FormatDimension(NominalSize)} NPS Sch {Schedule}"
+ : $"{FormatDimension(NominalSize)} NPS x {FormatDimension(Wall ?? 0)} wall";
+
+ public override int GetSortOrder() => (int)(NominalSize * 1000);
+
+ private static string FormatDimension(decimal value) =>
+ CutList.Core.Formatting.ArchUnits.FormatFromInches((double)value);
+}
diff --git a/CutList.Web/Data/Entities/MaterialShape.cs b/CutList.Web/Data/Entities/MaterialShape.cs
new file mode 100644
index 0000000..6ce254d
--- /dev/null
+++ b/CutList.Web/Data/Entities/MaterialShape.cs
@@ -0,0 +1,89 @@
+namespace CutList.Web.Data.Entities;
+
+///
+/// Enumeration of supported material shapes.
+///
+public enum MaterialShape
+{
+ RoundBar,
+ RoundTube,
+ FlatBar,
+ SquareBar,
+ SquareTube,
+ RectangularTube,
+ Angle,
+ Channel,
+ IBeam,
+ Pipe
+}
+
+///
+/// Extension methods for MaterialShape enum.
+///
+public static class MaterialShapeExtensions
+{
+ ///
+ /// Gets the display name for a material shape.
+ ///
+ public static string GetDisplayName(this MaterialShape shape) => shape switch
+ {
+ MaterialShape.RoundBar => "Round Bar",
+ MaterialShape.RoundTube => "Round Tube",
+ MaterialShape.FlatBar => "Flat Bar",
+ MaterialShape.SquareBar => "Square Bar",
+ MaterialShape.SquareTube => "Square Tube",
+ MaterialShape.RectangularTube => "Rectangular Tube",
+ MaterialShape.Angle => "Angle",
+ MaterialShape.Channel => "Channel",
+ MaterialShape.IBeam => "I-Beam",
+ MaterialShape.Pipe => "Pipe",
+ _ => shape.ToString()
+ };
+
+ ///
+ /// Parses a display name or enum value string to a MaterialShape.
+ ///
+ public static MaterialShape? ParseShape(string? input)
+ {
+ if (string.IsNullOrWhiteSpace(input))
+ return null;
+
+ // Try exact enum parse first
+ if (Enum.TryParse(input, ignoreCase: true, out var result))
+ return result;
+
+ // Try display name matching
+ return input.Trim().ToLowerInvariant() switch
+ {
+ "round bar" => MaterialShape.RoundBar,
+ "round tube" => MaterialShape.RoundTube,
+ "flat bar" => MaterialShape.FlatBar,
+ "square bar" => MaterialShape.SquareBar,
+ "square tube" => MaterialShape.SquareTube,
+ "rectangular tube" or "rect tube" => MaterialShape.RectangularTube,
+ "angle" => MaterialShape.Angle,
+ "channel" => MaterialShape.Channel,
+ "i-beam" or "ibeam" or "i beam" => MaterialShape.IBeam,
+ "pipe" => MaterialShape.Pipe,
+ _ => null
+ };
+ }
+
+ ///
+ /// Gets the dimension field names used by a given shape.
+ ///
+ public static string[] GetDimensionFields(this MaterialShape shape) => shape switch
+ {
+ MaterialShape.RoundBar => new[] { "Diameter" },
+ MaterialShape.RoundTube => new[] { "OuterDiameter", "Wall" },
+ MaterialShape.FlatBar => new[] { "Width", "Thickness" },
+ MaterialShape.SquareBar => new[] { "Size" },
+ MaterialShape.SquareTube => new[] { "Size", "Wall" },
+ MaterialShape.RectangularTube => new[] { "Width", "Height", "Wall" },
+ MaterialShape.Angle => new[] { "Leg1", "Leg2", "Thickness" },
+ MaterialShape.Channel => new[] { "Height", "Flange", "Web" },
+ MaterialShape.IBeam => new[] { "Height", "WeightPerFoot" },
+ MaterialShape.Pipe => new[] { "NominalSize", "Wall", "Schedule" },
+ _ => Array.Empty()
+ };
+}
diff --git a/CutList.Web/Data/Entities/MaterialType.cs b/CutList.Web/Data/Entities/MaterialType.cs
new file mode 100644
index 0000000..c0988b7
--- /dev/null
+++ b/CutList.Web/Data/Entities/MaterialType.cs
@@ -0,0 +1,13 @@
+namespace CutList.Web.Data.Entities;
+
+///
+/// Type of material (metal).
+///
+public enum MaterialType
+{
+ Steel,
+ Aluminum,
+ Stainless,
+ Brass,
+ Copper
+}
diff --git a/CutList.Web/Migrations/20260205012737_AddMaterialDimensions.Designer.cs b/CutList.Web/Migrations/20260205012737_AddMaterialDimensions.Designer.cs
new file mode 100644
index 0000000..8df612e
--- /dev/null
+++ b/CutList.Web/Migrations/20260205012737_AddMaterialDimensions.Designer.cs
@@ -0,0 +1,811 @@
+//
+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("20260205012737_AddMaterialDimensions")]
+ partial class AddMaterialDimensions
+ {
+ ///
+ 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("IsActive")
+ .HasColumnType("bit");
+
+ b.Property("Shape")
+ .IsRequired()
+ .HasMaxLength(50)
+ .HasColumnType("nvarchar(50)");
+
+ b.Property("Size")
+ .IsRequired()
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ 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/20260205012737_AddMaterialDimensions.cs b/CutList.Web/Migrations/20260205012737_AddMaterialDimensions.cs
new file mode 100644
index 0000000..162dd0b
--- /dev/null
+++ b/CutList.Web/Migrations/20260205012737_AddMaterialDimensions.cs
@@ -0,0 +1,96 @@
+using Microsoft.EntityFrameworkCore.Migrations;
+
+#nullable disable
+
+namespace CutList.Web.Migrations
+{
+ ///
+ public partial class AddMaterialDimensions : Migration
+ {
+ ///
+ protected override void Up(MigrationBuilder migrationBuilder)
+ {
+ migrationBuilder.CreateTable(
+ name: "MaterialDimensions",
+ columns: table => new
+ {
+ Id = table.Column(type: "int", nullable: false)
+ .Annotation("SqlServer:Identity", "1, 1"),
+ MaterialId = table.Column(type: "int", nullable: false),
+ DimensionType = table.Column(type: "nvarchar(21)", maxLength: 21, nullable: false),
+ Leg1 = table.Column(type: "decimal(10,4)", precision: 10, scale: 4, nullable: true),
+ Leg2 = table.Column(type: "decimal(10,4)", precision: 10, scale: 4, nullable: true),
+ Thickness = table.Column(type: "decimal(10,4)", precision: 10, scale: 4, nullable: true),
+ Height = table.Column(type: "decimal(10,4)", precision: 10, scale: 4, nullable: true),
+ Flange = table.Column(type: "decimal(10,4)", precision: 10, scale: 4, nullable: true),
+ Web = table.Column(type: "decimal(10,4)", precision: 10, scale: 4, nullable: true),
+ Width = table.Column(type: "decimal(10,4)", precision: 10, scale: 4, nullable: true),
+ WeightPerFoot = table.Column(type: "decimal(10,4)", precision: 10, scale: 4, nullable: true),
+ NominalSize = table.Column(type: "decimal(10,4)", precision: 10, scale: 4, nullable: true),
+ Wall = table.Column(type: "decimal(10,4)", precision: 10, scale: 4, nullable: true),
+ Schedule = table.Column(type: "nvarchar(20)", maxLength: 20, nullable: true),
+ Diameter = table.Column(type: "decimal(10,4)", precision: 10, scale: 4, nullable: true),
+ OuterDiameter = table.Column(type: "decimal(10,4)", precision: 10, scale: 4, nullable: true),
+ Size = table.Column(type: "decimal(10,4)", precision: 10, scale: 4, nullable: true)
+ },
+ constraints: table =>
+ {
+ table.PrimaryKey("PK_MaterialDimensions", x => x.Id);
+ table.ForeignKey(
+ name: "FK_MaterialDimensions_Materials_MaterialId",
+ column: x => x.MaterialId,
+ principalTable: "Materials",
+ principalColumn: "Id",
+ onDelete: ReferentialAction.Cascade);
+ });
+
+ migrationBuilder.CreateIndex(
+ name: "IX_MaterialDimensions_Diameter",
+ table: "MaterialDimensions",
+ column: "Diameter");
+
+ migrationBuilder.CreateIndex(
+ name: "IX_MaterialDimensions_Height",
+ table: "MaterialDimensions",
+ column: "Height");
+
+ migrationBuilder.CreateIndex(
+ name: "IX_MaterialDimensions_Leg1",
+ table: "MaterialDimensions",
+ column: "Leg1");
+
+ migrationBuilder.CreateIndex(
+ name: "IX_MaterialDimensions_MaterialId",
+ table: "MaterialDimensions",
+ column: "MaterialId",
+ unique: true);
+
+ migrationBuilder.CreateIndex(
+ name: "IX_MaterialDimensions_NominalSize",
+ table: "MaterialDimensions",
+ column: "NominalSize");
+
+ migrationBuilder.CreateIndex(
+ name: "IX_MaterialDimensions_OuterDiameter",
+ table: "MaterialDimensions",
+ column: "OuterDiameter");
+
+ migrationBuilder.CreateIndex(
+ name: "IX_MaterialDimensions_Size",
+ table: "MaterialDimensions",
+ column: "Size");
+
+ migrationBuilder.CreateIndex(
+ name: "IX_MaterialDimensions_Width",
+ table: "MaterialDimensions",
+ column: "Width");
+ }
+
+ ///
+ protected override void Down(MigrationBuilder migrationBuilder)
+ {
+ migrationBuilder.DropTable(
+ name: "MaterialDimensions");
+ }
+ }
+}
diff --git a/CutList.Web/Migrations/20260205014058_FixMaterialShapeEnumValues.Designer.cs b/CutList.Web/Migrations/20260205014058_FixMaterialShapeEnumValues.Designer.cs
new file mode 100644
index 0000000..1d61ee1
--- /dev/null
+++ b/CutList.Web/Migrations/20260205014058_FixMaterialShapeEnumValues.Designer.cs
@@ -0,0 +1,811 @@
+//
+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("20260205014058_FixMaterialShapeEnumValues")]
+ partial class FixMaterialShapeEnumValues
+ {
+ ///
+ 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("IsActive")
+ .HasColumnType("bit");
+
+ b.Property("Shape")
+ .IsRequired()
+ .HasMaxLength(50)
+ .HasColumnType("nvarchar(50)");
+
+ b.Property("Size")
+ .IsRequired()
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ 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/20260205014058_FixMaterialShapeEnumValues.cs b/CutList.Web/Migrations/20260205014058_FixMaterialShapeEnumValues.cs
new file mode 100644
index 0000000..a3bfcd6
--- /dev/null
+++ b/CutList.Web/Migrations/20260205014058_FixMaterialShapeEnumValues.cs
@@ -0,0 +1,31 @@
+using Microsoft.EntityFrameworkCore.Migrations;
+
+#nullable disable
+
+namespace CutList.Web.Migrations
+{
+ ///
+ public partial class FixMaterialShapeEnumValues : Migration
+ {
+ ///
+ protected override void Up(MigrationBuilder migrationBuilder)
+ {
+ // Convert display names to enum names
+ migrationBuilder.Sql(@"
+ UPDATE Materials SET Shape = 'RoundBar' WHERE Shape = 'Round Bar';
+ UPDATE Materials SET Shape = 'RoundTube' WHERE Shape = 'Round Tube';
+ UPDATE Materials SET Shape = 'FlatBar' WHERE Shape = 'Flat Bar';
+ UPDATE Materials SET Shape = 'SquareBar' WHERE Shape = 'Square Bar';
+ UPDATE Materials SET Shape = 'SquareTube' WHERE Shape = 'Square Tube';
+ UPDATE Materials SET Shape = 'RectangularTube' WHERE Shape = 'Rectangular Tube';
+ UPDATE Materials SET Shape = 'IBeam' WHERE Shape = 'I-Beam';
+ ");
+ }
+
+ ///
+ protected override void Down(MigrationBuilder migrationBuilder)
+ {
+
+ }
+ }
+}
diff --git a/CutList.Web/Migrations/20260205025654_AddMaterialSortOrder.Designer.cs b/CutList.Web/Migrations/20260205025654_AddMaterialSortOrder.Designer.cs
new file mode 100644
index 0000000..c201d4f
--- /dev/null
+++ b/CutList.Web/Migrations/20260205025654_AddMaterialSortOrder.Designer.cs
@@ -0,0 +1,814 @@
+//
+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("20260205025654_AddMaterialSortOrder")]
+ partial class AddMaterialSortOrder
+ {
+ ///
+ 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