2012-11-03 04:16:51 +01:00
|
|
|
<?php
|
|
|
|
|
2016-06-15 06:03:16 +02:00
|
|
|
namespace SilverStripe\ORM;
|
|
|
|
|
|
|
|
use InvalidArgumentException;
|
|
|
|
use ArrayIterator;
|
2016-08-19 00:51:35 +02:00
|
|
|
use SilverStripe\ORM\FieldType\DBField;
|
2015-02-13 05:35:39 +01:00
|
|
|
|
2012-11-03 04:16:51 +01:00
|
|
|
/**
|
2013-05-20 12:18:07 +02:00
|
|
|
* An {@link ArrayList} that represents an unsaved relation.
|
2012-11-03 04:16:51 +01:00
|
|
|
*
|
2014-08-15 08:53:05 +02:00
|
|
|
* has_many and many_many relations cannot be saved until after the DataObject
|
|
|
|
* they're on has been written. This List pretends to be a RelationList and
|
2013-05-20 12:18:07 +02:00
|
|
|
* stores the related objects in memory.
|
2012-11-03 04:16:51 +01:00
|
|
|
*
|
2014-08-15 08:53:05 +02:00
|
|
|
* It can store both saved objects (as IDs) or unsaved objects (as instances
|
|
|
|
* of $dataClass). Unsaved objects are then written when the list is saved
|
2013-05-20 12:18:07 +02:00
|
|
|
* into an instance of {@link RelationList}.
|
2012-11-03 04:16:51 +01:00
|
|
|
*/
|
2016-11-29 00:31:16 +01:00
|
|
|
class UnsavedRelationList extends ArrayList implements Relation
|
|
|
|
{
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* The DataObject class name that this relation is on
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $baseClass;
|
2012-11-03 04:16:51 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* The name of the relation
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $relationName;
|
2012-11-03 04:16:51 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* The DataObject class name that this relation is querying
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $dataClass;
|
2012-11-03 04:16:51 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* The extra fields associated with the relation
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $extraFields = array();
|
2012-11-03 04:16:51 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* Create a new UnsavedRelationList
|
|
|
|
*
|
2017-05-17 07:40:13 +02:00
|
|
|
* @param string $baseClass
|
2016-11-29 00:31:16 +01:00
|
|
|
* @param string $relationName
|
|
|
|
* @param string $dataClass The DataObject class used in the relation
|
|
|
|
*/
|
|
|
|
public function __construct($baseClass, $relationName, $dataClass)
|
|
|
|
{
|
|
|
|
$this->baseClass = $baseClass;
|
|
|
|
$this->relationName = $relationName;
|
|
|
|
$this->dataClass = $dataClass;
|
|
|
|
parent::__construct();
|
|
|
|
}
|
2012-11-03 04:16:51 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* Add an item to this relationship
|
|
|
|
*
|
|
|
|
* @param mixed $item
|
|
|
|
* @param array $extraFields A map of additional columns to insert into the joinTable in the case of a many_many relation
|
|
|
|
*/
|
|
|
|
public function add($item, $extraFields = null)
|
|
|
|
{
|
|
|
|
$this->push($item, $extraFields);
|
|
|
|
}
|
2012-11-03 04:16:51 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* Save all the items in this list into the RelationList
|
|
|
|
*
|
|
|
|
* @param RelationList $list
|
|
|
|
*/
|
|
|
|
public function changeToList(RelationList $list)
|
|
|
|
{
|
|
|
|
foreach ($this->items as $key => $item) {
|
|
|
|
if (is_object($item)) {
|
|
|
|
/** @var DataObject $item */
|
|
|
|
$item->write();
|
|
|
|
}
|
|
|
|
$list->add($item, $this->extraFields[$key]);
|
|
|
|
}
|
|
|
|
}
|
2012-11-03 04:16:51 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* Pushes an item onto the end of this list.
|
|
|
|
*
|
|
|
|
* @param array|object $item
|
|
|
|
* @param array $extraFields
|
|
|
|
*/
|
|
|
|
public function push($item, $extraFields = null)
|
|
|
|
{
|
|
|
|
if ((is_object($item) && !$item instanceof $this->dataClass)
|
|
|
|
|| (!is_object($item) && !is_numeric($item))) {
|
|
|
|
throw new InvalidArgumentException(
|
|
|
|
"UnsavedRelationList::add() expecting a $this->dataClass object, or ID value",
|
|
|
|
E_USER_ERROR
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if (is_object($item) && $item->ID) {
|
|
|
|
$item = $item->ID;
|
|
|
|
}
|
|
|
|
$this->extraFields[] = $extraFields;
|
|
|
|
parent::push($item);
|
|
|
|
}
|
2012-11-03 04:16:51 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* Get the dataClass name for this relation, ie the DataObject ClassName
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function dataClass()
|
|
|
|
{
|
|
|
|
return $this->dataClass;
|
|
|
|
}
|
2012-11-03 04:16:51 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* Returns an Iterator for this relation.
|
|
|
|
*
|
|
|
|
* @return ArrayIterator
|
|
|
|
*/
|
|
|
|
public function getIterator()
|
|
|
|
{
|
|
|
|
return new ArrayIterator($this->toArray());
|
|
|
|
}
|
2012-11-03 04:16:51 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* Return an array of the actual items that this relation contains at this stage.
|
|
|
|
* This is when the query is actually executed.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function toArray()
|
|
|
|
{
|
|
|
|
$items = array();
|
|
|
|
foreach ($this->items as $key => $item) {
|
|
|
|
if (is_numeric($item)) {
|
|
|
|
$item = DataObject::get_by_id($this->dataClass, $item);
|
|
|
|
}
|
|
|
|
if (!empty($this->extraFields[$key])) {
|
|
|
|
$item->update($this->extraFields[$key]);
|
|
|
|
}
|
|
|
|
$items[] = $item;
|
|
|
|
}
|
|
|
|
return $items;
|
|
|
|
}
|
2012-11-03 04:16:51 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* Add a number of items to the relation.
|
|
|
|
*
|
|
|
|
* @param array $items Items to add, as either DataObjects or IDs.
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function addMany($items)
|
|
|
|
{
|
|
|
|
foreach ($items as $item) {
|
|
|
|
$this->add($item);
|
|
|
|
}
|
|
|
|
return $this;
|
|
|
|
}
|
2012-11-03 04:16:51 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
2012-11-03 04:16:51 +01:00
|
|
|
* Remove all items from this relation.
|
|
|
|
*/
|
2016-11-29 00:31:16 +01:00
|
|
|
public function removeAll()
|
|
|
|
{
|
|
|
|
$this->items = array();
|
|
|
|
$this->extraFields = array();
|
|
|
|
}
|
2012-11-03 04:16:51 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* Remove the items from this list with the given IDs
|
|
|
|
*
|
|
|
|
* @param array $items
|
|
|
|
* @param array $items
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function removeMany($items)
|
|
|
|
{
|
|
|
|
$this->items = array_diff($this->items, $items);
|
|
|
|
return $this;
|
|
|
|
}
|
2012-11-03 04:16:51 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* Removes items from this list which are equal.
|
|
|
|
*
|
|
|
|
* @param string $field unused
|
|
|
|
*/
|
|
|
|
public function removeDuplicates($field = 'ID')
|
|
|
|
{
|
|
|
|
$this->items = array_unique($this->items);
|
|
|
|
}
|
2012-11-03 04:16:51 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* Sets the Relation to be the given ID list.
|
|
|
|
* Records will be added and deleted as appropriate.
|
|
|
|
*
|
|
|
|
* @param array $idList List of IDs.
|
|
|
|
*/
|
|
|
|
public function setByIDList($idList)
|
|
|
|
{
|
|
|
|
$this->removeAll();
|
|
|
|
$this->addMany($idList);
|
|
|
|
}
|
2012-11-03 04:16:51 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* Returns an array with both the keys and values set to the IDs of the records in this list.
|
|
|
|
* Does not respect sort order. Use ->column("ID") to get an ID list with the current sort.
|
|
|
|
* Does not return the IDs for unsaved DataObjects.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function getIDList()
|
|
|
|
{
|
|
|
|
// Get a list of IDs of our current items - if it's not a number then object then assume it's a DO.
|
|
|
|
$ids = array_map(function ($obj) {
|
|
|
|
return is_numeric($obj) ? $obj : $obj->ID;
|
|
|
|
}, $this->items);
|
2013-03-08 00:33:27 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
// Strip out duplicates and anything resolving to False.
|
|
|
|
$ids = array_filter(array_unique($ids));
|
2013-03-08 00:33:27 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
// Change the array from (1, 2, 3) to (1 => 1, 2 => 2, 3 => 3)
|
|
|
|
if ($ids) {
|
|
|
|
$ids = array_combine($ids, $ids);
|
|
|
|
}
|
2013-03-08 00:33:27 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
return $ids;
|
|
|
|
}
|
2013-03-08 00:33:27 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* Returns the first item in the list
|
|
|
|
*
|
|
|
|
* @return mixed
|
|
|
|
*/
|
|
|
|
public function first()
|
|
|
|
{
|
|
|
|
$item = reset($this->items);
|
|
|
|
if (is_numeric($item)) {
|
|
|
|
$item = DataObject::get_by_id($this->dataClass, $item);
|
|
|
|
}
|
|
|
|
if (!empty($this->extraFields[key($this->items)])) {
|
|
|
|
$item->update($this->extraFields[key($this->items)]);
|
|
|
|
}
|
|
|
|
return $item;
|
|
|
|
}
|
2012-11-03 04:16:51 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* Returns the last item in the list
|
|
|
|
*
|
|
|
|
* @return mixed
|
|
|
|
*/
|
|
|
|
public function last()
|
|
|
|
{
|
|
|
|
$item = end($this->items);
|
|
|
|
if (!empty($this->extraFields[key($this->items)])) {
|
|
|
|
$item->update($this->extraFields[key($this->items)]);
|
|
|
|
}
|
|
|
|
return $item;
|
|
|
|
}
|
2012-11-03 04:16:51 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* Returns an array of a single field value for all items in the list.
|
|
|
|
*
|
|
|
|
* @param string $colName
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function column($colName = 'ID')
|
|
|
|
{
|
|
|
|
$list = new ArrayList($this->toArray());
|
|
|
|
return $list->column($colName);
|
|
|
|
}
|
2012-11-03 04:16:51 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* Returns a copy of this list with the relationship linked to the given foreign ID.
|
|
|
|
* @param int|array $id An ID or an array of IDs.
|
2017-11-09 05:08:31 +01:00
|
|
|
* @return Relation
|
2016-11-29 00:31:16 +01:00
|
|
|
*/
|
|
|
|
public function forForeignID($id)
|
|
|
|
{
|
2017-11-09 05:08:31 +01:00
|
|
|
$singleton = DataObject::singleton($this->baseClass);
|
|
|
|
/** @var Relation $relation */
|
|
|
|
$relation = $singleton->{$this->relationName}($id);
|
|
|
|
return $relation;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $relationName
|
|
|
|
* @return Relation
|
|
|
|
*/
|
|
|
|
public function relation($relationName)
|
|
|
|
{
|
|
|
|
$ids = $this->column('ID');
|
|
|
|
$singleton = DataObject::singleton($this->dataClass);
|
|
|
|
/** @var Relation $relation */
|
|
|
|
$relation = $singleton->$relationName($ids);
|
|
|
|
return $relation;
|
2016-11-29 00:31:16 +01:00
|
|
|
}
|
2012-11-03 04:16:51 +01:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* Return the DBField object that represents the given field on the related class.
|
|
|
|
*
|
|
|
|
* @param string $fieldName Name of the field
|
|
|
|
* @return DBField The field as a DBField object
|
|
|
|
*/
|
|
|
|
public function dbObject($fieldName)
|
|
|
|
{
|
2017-11-09 05:08:31 +01:00
|
|
|
return DataObject::singleton($this->dataClass)->dbObject($fieldName);
|
2016-11-29 00:31:16 +01:00
|
|
|
}
|
2016-03-30 23:37:12 +02:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
protected function extractValue($item, $key)
|
|
|
|
{
|
|
|
|
if (is_numeric($item)) {
|
|
|
|
$item = DataObject::get_by_id($this->dataClass, $item);
|
|
|
|
}
|
|
|
|
return parent::extractValue($item, $key);
|
|
|
|
}
|
2012-11-03 04:16:51 +01:00
|
|
|
}
|