## Product Variations System - Add ProductVariation model with quantity-based pricing (1 for £10, 2 for £19, 3 for £25) - Complete CRUD operations for product variations - Enhanced ProductService to include variations in all queries - Updated OrderItem to support ProductVariationId for variation-based orders - Graceful error handling for duplicate quantity constraints - Admin interface with variations management (Create/Edit/Delete) - API endpoints for programmatic variation management ## Enhanced Order Workflow Management - Redesigned OrderStatus enum with clear workflow states (Accept → Packing → Dispatched → Delivered) - Added workflow tracking fields (AcceptedAt, PackingStartedAt, DispatchedAt, ExpectedDeliveryDate) - User tracking for accountability (AcceptedByUser, PackedByUser, DispatchedByUser) - Automatic delivery date calculation (dispatch date + working days, skips weekends) - On Hold workflow for problem resolution with reason tracking - Tab-based orders interface focused on workflow stages - One-click workflow actions from list view ## Mobile-Responsive Design - Responsive orders interface: tables on desktop, cards on mobile - Touch-friendly buttons and spacing for mobile users - Horizontal scrolling tabs with condensed labels on mobile - Color-coded status borders for quick visual recognition - Smart text switching based on screen size ## Product Import/Export System - CSV import with product variations support - Template download with examples - Export existing products to CSV - Detailed import results with success/error reporting - Category name resolution (no need for GUIDs) - Photo URLs import support ## Enhanced Dashboard - Product variations count and metrics - Stock alerts (low stock/out of stock warnings) - Order workflow breakdown (pending, accepted, dispatched counts) - Enhanced layout with more detailed information ## Technical Improvements - Fixed form binding issues across all admin forms - Removed external CDN dependencies for isolated deployment - Bot Wizard form with auto-personality assignment - Proper authentication scheme configuration (Cookie + JWT) - Enhanced debug logging for troubleshooting ## Self-Contained Deployment - All external CDN references replaced with local libraries - Ready for air-gapped/isolated network deployment - No external internet dependencies 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
149 lines
4.0 KiB
C#
149 lines
4.0 KiB
C#
using System.ComponentModel.DataAnnotations;
|
|
using LittleShop.Enums;
|
|
|
|
namespace LittleShop.DTOs;
|
|
|
|
public class ProductDto
|
|
{
|
|
public Guid Id { get; set; }
|
|
public string Name { get; set; } = string.Empty;
|
|
public string Description { get; set; } = string.Empty;
|
|
public decimal Price { get; set; }
|
|
public decimal Weight { get; set; }
|
|
public ProductWeightUnit WeightUnit { get; set; }
|
|
public int StockQuantity { get; set; }
|
|
public Guid CategoryId { get; set; }
|
|
public string CategoryName { get; set; } = string.Empty;
|
|
public DateTime CreatedAt { get; set; }
|
|
public DateTime UpdatedAt { get; set; }
|
|
public bool IsActive { get; set; }
|
|
public List<ProductPhotoDto> Photos { get; set; } = new();
|
|
public List<ProductVariationDto> Variations { get; set; } = new();
|
|
}
|
|
|
|
public class ProductPhotoDto
|
|
{
|
|
public Guid Id { get; set; }
|
|
public string FileName { get; set; } = string.Empty;
|
|
public string FilePath { get; set; } = string.Empty;
|
|
public string? AltText { get; set; }
|
|
public int SortOrder { get; set; }
|
|
}
|
|
|
|
public class CreateProductDto
|
|
{
|
|
[Required]
|
|
[StringLength(200)]
|
|
public string Name { get; set; } = string.Empty;
|
|
|
|
[Required(AllowEmptyStrings = true)]
|
|
public string Description { get; set; } = string.Empty;
|
|
|
|
[Required]
|
|
[Range(0.01, double.MaxValue, ErrorMessage = "Price must be greater than 0")]
|
|
public decimal Price { get; set; }
|
|
|
|
[Required]
|
|
[Range(0.01, double.MaxValue, ErrorMessage = "Weight must be greater than 0")]
|
|
public decimal Weight { get; set; }
|
|
|
|
public ProductWeightUnit WeightUnit { get; set; } = ProductWeightUnit.Grams;
|
|
|
|
[Range(0, int.MaxValue)]
|
|
public int StockQuantity { get; set; } = 0;
|
|
|
|
[Required(ErrorMessage = "Please select a category")]
|
|
public Guid CategoryId { get; set; }
|
|
}
|
|
|
|
public class UpdateProductDto
|
|
{
|
|
[StringLength(200)]
|
|
public string? Name { get; set; }
|
|
|
|
public string? Description { get; set; }
|
|
|
|
[Range(0.01, double.MaxValue)]
|
|
public decimal? Price { get; set; }
|
|
|
|
public decimal? Weight { get; set; }
|
|
|
|
public ProductWeightUnit? WeightUnit { get; set; }
|
|
|
|
[Range(0, int.MaxValue)]
|
|
public int? StockQuantity { get; set; }
|
|
|
|
public Guid? CategoryId { get; set; }
|
|
|
|
public bool? IsActive { get; set; }
|
|
}
|
|
|
|
public class CreateProductPhotoDto
|
|
{
|
|
[Required]
|
|
public Guid ProductId { get; set; }
|
|
|
|
[Required]
|
|
public string PhotoUrl { get; set; } = string.Empty;
|
|
|
|
public string? AltText { get; set; }
|
|
|
|
public int DisplayOrder { get; set; }
|
|
}
|
|
|
|
public class ProductVariationDto
|
|
{
|
|
public Guid Id { get; set; }
|
|
public Guid ProductId { get; set; }
|
|
public string Name { get; set; } = string.Empty;
|
|
public string Description { get; set; } = string.Empty;
|
|
public int Quantity { get; set; }
|
|
public decimal Price { get; set; }
|
|
public decimal PricePerUnit { get; set; }
|
|
public int SortOrder { get; set; }
|
|
public bool IsActive { get; set; }
|
|
public DateTime CreatedAt { get; set; }
|
|
public DateTime UpdatedAt { get; set; }
|
|
}
|
|
|
|
public class CreateProductVariationDto
|
|
{
|
|
[Required]
|
|
public Guid ProductId { get; set; }
|
|
|
|
[Required]
|
|
[StringLength(100)]
|
|
public string Name { get; set; } = string.Empty;
|
|
|
|
public string Description { get; set; } = string.Empty;
|
|
|
|
[Required]
|
|
[Range(1, int.MaxValue, ErrorMessage = "Quantity must be at least 1")]
|
|
public int Quantity { get; set; }
|
|
|
|
[Required]
|
|
[Range(0.01, double.MaxValue, ErrorMessage = "Price must be greater than 0")]
|
|
public decimal Price { get; set; }
|
|
|
|
[Range(0, int.MaxValue)]
|
|
public int SortOrder { get; set; }
|
|
}
|
|
|
|
public class UpdateProductVariationDto
|
|
{
|
|
[StringLength(100)]
|
|
public string? Name { get; set; }
|
|
|
|
public string? Description { get; set; }
|
|
|
|
[Range(1, int.MaxValue)]
|
|
public int? Quantity { get; set; }
|
|
|
|
[Range(0.01, double.MaxValue)]
|
|
public decimal? Price { get; set; }
|
|
|
|
[Range(0, int.MaxValue)]
|
|
public int? SortOrder { get; set; }
|
|
|
|
public bool? IsActive { get; set; }
|
|
} |