2017-01-25 17:10:08 +01:00
|
|
|
|
using Microsoft.eShopOnContainers.Services.Ordering.Domain.AggregatesModel.BuyerAggregate;
|
2017-01-30 15:46:43 +01:00
|
|
|
|
using Microsoft.eShopOnContainers.Services.Ordering.Domain.Seedwork;
|
2017-01-25 17:10:08 +01:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
|
|
|
|
|
namespace Microsoft.eShopOnContainers.Services.Ordering.Domain.AggregatesModel.OrderAggregate
|
2016-11-21 12:41:36 +01:00
|
|
|
|
{
|
|
|
|
|
public class Order
|
2017-01-25 17:10:08 +01:00
|
|
|
|
: Entity
|
2016-11-21 12:41:36 +01:00
|
|
|
|
{
|
2017-02-02 13:34:44 -08:00
|
|
|
|
// DDD Patterns comment
|
|
|
|
|
// Using private fields, allowed since EF Core 1.1, is a much better encapsulation
|
|
|
|
|
// aligned with DDD Aggregates and Domain Entities (Instead of properties and property collections)
|
|
|
|
|
private DateTime _orderDate;
|
|
|
|
|
|
|
|
|
|
//TO DO: These fields need to be converted to a VALUE-OBJECT "Address"
|
2017-01-25 17:10:08 +01:00
|
|
|
|
private string _street;
|
|
|
|
|
private string _city;
|
|
|
|
|
private string _state;
|
|
|
|
|
private string _country;
|
|
|
|
|
private string _zipCode;
|
2016-11-22 18:40:47 +01:00
|
|
|
|
|
2017-01-25 17:10:08 +01:00
|
|
|
|
public Buyer Buyer { get; private set; }
|
2017-02-02 13:34:44 -08:00
|
|
|
|
private int _buyerId;
|
2016-11-21 12:41:36 +01:00
|
|
|
|
|
2017-01-25 17:10:08 +01:00
|
|
|
|
public OrderStatus OrderStatus { get; private set; }
|
2017-02-02 13:34:44 -08:00
|
|
|
|
private int _orderStatusId;
|
2016-11-21 12:41:36 +01:00
|
|
|
|
|
2017-02-02 13:34:44 -08:00
|
|
|
|
// DDD Patterns comment
|
|
|
|
|
// Using a private collection field, better for DDD Aggregate's encapsulation
|
|
|
|
|
// so OrderItems cannot be added from "outside the AggregateRoot" directly to the collection,
|
|
|
|
|
// but only through the method OrderAggrergateRoot.AddOrderItem() which includes behaviour.
|
|
|
|
|
private readonly HashSet<OrderItem> _orderItems;
|
|
|
|
|
|
|
|
|
|
//TODO: Use List<> instead of HashSet<> because of the comment below
|
|
|
|
|
// So we won't need the ".ToList()"
|
|
|
|
|
|
|
|
|
|
public IEnumerable<OrderItem> OrderItems => _orderItems.ToList().AsReadOnly();
|
|
|
|
|
// Using List<>.AsReadOnly()
|
|
|
|
|
//This will create a read only wrapper around the private list.
|
|
|
|
|
//It's much cheaper than .ToList() because it will not have to copy all items in a new collection. (Just one heap alloc for the wrapper instance)
|
|
|
|
|
//https://msdn.microsoft.com/en-us/library/e78dcd75(v=vs.110).aspx
|
2016-11-22 18:40:47 +01:00
|
|
|
|
|
2017-01-27 11:38:23 +01:00
|
|
|
|
public PaymentMethod PaymentMethod { get; private set; }
|
2017-02-02 13:34:44 -08:00
|
|
|
|
private int _paymentMethodId;
|
2016-11-22 18:40:47 +01:00
|
|
|
|
|
2016-11-21 12:41:36 +01:00
|
|
|
|
protected Order() { }
|
2016-11-24 14:59:25 +01:00
|
|
|
|
|
2017-01-27 11:38:23 +01:00
|
|
|
|
public Order(int buyerId, int paymentMethodId, Address address)
|
2016-11-24 14:59:25 +01:00
|
|
|
|
{
|
2017-01-25 17:10:08 +01:00
|
|
|
|
|
|
|
|
|
_buyerId = buyerId;
|
2017-01-27 11:38:23 +01:00
|
|
|
|
_paymentMethodId = paymentMethodId;
|
2017-01-25 17:10:08 +01:00
|
|
|
|
_orderStatusId = OrderStatus.InProcess.Id;
|
|
|
|
|
_orderDate = DateTime.UtcNow;
|
|
|
|
|
_street = address.Street;
|
|
|
|
|
_city = address.City;
|
|
|
|
|
_state = address.State;
|
|
|
|
|
_country = address.Country;
|
|
|
|
|
_zipCode = address.ZipCode;
|
|
|
|
|
|
|
|
|
|
_orderItems = new HashSet<OrderItem>();
|
2016-11-24 14:59:25 +01:00
|
|
|
|
}
|
|
|
|
|
|
2017-02-02 13:34:44 -08:00
|
|
|
|
// DDD Patterns comment
|
|
|
|
|
// This Order AggregateRoot's method "AddOrderitem()" should be the only way to add Items to the Order,
|
|
|
|
|
// so any behavior (discounts, etc.) and validations are controlled by the AggregateRoot
|
|
|
|
|
// in order to maintain consistency between the whole Aggregate.
|
2017-01-30 15:46:43 +01:00
|
|
|
|
public void AddOrderItem(int productId, string productName, decimal unitPrice, decimal discount, string pictureUrl, int units = 1)
|
2016-11-24 14:59:25 +01:00
|
|
|
|
{
|
2017-01-25 17:10:08 +01:00
|
|
|
|
var existingOrderForProduct = _orderItems.Where(o => o.ProductId == productId)
|
|
|
|
|
.SingleOrDefault();
|
|
|
|
|
|
|
|
|
|
if (existingOrderForProduct != null)
|
2016-11-24 14:59:25 +01:00
|
|
|
|
{
|
2017-01-25 17:10:08 +01:00
|
|
|
|
//if previous line exist modify it with higher discount and units..
|
|
|
|
|
|
|
|
|
|
if (discount > existingOrderForProduct.GetCurrentDiscount())
|
|
|
|
|
{
|
|
|
|
|
existingOrderForProduct.SetNewDiscount(discount);
|
|
|
|
|
existingOrderForProduct.AddUnits(units);
|
|
|
|
|
}
|
2016-11-24 14:59:25 +01:00
|
|
|
|
}
|
2017-01-25 17:10:08 +01:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
//add validated new order item
|
2016-11-24 14:59:25 +01:00
|
|
|
|
|
2017-01-30 15:46:43 +01:00
|
|
|
|
var orderItem = new OrderItem(productId, productName, unitPrice, discount, pictureUrl, units);
|
2016-12-14 18:23:57 +01:00
|
|
|
|
|
2017-01-25 17:10:08 +01:00
|
|
|
|
_orderItems.Add(orderItem);
|
|
|
|
|
}
|
2016-12-14 18:23:57 +01:00
|
|
|
|
}
|
2016-11-21 12:41:36 +01:00
|
|
|
|
}
|
|
|
|
|
}
|