Inital commit
This commit is contained in:
@@ -14,15 +14,27 @@ namespace Thelia\Action;
|
||||
|
||||
use Symfony\Component\EventDispatcher\EventDispatcherInterface;
|
||||
use Symfony\Component\EventDispatcher\EventSubscriberInterface;
|
||||
use Symfony\Component\HttpFoundation\RequestStack;
|
||||
use Thelia\Core\Event\Cart\CartCreateEvent;
|
||||
use Thelia\Core\Event\Cart\CartDuplicationEvent;
|
||||
use Thelia\Core\Event\Cart\CartPersistEvent;
|
||||
use Thelia\Core\Event\Cart\CartRestoreEvent;
|
||||
use Thelia\Core\Event\Cart\CartEvent;
|
||||
use Thelia\Core\Event\Currency\CurrencyChangeEvent;
|
||||
use Thelia\Core\Event\TheliaEvents;
|
||||
use Thelia\Exception\TheliaProcessException;
|
||||
use Thelia\Model\Base\CustomerQuery;
|
||||
use Thelia\Model\Base\ProductSaleElementsQuery;
|
||||
use Thelia\Model\Currency as CurrencyModel;
|
||||
use Thelia\Model\CartItem;
|
||||
use Thelia\Model\Cart as CartModel;
|
||||
use Thelia\Model\CartItemQuery;
|
||||
use Thelia\Model\CartQuery;
|
||||
use Thelia\Model\ConfigQuery;
|
||||
use Thelia\Model\Currency;
|
||||
use Thelia\Model\ProductSaleElementsQuery;
|
||||
use Thelia\Model\Customer as CustomerModel;
|
||||
use Thelia\Model\ProductSaleElements;
|
||||
use Thelia\Model\Tools\ProductPriceTools;
|
||||
use Thelia\Tools\TokenProvider;
|
||||
|
||||
/**
|
||||
*
|
||||
@@ -30,19 +42,44 @@ use Thelia\Model\Tools\ProductPriceTools;
|
||||
*
|
||||
* Class Cart
|
||||
* @package Thelia\Action
|
||||
* @author Manuel Raynaud <mraynaud@openstudio.fr>
|
||||
* @author Manuel Raynaud <manu@raynaud.io>
|
||||
*/
|
||||
class Cart extends BaseAction implements EventSubscriberInterface
|
||||
{
|
||||
/** @var RequestStack */
|
||||
protected $requestStack;
|
||||
|
||||
/** @var TokenProvider */
|
||||
protected $tokenProvider;
|
||||
|
||||
public function __construct(RequestStack $requestStack, TokenProvider $tokenProvider)
|
||||
{
|
||||
$this->requestStack = $requestStack;
|
||||
|
||||
$this->tokenProvider = $tokenProvider;
|
||||
}
|
||||
|
||||
public function persistCart(CartPersistEvent $event)
|
||||
{
|
||||
$cart = $event->getCart();
|
||||
|
||||
if ($cart->isNew()) {
|
||||
$cart
|
||||
->setToken($this->generateCartCookieIdentifier())
|
||||
->save();
|
||||
$this->getSession()->setSessionCart($cart);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* add an article in the current cart
|
||||
*
|
||||
* @param \Thelia\Core\Event\Cart\CartEvent $event
|
||||
* @param $eventName
|
||||
* @param EventDispatcherInterface $dispatcher
|
||||
*/
|
||||
public function addItem(CartEvent $event)
|
||||
public function addItem(CartEvent $event, $eventName, EventDispatcherInterface $dispatcher)
|
||||
{
|
||||
|
||||
$cart = $event->getCart();
|
||||
$newness = $event->getNewness();
|
||||
$append = $event->getAppend();
|
||||
@@ -51,6 +88,11 @@ class Cart extends BaseAction implements EventSubscriberInterface
|
||||
$customer = $cart->getCustomer();
|
||||
$discount = 0;
|
||||
|
||||
if ($cart->isNew()) {
|
||||
$persistEvent = new CartPersistEvent($cart);
|
||||
$dispatcher->dispatch(TheliaEvents::CART_PERSIST, $persistEvent);
|
||||
}
|
||||
|
||||
if (null !== $customer && $customer->getDiscount() > 0) {
|
||||
$discount = $customer->getDiscount();
|
||||
}
|
||||
@@ -58,29 +100,29 @@ class Cart extends BaseAction implements EventSubscriberInterface
|
||||
$productSaleElementsId = $event->getProductSaleElementsId();
|
||||
$productId = $event->getProduct();
|
||||
|
||||
$cartItem = $this->findItem($cart->getId(), $productId, $productSaleElementsId);
|
||||
// Search for an identical item in the cart
|
||||
$findItemEvent = clone $event;
|
||||
|
||||
$dispatcher->dispatch(TheliaEvents::CART_FINDITEM, $findItemEvent);
|
||||
|
||||
$cartItem = $findItemEvent->getCartItem();
|
||||
|
||||
if ($cartItem === null || $newness) {
|
||||
|
||||
$productSaleElements = ProductSaleElementsQuery::create()
|
||||
->findPk($productSaleElementsId);
|
||||
$productSaleElements = ProductSaleElementsQuery::create()->findPk($productSaleElementsId);
|
||||
|
||||
if (null !== $productSaleElements) {
|
||||
$productPrices = $productSaleElements->getPricesByCurrency($currency, $discount);
|
||||
$event->setCartItem(
|
||||
$this->doAddItem($event->getDispatcher(), $cart, $productId, $productSaleElements, $quantity, $productPrices)
|
||||
);
|
||||
|
||||
$cartItem = $this->doAddItem($dispatcher, $cart, $productId, $productSaleElements, $quantity, $productPrices);
|
||||
} else {
|
||||
// We did no find any PSE... Something is wrong with the DB, just throw an exception.
|
||||
throw new TheliaProcessException("This item cannot be added to the cart: no matching product sale element was found.");
|
||||
}
|
||||
} elseif ($append && $cartItem !== null) {
|
||||
$cartItem->addQuantity($quantity)->save();
|
||||
}
|
||||
|
||||
if ($append && $cartItem !== null) {
|
||||
$cartItem->addQuantity($quantity)
|
||||
->save();
|
||||
|
||||
$event->setCartItem(
|
||||
$cartItem
|
||||
);
|
||||
}
|
||||
$event->setCartItem($cartItem);
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -91,13 +133,16 @@ class Cart extends BaseAction implements EventSubscriberInterface
|
||||
*/
|
||||
public function deleteItem(CartEvent $event)
|
||||
{
|
||||
if (null !== $cartItemId = $event->getCartItem()) {
|
||||
if (null !== $cartItemId = $event->getCartItemId()) {
|
||||
$cart = $event->getCart();
|
||||
CartItemQuery::create()
|
||||
->filterByCartId($cart->getId())
|
||||
->filterById($cartItemId)
|
||||
->delete();
|
||||
|
||||
// Force an update of the Cart object to provide
|
||||
// to other listeners an updated CartItem collection.
|
||||
$cart->clearCartItems();
|
||||
}
|
||||
}
|
||||
|
||||
@@ -119,10 +164,12 @@ class Cart extends BaseAction implements EventSubscriberInterface
|
||||
* don't use Form here just test the Request.
|
||||
*
|
||||
* @param \Thelia\Core\Event\Cart\CartEvent $event
|
||||
* @param $eventName
|
||||
* @param EventDispatcherInterface $dispatcher
|
||||
*/
|
||||
public function changeItem(CartEvent $event)
|
||||
public function changeItem(CartEvent $event, $eventName, EventDispatcherInterface $dispatcher)
|
||||
{
|
||||
if ((null !== $cartItemId = $event->getCartItem()) && (null !== $quantity = $event->getQuantity())) {
|
||||
if ((null !== $cartItemId = $event->getCartItemId()) && (null !== $quantity = $event->getQuantity())) {
|
||||
$cart = $event->getCart();
|
||||
|
||||
$cartItem = CartItemQuery::create()
|
||||
@@ -132,16 +179,16 @@ class Cart extends BaseAction implements EventSubscriberInterface
|
||||
|
||||
if ($cartItem) {
|
||||
$event->setCartItem(
|
||||
$this->updateQuantity($event->getDispatcher(), $cartItem, $quantity)
|
||||
$this->updateQuantity($dispatcher, $cartItem, $quantity)
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public function updateCart(CurrencyChangeEvent $event)
|
||||
public function updateCart(CurrencyChangeEvent $event, $eventName, EventDispatcherInterface $dispatcher)
|
||||
{
|
||||
$session = $event->getRequest()->getSession();
|
||||
$cart = $session->getCart();
|
||||
$cart = $event->getRequest()->getSession()->getSessionCart($dispatcher);
|
||||
|
||||
if (null !== $cart) {
|
||||
$this->updateCartPrices($cart, $event->getCurrency());
|
||||
}
|
||||
@@ -154,9 +201,8 @@ class Cart extends BaseAction implements EventSubscriberInterface
|
||||
* @param \Thelia\Model\Cart $cart
|
||||
* @param \Thelia\Model\Currency $currency
|
||||
*/
|
||||
public function updateCartPrices(\Thelia\Model\Cart $cart, Currency $currency)
|
||||
public function updateCartPrices(CartModel $cart, CurrencyModel $currency)
|
||||
{
|
||||
|
||||
$customer = $cart->getCustomer();
|
||||
$discount = 0;
|
||||
|
||||
@@ -180,47 +226,17 @@ class Cart extends BaseAction implements EventSubscriberInterface
|
||||
// update the currency cart
|
||||
$cart->setCurrencyId($currency->getId());
|
||||
$cart->save();
|
||||
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Returns an array of event names this subscriber wants to listen to.
|
||||
*
|
||||
* The array keys are event names and the value can be:
|
||||
*
|
||||
* * The method name to call (priority defaults to 0)
|
||||
* * An array composed of the method name to call and the priority
|
||||
* * An array of arrays composed of the method names to call and respective
|
||||
* priorities, or 0 if unset
|
||||
*
|
||||
* For instance:
|
||||
*
|
||||
* * array('eventName' => 'methodName')
|
||||
* * array('eventName' => array('methodName', $priority))
|
||||
* * array('eventName' => array(array('methodName1', $priority), array('methodName2'))
|
||||
*
|
||||
* @return array The event names to listen to
|
||||
*
|
||||
* @api
|
||||
*/
|
||||
public static function getSubscribedEvents()
|
||||
{
|
||||
return array(
|
||||
TheliaEvents::CART_ADDITEM => array("addItem", 128),
|
||||
TheliaEvents::CART_DELETEITEM => array("deleteItem", 128),
|
||||
TheliaEvents::CART_UPDATEITEM => array("changeItem", 128),
|
||||
TheliaEvents::CART_CLEAR => array("clear", 128),
|
||||
TheliaEvents::CHANGE_DEFAULT_CURRENCY => array("updateCart", 128),
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* increase the quantity for an existing cartItem
|
||||
*
|
||||
* @param EventDispatcherInterface $dispatcher
|
||||
* @param CartItem $cartItem
|
||||
* @param float $quantity
|
||||
* @param float $quantity
|
||||
*
|
||||
* @throws \Exception
|
||||
* @throws \Propel\Runtime\Exception\PropelException
|
||||
* @return CartItem
|
||||
*/
|
||||
protected function updateQuantity(EventDispatcherInterface $dispatcher, CartItem $cartItem, $quantity)
|
||||
@@ -235,17 +251,23 @@ class Cart extends BaseAction implements EventSubscriberInterface
|
||||
/**
|
||||
* try to attach a new item to an existing cart
|
||||
*
|
||||
* @param \Symfony\Component\EventDispatcher\EventDispatcherInterface $dispatcher
|
||||
* @param \Thelia\Model\Cart $cart
|
||||
* @param int $productId
|
||||
* @param \Thelia\Model\ProductSaleElements $productSaleElements
|
||||
* @param float $quantity
|
||||
* @param ProductPriceTools $productPrices
|
||||
* @param EventDispatcherInterface $dispatcher
|
||||
* @param \Thelia\Model\Cart $cart
|
||||
* @param int $productId
|
||||
* @param ProductSaleElements $productSaleElements
|
||||
* @param float $quantity
|
||||
* @param ProductPriceTools $productPrices
|
||||
*
|
||||
* @return CartItem
|
||||
*/
|
||||
protected function doAddItem(EventDispatcherInterface $dispatcher, \Thelia\Model\Cart $cart, $productId, \Thelia\Model\ProductSaleElements $productSaleElements, $quantity, ProductPriceTools $productPrices)
|
||||
{
|
||||
protected function doAddItem(
|
||||
EventDispatcherInterface $dispatcher,
|
||||
CartModel $cart,
|
||||
$productId,
|
||||
ProductSaleElements $productSaleElements,
|
||||
$quantity,
|
||||
ProductPriceTools $productPrices
|
||||
) {
|
||||
$cartItem = new CartItem();
|
||||
$cartItem->setDisptacher($dispatcher);
|
||||
$cartItem
|
||||
@@ -269,7 +291,9 @@ class Cart extends BaseAction implements EventSubscriberInterface
|
||||
* @param int $cartId
|
||||
* @param int $productId
|
||||
* @param int $productSaleElementsId
|
||||
* @return ChildCartItem
|
||||
* @return CartItem
|
||||
*
|
||||
* @deprecated this method is deprecated. Dispatch a TheliaEvents::CART_FINDITEM instead
|
||||
*/
|
||||
protected function findItem($cartId, $productId, $productSaleElementsId)
|
||||
{
|
||||
@@ -280,4 +304,247 @@ class Cart extends BaseAction implements EventSubscriberInterface
|
||||
->findOne();
|
||||
}
|
||||
|
||||
/**
|
||||
* Find a specific record in CartItem table using the current CartEvent
|
||||
*
|
||||
* @param CartEvent $event the cart event
|
||||
*/
|
||||
public function findCartItem(CartEvent $event)
|
||||
{
|
||||
// Do not try to find a cartItem if one exists in the event, as previous event handlers
|
||||
// mays have put it in th event.
|
||||
if (null === $event->getCartItem() && null !== $foundItem = CartItemQuery::create()
|
||||
->filterByCartId($event->getCart()->getId())
|
||||
->filterByProductId($event->getProduct())
|
||||
->filterByProductSaleElementsId($event->getProductSaleElementsId())
|
||||
->findOne()) {
|
||||
$event->setCartItem($foundItem);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Search if cart already exists in session. If not try to restore it from the cart cookie,
|
||||
* or duplicate an old one.
|
||||
*
|
||||
* @param CartRestoreEvent $cartRestoreEvent
|
||||
* @param $eventName
|
||||
* @param EventDispatcherInterface $dispatcher
|
||||
*/
|
||||
public function restoreCurrentCart(CartRestoreEvent $cartRestoreEvent, $eventName, EventDispatcherInterface $dispatcher)
|
||||
{
|
||||
$cookieName = ConfigQuery::read("cart.cookie_name", 'thelia_cart');
|
||||
$persistentCookie = ConfigQuery::read("cart.use_persistent_cookie", 1);
|
||||
|
||||
$cart = null;
|
||||
|
||||
if ($this->requestStack->getCurrentRequest()->cookies->has($cookieName) && $persistentCookie) {
|
||||
$cart = $this->managePersistentCart($cartRestoreEvent, $cookieName, $dispatcher);
|
||||
} elseif (!$persistentCookie) {
|
||||
$cart = $this->manageNonPersistentCookie($cartRestoreEvent, $dispatcher);
|
||||
}
|
||||
|
||||
// Still no cart ? Create a new one.
|
||||
if (null === $cart) {
|
||||
$cart = $this->dispatchNewCart($dispatcher);
|
||||
}
|
||||
|
||||
$cartRestoreEvent->setCart($cart);
|
||||
}
|
||||
|
||||
/**
|
||||
* The cart token is not saved in a cookie, if the cart is present in session, we just change the customer id
|
||||
* if needed or create duplicate the current cart if the customer is not the same as customer already present in
|
||||
* the cart.
|
||||
*
|
||||
* @param CartRestoreEvent $cartRestoreEvent
|
||||
* @param EventDispatcherInterface $dispatcher
|
||||
* @return CartModel
|
||||
* @throws \Exception
|
||||
* @throws \Propel\Runtime\Exception\PropelException
|
||||
*/
|
||||
protected function manageNonPersistentCookie(CartRestoreEvent $cartRestoreEvent, EventDispatcherInterface $dispatcher)
|
||||
{
|
||||
$cart = $cartRestoreEvent->getCart();
|
||||
|
||||
if (null === $cart) {
|
||||
$cart = $this->dispatchNewCart($dispatcher);
|
||||
} else {
|
||||
$cart = $this->manageCartDuplicationAtCustomerLogin($cart, $dispatcher);
|
||||
}
|
||||
|
||||
return $cart;
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* The cart token is saved in a cookie so we try to retrieve it. Then the customer is checked.
|
||||
*
|
||||
* @param CartRestoreEvent $cartRestoreEvent
|
||||
* @param $cookieName
|
||||
* @param EventDispatcherInterface $dispatcher
|
||||
* @return CartModel
|
||||
* @throws \Exception
|
||||
* @throws \Propel\Runtime\Exception\PropelException
|
||||
*/
|
||||
protected function managePersistentCart(CartRestoreEvent $cartRestoreEvent, $cookieName, EventDispatcherInterface $dispatcher)
|
||||
{
|
||||
// The cart cookie exists -> get the cart token
|
||||
$token = $this->requestStack->getCurrentRequest()->cookies->get($cookieName);
|
||||
|
||||
// Check if a cart exists for this token
|
||||
if (null !== $cart = CartQuery::create()->findOneByToken($token)) {
|
||||
$cart = $this->manageCartDuplicationAtCustomerLogin($cart, $dispatcher);
|
||||
}
|
||||
|
||||
return $cart;
|
||||
}
|
||||
|
||||
protected function manageCartDuplicationAtCustomerLogin(CartModel $cart, EventDispatcherInterface $dispatcher)
|
||||
{
|
||||
/** @var CustomerModel $customer */
|
||||
if (null !== $customer = $this->getSession()->getCustomerUser()) {
|
||||
// Check if we have to duplicate the existing cart.
|
||||
|
||||
$duplicateCart = true;
|
||||
|
||||
// A customer is logged in.
|
||||
if (null === $cart->getCustomerId()) {
|
||||
// If the customer has a discount, whe have to duplicate the cart,
|
||||
// so that the discount will be applied to the products in cart.
|
||||
|
||||
if (0 === $customer->getDiscount() || 0 === $cart->countCartItems()) {
|
||||
// If no discount, or an empty cart, there's no need to duplicate.
|
||||
$duplicateCart = false;
|
||||
}
|
||||
}
|
||||
|
||||
if ($duplicateCart) {
|
||||
// Duplicate the cart
|
||||
$cart = $this->duplicateCart($dispatcher, $cart, $customer);
|
||||
} else {
|
||||
// No duplication required, just assign the cart to the customer
|
||||
$cart->setCustomerId($customer->getId())->save();
|
||||
}
|
||||
} elseif ($cart->getCustomerId() != null) {
|
||||
// The cart belongs to another user
|
||||
if (0 === $cart->countCartItems()) {
|
||||
// No items in cart, assign it to nobody.
|
||||
$cart->setCustomerId(null)->save();
|
||||
} else {
|
||||
// Some itemls in cart, duplicate it without assigning a customer ID.
|
||||
$cart = $this->duplicateCart($dispatcher, $cart);
|
||||
}
|
||||
}
|
||||
|
||||
return $cart;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param EventDispatcherInterface $dispatcher
|
||||
* @return CartModel
|
||||
*/
|
||||
protected function dispatchNewCart(EventDispatcherInterface $dispatcher)
|
||||
{
|
||||
$cartCreateEvent = new CartCreateEvent();
|
||||
|
||||
$dispatcher->dispatch(TheliaEvents::CART_CREATE_NEW, $cartCreateEvent);
|
||||
|
||||
return $cartCreateEvent->getCart();
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a new, empty cart object, and assign it to the current customer, if any.
|
||||
*
|
||||
* @param CartCreateEvent $cartCreateEvent
|
||||
*/
|
||||
public function createEmptyCart(CartCreateEvent $cartCreateEvent)
|
||||
{
|
||||
$cart = new CartModel();
|
||||
|
||||
$cart->setCurrency($this->getSession()->getCurrency(true));
|
||||
|
||||
/** @var CustomerModel $customer */
|
||||
if (null !== $customer = $this->getSession()->getCustomerUser()) {
|
||||
$cart->setCustomer(CustomerQuery::create()->findPk($customer->getId()));
|
||||
}
|
||||
|
||||
$this->getSession()->setSessionCart($cart);
|
||||
|
||||
if (ConfigQuery::read("cart.use_persistent_cookie", 1) == 1) {
|
||||
// set cart_use_cookie to "" to remove the cart cookie
|
||||
// see Thelia\Core\EventListener\ResponseListener
|
||||
$this->getSession()->set("cart_use_cookie", "");
|
||||
}
|
||||
|
||||
$cartCreateEvent->setCart($cart);
|
||||
}
|
||||
|
||||
/**
|
||||
* Duplicate an existing Cart. If a customer ID is provided the created cart will be attached to this customer.
|
||||
*
|
||||
* @param EventDispatcherInterface $dispatcher
|
||||
* @param CartModel $cart
|
||||
* @param CustomerModel $customer
|
||||
* @return CartModel
|
||||
*/
|
||||
protected function duplicateCart(EventDispatcherInterface $dispatcher, CartModel $cart, CustomerModel $customer = null)
|
||||
{
|
||||
$newCart = $cart->duplicate(
|
||||
$this->generateCartCookieIdentifier(),
|
||||
$customer,
|
||||
$this->getSession()->getCurrency(),
|
||||
$dispatcher
|
||||
);
|
||||
|
||||
$cartEvent = new CartDuplicationEvent($newCart, $cart);
|
||||
$dispatcher->dispatch(TheliaEvents::CART_DUPLICATE, $cartEvent);
|
||||
|
||||
return $cartEvent->getDuplicatedCart();
|
||||
}
|
||||
|
||||
/**
|
||||
* Generate the cart cookie identifier, or return null if the cart is only managed in the session object,
|
||||
* not in a client cookie.
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
protected function generateCartCookieIdentifier()
|
||||
{
|
||||
$id = null;
|
||||
|
||||
if (ConfigQuery::read("cart.use_persistent_cookie", 1) == 1) {
|
||||
$id = $this->tokenProvider->getToken();
|
||||
$this->getSession()->set('cart_use_cookie', $id);
|
||||
}
|
||||
|
||||
return $id;
|
||||
}
|
||||
|
||||
/**
|
||||
* {@inheritdoc}
|
||||
*/
|
||||
public static function getSubscribedEvents()
|
||||
{
|
||||
return array(
|
||||
TheliaEvents::CART_PERSIST => array("persistCart", 128),
|
||||
TheliaEvents::CART_RESTORE_CURRENT => array("restoreCurrentCart", 128),
|
||||
TheliaEvents::CART_CREATE_NEW => array("createEmptyCart", 128),
|
||||
TheliaEvents::CART_ADDITEM => array("addItem", 128),
|
||||
TheliaEvents::CART_FINDITEM => array("findCartItem", 128),
|
||||
TheliaEvents::CART_DELETEITEM => array("deleteItem", 128),
|
||||
TheliaEvents::CART_UPDATEITEM => array("changeItem", 128),
|
||||
TheliaEvents::CART_CLEAR => array("clear", 128),
|
||||
TheliaEvents::CHANGE_DEFAULT_CURRENCY => array("updateCart", 128),
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the session from the current request
|
||||
*
|
||||
* @return \Thelia\Core\HttpFoundation\Session\Session
|
||||
*/
|
||||
protected function getSession()
|
||||
{
|
||||
return $this->requestStack->getCurrentRequest()->getSession();
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user