#Cart
A modern PHP >=5.3.0 shopping cart
###Features
- Flexible, extensible, component based architecture
- Handles one or several cart instances (via cart manager)
- Configurable cart and cart items
- Cart and cart items support meta data
- Importable / Exportable carts
- Flexible state persistence
###Prerequisites
- PHP >=5.3.0
- This package can be installed using composer or can be integrated manually. If you are not using an autoloader make sure you include all of the php files in the
src
directory.
Note: All of the code examples in the rest of this document assume you have the required files above included manually or autoloaded.
###Architecture
This package is built out of a few components that work together:
- Cart - Stores cart items
- Cart Item - Item that will be stored in the cart
- Manager - Manages multiple cart instances + state persistence
- Storage - Manages persistence of a carts state
- Manager / Cart Facades - Facades that provide static access to the cart manager or a cart instance.
The cart component can be used with or without the manager component. If you choose not to use the manager component you will have to manage your storage implementation manually and you will not beable to use the facades (unless you extend and modify yourself).
The storage component is swapable. If you have a certain way you need to implement state persistence you can do this by implementing the storage interface: \Cart\Storage\StorageInterface
(see custom state persistence).
Note: Check out the demo for an example of how to implement a custom storage component.
This section will guide you through setting up the cart component with the manager component. 9 Times out of the 10 this will be the setup you want to use.
I recommend aliasing the manager and cart facades to something easier to write and reference:
use \Cart\Facade\Manager as CartManager;
use \Cart\Facade\Cart as Cart;
// this will enable you make calls like:
$numOfItems = Cart::itemCount():
CartManager::destroyCart();
Note: The rest of the code examples in this section will assume the above has been done.
You will need to pass an array of configuration options to the cart managers constructor. The configuration options would be best saved in their own file and included into the script when needed:
//config.php
return array(
'defaults' => array(
/**
* The decimal character to be used when formatting numbers
*/
'decimal_point' => '.',
/**
* The number of decimal places after a number
*/
'decimal_places' => 2,
/**
* The thousands separator character to be used when formatting numbers
*/
'thousands_separator' => ',',
/**
* Storage configuration options
*/
'storage' => array(
/**
* Should the cart autosave at the end of script execution. If true, need
* to define storage driver
*/
'autosave' => true,
/**
* The FQN of the class handling the storage implementation
*/
'driver'=> 'SessionStorage',
/**
* This is prepended to the storage identifier for the cart instance
*/
'storage_key_prefix' => 'cart_',
/**
* This is appended to the storage identifier for the cart instance
*/
'storage_key_suffix' => '_instance'
)
),
/**
* Carts to be instantiated when the cart manager is initialized. Each instance
* will inherit the default values above if not specified
*/
'carts' => array(
'Cart-01' => array(), //will inherit all of the above options
'Cart-02' => array(),
//'cart3' => array()
)
);
use \Cart\Manager;
$config = include '<path-to-config>/config.php';
$cartManager = new Cart\Manager($config);
From the configuration file you can define multiple cart instances. Each instance can have its own unique set of properties, otherwise it will just inherit from the default options.
Internally the cart and cart item components use number_format()
to format currency values. The configuration options decimal_point
, decimal_places
and thousands_separator
all relate to this.
To start using the cart manager facade you need to pass the cart manager instance to the init
method of the cart manager facade:
$cartManager = new Cart\Manager($config);
CartManager::init($cartManager);
The cart manager can only manage 1 cart instance at a time. This cart will be the cart that is in the current context. If you have multiple carts you can switch between them. This is known as switching context. You can control the context using the context
method of the manager component:
CartManager::context('Cart-02'); //switches the context to cart 2. Cart-02 is the ID of the cart as specified in the configuration file.
The cart proxy component makes use of the cart manager to retrieve the current cart in context. This is what allows you to directly make calls on a cart instance:
CartManager::context('Cart-02'); //switches the context to cart 2
$numOfItemsCart2 = Cart::itemsCount(); //get the number of items in cart 2
CartManager::context('Cart-01');
$numOfItemsCart1 = Cart::itemsCount(); //get the number of items in cart 1
By default CartManager::init()
will set the first cart in the carts
array, in the configuration file, to be the cart in context (in our example above this will be Cart-01
);
You can define the storage options in the configuration file.
- autosave - If set to true, the cart state will be saved automatically with out you having to do anything. If set to false, the state can be saved manually by calling
CartManager::saveState()
. - driver - This is the name of the class that contains your storage implementation. This class should implement
\Cart\Storage\StorageInterface
. If this option is set to a blank string, the cart manager will not attempt to preserve or restore state. - storage_key_prefix, storage_key_suffix - These are strings that will be added to the cart ID, which will be used as the identifier in your storage implementation. i.e If you are using the session to save the cart state, this would look something like
php$_SESSION['<cart_storage_prefix><cart_id><cart_storage_suffix>']
php or in our example abovephp$_SESSION['cart_Cart-01_instance']
php.
If you chose to autosave (recommended), internally this is registered as a shutdown function:
register_shutdown_function(array($this, 'saveState'), $cartID);
The state is restored when the cart manager is instantiated.
##Setup: Not Using The Cart Manager
This section will guide you through setting up the cart without the cart manager.
You can use the cart component without having to use the cart manager. You may want to do this for simpler setups where you do not need all the bells and whistles of the cart manager. The drawbacks to not using the cart manager are:
- You will have manage state persistance manually
- You will not beable to use the cart facade
- You cannot manage multiple cart instances from one central place
You will need to pass an array of configuration options to the carts constructor
. This kick starts the cart. This should be the first thing you do before you try and use the cart. The configuration options would be best saved in their own file and included into the script when needed:
//config.php
return array(
/**
* The decimal character to be used when formatting numbers
*/
'decimal_point' => '.',
/**
* The number of decimal places after a number
*/
'decimal_places' => 2,
/**
* The thousands separator character to be used when formatting numbers
*/
'thousands_separator' => ','
);
$config = require '<path-to-config>/config.php';
// the carts constructor takes the cart id and the conifg options
// as arguments. if the cart id is set to false then the cart will
// automatically generate an id for this cart instance
$cart = new \Cart\Cart('cart01', $config);
The cart component has import
and export
methods that you can use to manually control the state of the cart:
// saving cart state using sessions
$cartState = $cart->export();
$_SESSION['cart'] = $cartState;
…
// importing cart state using sessions
$cart->import($_SESSION['cart']);
The export
method returns an array of all the of carts items and any meta data stored on the cart.
The import
method expects an array formatted the same as what export
produces.
You are free to use whatever storage implementation you want as long as the data that is being imported is compatible.
You define an item with a simple array:
$item = array(
'name' => 'Apple Macbook Pro 13 inch Laptop',
'sku' => 'B004P8JCY8',
'id' => '2',
'price' => '824.17',
'tax' => '164.83',
'weight' => '3900',
'quantity' => 1
);
You can have as many or as little properties as required (it is always a good idea to have price, tax and name though). If a quantity is not supplied, the assumed quantity is 1.
The item is then passed to the carts add
:
// using the cart manager
Cart::add($item);
// not using the cart manager
$cart->add($item);
If an item is added to the cart that has already been added before, the quantity for that item is automatically updated. The same item is not added again, unless its properties are different (i.e has a different price or sku).
Internally each item is assigned a UID (unique identifier) and you will use this to interact with items already in the cart. The carts add
method returns the UID of the new item that is added:
$itemUID = Cart::add($item);
// $itemUID = 1b5792a046e0597daa4186169adb8d66
To remove an item you need to pass the items UID to the carts remove
method:
$uid = '1b5792a046e0597daa4186169adb8d66';
// using the cart manager
Cart::remove($uid);
// not using the cart manager
$cart->remove($uid);
To update an item you need to pass the items UID, the property to update and the new value of the property to the carts update
method:
$uid = '1b5792a046e0597daa4186169adb8d66';
try {
// using the cart manager
Cart::update($uid, 'quantity', 10);
// not using the cart manager
$cart->update($uid, 'quantity', 10);
}
If any property other than the qauntity is changed then the UID will be recalculated:
$uid = '1b5792a046e0597daa4186169adb8d66';
try {
$uid = Cart::update($uid, 'price', '900.00');
}
// $uid = 35a1699a3471501c6927cb93aa181c88
A \Cart\Exception\InvalidCartItemException
exception is thrown if the item you are trying to update does not exist (see below for how to check if an item exists).
Tip: You can also remove an item from the cart by setting its quantity to zero using the update
method.
You can check if an item exists in cart using the carts exists
method:
$uid = '1b5792a046e0597daa4186169adb8d66';
// using the cart manager
$itemExists = Cart::exists($uid);
// not using the cart manager
$itemExists = $cart->exists($uid);
You can also pass an item array to the exists
method:
$item = array(
'name' => 'Apple Macbook Pro 13 inch Laptop',
'sku' => 'B004P8JCY8',
'id' => '2',
'price' => '824.17',
'tax' => '164.83',
'weight' => '3900',
'quantity' => 1
);
if (Cart::exists($item))
{
//...
}
You may want to use this to check if an item has already been added to the cart before trying to add it.
// using the cart manager
Cart::itemCount();
// not using the cart manager
$cart->itemCount();
You can also get the total unique number of items in the cart (quantity is ignored) by passing true
to the carts itemCount
method:
Cart::itemCount(true);
The carts items
method returns an array of \Cart\Item
objects:
// using the cart manager
$items = Cart::items();
// not using the cart manager
$items = $cart->items();
If you just want 1 item you can use the carts item
method:
$uid = '1b5792a046e0597daa4186169adb8d66';
try {
// using the cart manager
$item = Cart::item($uid);
// not using the cart manager
$item = $cart->item($uid);
}
The carts item
method throws a \Exception\InvalidCartItemException
exception if the item does not exist in the cart.
The carts total
method returns the total value of the cart. This returns the total value including tax. You can return the total without tax by passing true
as a parameter:
// using the cart manager
$total = Cart::total();
$totalExcludingTax = Cart::total(true);
// not using the cart manager
$total = $cart->total();
$totalExcludingTax = $cart->total(true);
To just get the total tax you can use the tax
method:
// using the cart manager
$tax = Cart::tax();
// not using the cart manager
$tax = $cart->tax();
If the items in the cart have countable properties (weight etc.) you can use the carts getTotal
method to get the carts total value for this property:
// using the cart manager
$totalWeight = Cart::getTotal('weight');
// not using the cart manager
$totalWeight = $cart->getTotal('weight');
Internally the carts itemCount
, total
and tax
methods use this method.
The carts getTotal
method can be used in other ways - i.e to check if the cart has any discounted items:
$item1 = array(
'name' => 'Apple Macbook Pro 13 inch Laptop',
'sku' => 'B004P8JCY8',
'id' => '2',
'price' => '824.17',
'tax' => '164.83',
'weight' => '3900',
'quantity' => 1
);
$item2 = array(
'name' => 'Apple Macbook Pro 13 inch Laptop - Discounted',
'sku' => 'B004P8JCY8',
'id' => '2',
'price' => '724.17',
'tax' => '164.83',
'weight' => '3900',
'quantity' => 1,
'discounted' => 1
);
$item2 = array(
'name' => 'Apple Macbook Pro 13 inch Laptop - Heavily Discounted',
'sku' => 'B004P8JCY8',
'id' => '2',
'price' => '624.17',
'tax' => '164.83',
'weight' => '3900',
'quantity' => 1,
'discounted' => 1
);
Cart::add($item1);
Cart::add($item2);
$hasDiscounts = Cart::getTotal('discounted') > 0; // will be true as 2 items have the discounted property
if ($hasDiscounts)
{
//...
}
// using the cart manager
Cart::destroyCart();
// not using the cart manager
$cart->clear();
You store meta data on the cart. This can be useful for things like recording a customers checkout message, or keeping track of applied discounts etc.
// using the cart manager
Cart::setMeta('checkout_message', 'This is a checkout message');
// not using the cart manager
$cart->setMeta('checkout_message', 'This is a checkout message');
// using the cart manager
$checkoutMessage = Cart::getMeta('checkout_message');
// not using the cart manager
$cart->getMeta('checkout_message');
// using the cart manager
Cart::removeMeta('checkout_message');
// not using the cart manager
$cart->removeMeta('checkout_message');
You can also check if the cart has meta data set against it using the hasMeta
method. You can pass a key to this method to check if a specific piece of meta data exists:
// using the cart manager
$hasMeta = Cart::hasMeta();
$hasCheckoutMessage = Cart::hasMeta('checkout_message');
// not using the cart manager
$hasMeta = $cart->hasMeta();
$hasCheckoutMessage = $cart->hasMeta('checkout_message');
The cart has import
and export
methods that you can use:
// using the cart manager
$cartState = Cart::export();
$_SESSION['cart'] = $cartState;
// importing cart state using sessions
Cart::import($_SESSION['cart']);
// not using the cart manager
// saving cart state using sessions
$cartState = $cart->export();
$_SESSION['cart'] = $cartState;
// importing cart state using sessions
$cart->import($_SESSION['cart']);
The export
method returns an array of all the of carts items and any meta data stored on the cart.
The import
method expects an array formatted the same as what export
produces.
The cart items are instances of \Cart\Item
and expose methods for interacting with them:
try {
$item = Cart::item($uid);
$itemName = $item->get('name');
$itemSKU = $item->get('sku');
}
$cartItems = Cart::items();
foreach ($cartItems as $item)
{
$uid = $item->uid();
}
try {
$item = Cart::item($uid);
$itemQauntity = $item->quantity();
// or
$itemQauntity = $item->get('quantity');
}
You can update an items quantity via the item itself using the items setQuantity
method:
try {
Cart::item($uid)->setQuantity(10);
}
This is the same as doing:
try {
Cart::update($uid, 'quantity', 10);
}
The items totalPrice
method returns the total value of the item (taking into account quantity). This returns the total value including tax. You can return the total value without tax by passing true
as a parameter:
try {
$item = Cart::item($uid);
$itemPrice = $item->totalPrice();
$itemPriceExcludingTax = $item->totalPrice(true);
}
To just get the items total tax you can use the totalTax
method:
try {
$item = Cart::item($uid);
$itemTax = $item->totalTax();
}
You may want to get just the single price or single tax value for an item, this can be done with the items singlePrice
and singleTax
methods:
try {
$item = Cart::item($uid);
$itemSinglePrice = $item->singlePrice();
$itemSinglePriceExcludingTax = $item->singlePrice(true);
$itemSingleTax = $item->singleTax();
}
Cart items also support meta data in the same way the cart does (see above):
try {
$item = Cart::item($uid);
// set meta
$item->setMeta('engraving_text', 'Some engraving text here!');
// get meta
$engravingText = $item->getMeta('engraving_text');
// remove meta
$item->removeMeta('engraving_text');
// has meta
if ($item->hasMeta())
{
//...
}
}
You can export an item using the items export
method. This method returns an array of the items properties and meta data. You can also pass true to the export
method to have the items UID includeded in the export:
try {
$item = Cart::item($uid);
$data = $item->export();
$dataWithUID = $item->export(true);
}
You can retrieve a cart instance from the cart manager using the getCart
method. If you do not supply a cart id the cart in the current context (see context section above) will be returned. If the cart does not exist a \Cart\Exception\InvalidCartInstanceException
exception is thrown:
try {
$cartId = 'cart01';
// get cart with the ID 'cart01'
$cart01 = CartManager::getCart($cartId);
// gets the cart in the current context
$currentCart = CartManager::getCart();
}
You can check if a cart exists using the cart managers exists
method:
$cartId = 'cart01';
if (CartManager::exists($cartId))
{
//...
}
You can destroy a cart instance using the cart managers destroyCart
method:
$cartId = 'cart01';
CartManager::destroyCart($cartId);
If a cart id is not passed then the cart in the current context will be destroyed.
When a cart is destroyed it is removed from the cart manager and its saved state is cleared (via the state persitence handler). If you do not want to clear the carts state you can pass a second parameter of true
to the destroyCart
method:
$cartId = 'cart01';
// cart will be destroyed in the cart manager but its state will be saved
CartManager::destroyCart($cartId, true);
You can destroy all carts managed by the cart manager using the destroyAllCarts
method. You can preserve the carts saved state by passing a second parameter of true
:
CartManager::destroyAllCarts();
// all carts will be destroyed in the cart manager but there state will be saved
CartManager::destroyAllCarts();
You can create a new cart instance using the cart managers newCart
method. This method takes 4 parameters:
$cartId
- The ID for this cart$config
- The configuration options for this cart. If false is passed for this parameter the default configuration options will be used (see configuration section above)$overwrite
- Boolean to indicate if a cart with same ID exists, whether it should be overwritten or not$switchContext
- Once the new cart has been created should the current cart context be switched to this cart
CartManager::newCart('cart02');
$config = array(
'decimal_point' => '.',
'decimal_places' => 2,
'thousands_separator' => ',',
'storage' => array(
'autosave' => true,
'driver'=> 'CookieStorage',
'storage_key_prefix' => 'cart_',
'storage_key_suffix' => '_instance'
)
),
);
CartManager::newCart('cart02', $config);
newCart
throws an \Cart\Exception\DuplicateCartInstanceException
exception if $overwrite
is set to false and the cart exists:
try {
CartManager::newCart('cart02', false, false);
}
You can manually for a cart to save its state using the saveCartState
method:
$cartId = 'cart01';
CartManager::saveCartState($cartId);
You can also manually restore state for a cart using the restoreCartState
method:
$cartId = 'cart01';
CartManager::restoreCartState($cartId);
You can create your own custom state persistence component by extending \Cart\Storage\StorageInterface
:
namespace Cart\Storage;
class CustomStorage extends StorageInterface
{
//...
}
In the config you just need to set the storage driver to be your custom component:
$config = array(
'decimal_point' => '.',
'decimal_places' => 2,
'thousands_separator' => ',',
'storage' => array(
'autosave' => true,
'driver'=> 'CustomStorage',
'storage_key_prefix' => 'cart_',
'storage_key_suffix' => '_instance'
)
),
);
For more information on the inner workings i highly encourage you to take a deeper look at the source code.
There is also an demo provided with this repo. I highly recommend you check this out as well for a real world useage demo.