Enables individual variants to have their own prices, overriding the base product price. **Database Changes:** - Added Price (decimal?, nullable) to ProductVariants table - Added ProductVariantId to OrderItems table with foreign key relationship - Created index on OrderItems.ProductVariantId for performance **API Changes:** - ProductVariantDto: Added Price field - CreateProductVariantDto: Added Price field with validation - UpdateProductVariantDto: Added Price field - OrderItemDto: Added ProductVariantId and ProductVariantName - CreateOrderItemDto: Added ProductVariantId **Business Logic:** - OrderService: Variant price overrides base price (but multi-buy takes precedence) - ProductService: All variant CRUD operations support Price field **Admin UI:** - CreateVariant: Price input with £ symbol and base price placeholder - EditVariant: Price editing with £ symbol - ProductVariants list: Shows variant price or "(base)" indicator **Client Library:** - Updated all DTOs to match server-side changes - Full support for variant pricing in order creation **Migration:** - EF Core migration: 20251003173458_AddVariantPricing - Backward compatible: NULL values supported for existing data **Use Case:** Products with size/color variants can now have different prices: - Small T-shirt: £15.00 (variant override) - Medium T-shirt: £18.00 (uses base price) - Large T-shirt: £20.00 (variant override) 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
151 lines
4.5 KiB
C#
151 lines
4.5 KiB
C#
using System.ComponentModel.DataAnnotations;
|
|
using LittleShop.Enums;
|
|
|
|
namespace LittleShop.DTOs;
|
|
|
|
public class OrderDto
|
|
{
|
|
public Guid Id { get; set; }
|
|
public Guid? CustomerId { get; set; }
|
|
public string? IdentityReference { get; set; }
|
|
public OrderStatus Status { get; set; }
|
|
|
|
// Customer Information (embedded for convenience)
|
|
public CustomerSummaryDto? Customer { get; set; }
|
|
public decimal TotalAmount { get; set; }
|
|
public string Currency { get; set; } = "GBP";
|
|
public string ShippingName { get; set; } = string.Empty;
|
|
public string ShippingAddress { get; set; } = string.Empty;
|
|
public string ShippingCity { get; set; } = string.Empty;
|
|
public string ShippingPostCode { get; set; } = string.Empty;
|
|
public string ShippingCountry { get; set; } = string.Empty;
|
|
public string? Notes { get; set; }
|
|
public string? TrackingNumber { get; set; }
|
|
public DateTime CreatedAt { get; set; }
|
|
public DateTime UpdatedAt { get; set; }
|
|
public DateTime? PaidAt { get; set; }
|
|
|
|
// Workflow timestamps
|
|
public DateTime? AcceptedAt { get; set; }
|
|
public DateTime? PackingStartedAt { get; set; }
|
|
public DateTime? DispatchedAt { get; set; }
|
|
public DateTime? ExpectedDeliveryDate { get; set; }
|
|
public DateTime? ActualDeliveryDate { get; set; }
|
|
public DateTime? OnHoldAt { get; set; }
|
|
|
|
// Workflow details
|
|
public string? AcceptedByUser { get; set; }
|
|
public string? PackedByUser { get; set; }
|
|
public string? DispatchedByUser { get; set; }
|
|
public string? OnHoldReason { get; set; }
|
|
|
|
// Legacy field (for backward compatibility)
|
|
public DateTime? ShippedAt { get; set; }
|
|
|
|
public List<OrderItemDto> Items { get; set; } = new();
|
|
public List<CryptoPaymentDto> Payments { get; set; } = new();
|
|
}
|
|
|
|
public class OrderItemDto
|
|
{
|
|
public Guid Id { get; set; }
|
|
public Guid ProductId { get; set; }
|
|
public Guid? ProductMultiBuyId { get; set; }
|
|
public Guid? ProductVariantId { get; set; }
|
|
public string ProductName { get; set; } = string.Empty;
|
|
public string? ProductMultiBuyName { get; set; }
|
|
public string? ProductVariantName { get; set; }
|
|
public string? SelectedVariant { get; set; }
|
|
public int Quantity { get; set; }
|
|
public decimal UnitPrice { get; set; }
|
|
public decimal TotalPrice { get; set; }
|
|
}
|
|
|
|
public class CreateOrderDto
|
|
{
|
|
// Either Customer ID (for registered customers) OR Identity Reference (for anonymous)
|
|
public Guid? CustomerId { get; set; }
|
|
|
|
public string? IdentityReference { get; set; }
|
|
|
|
// Customer Information (collected at checkout for anonymous orders)
|
|
public CreateCustomerDto? CustomerInfo { get; set; }
|
|
|
|
[Required]
|
|
public string ShippingName { get; set; } = string.Empty;
|
|
|
|
[Required]
|
|
public string ShippingAddress { get; set; } = string.Empty;
|
|
|
|
[Required]
|
|
public string ShippingCity { get; set; } = string.Empty;
|
|
|
|
[Required]
|
|
public string ShippingPostCode { get; set; } = string.Empty;
|
|
|
|
[Required]
|
|
public string ShippingCountry { get; set; } = "United Kingdom";
|
|
|
|
[Required]
|
|
public List<CreateOrderItemDto> Items { get; set; } = new();
|
|
|
|
public string? Notes { get; set; }
|
|
}
|
|
|
|
public class CreateOrderItemDto
|
|
{
|
|
[Required]
|
|
public Guid ProductId { get; set; }
|
|
|
|
public Guid? ProductMultiBuyId { get; set; } // Optional: if specified, use multi-buy pricing
|
|
|
|
public Guid? ProductVariantId { get; set; } // Optional: specific variant (used for variant price override)
|
|
|
|
public string? SelectedVariant { get; set; } // Optional: variant choice (color/flavor) - deprecated, use ProductVariantId
|
|
|
|
[Range(1, int.MaxValue)]
|
|
public int Quantity { get; set; }
|
|
}
|
|
|
|
public class UpdateOrderStatusDto
|
|
{
|
|
public OrderStatus Status { get; set; }
|
|
public string? TrackingNumber { get; set; }
|
|
public string? Notes { get; set; }
|
|
}
|
|
|
|
// New workflow DTOs
|
|
public class AcceptOrderDto
|
|
{
|
|
public string? Notes { get; set; }
|
|
}
|
|
|
|
public class StartPackingDto
|
|
{
|
|
public string? Notes { get; set; }
|
|
}
|
|
|
|
public class DispatchOrderDto
|
|
{
|
|
[Required]
|
|
[StringLength(100)]
|
|
public string TrackingNumber { get; set; } = string.Empty;
|
|
|
|
public int EstimatedDeliveryDays { get; set; } = 3; // Default 3 working days
|
|
public string? Notes { get; set; }
|
|
}
|
|
|
|
public class PutOnHoldDto
|
|
{
|
|
[Required]
|
|
[StringLength(500)]
|
|
public string Reason { get; set; } = string.Empty;
|
|
|
|
public string? Notes { get; set; }
|
|
}
|
|
|
|
public class MarkDeliveredDto
|
|
{
|
|
public DateTime? ActualDeliveryDate { get; set; }
|
|
public string? Notes { get; set; }
|
|
} |