mirror of
https://github.com/alchemy-fr/Phraseanet.git
synced 2025-10-24 10:23:17 +00:00
Add RecordCollection class to avoid multiple record_adapter fetching
This commit is contained in:
41
lib/Alchemy/Phrasea/Databox/DataboxGroupable.php
Normal file
41
lib/Alchemy/Phrasea/Databox/DataboxGroupable.php
Normal file
@@ -0,0 +1,41 @@
|
||||
<?php
|
||||
/*
|
||||
* This file is part of Phraseanet
|
||||
*
|
||||
* (c) 2005-2016 Alchemy
|
||||
*
|
||||
* For the full copyright and license information, please view the LICENSE
|
||||
* file that was distributed with this source code.
|
||||
*/
|
||||
|
||||
namespace Alchemy\Phrasea\Databox;
|
||||
|
||||
interface DataboxGroupable
|
||||
{
|
||||
/**
|
||||
* Group instance by Databox Id
|
||||
*
|
||||
* @return array<int,array>
|
||||
*/
|
||||
public function groupByDatabox();
|
||||
|
||||
/**
|
||||
* Returns databoxes ids
|
||||
*
|
||||
* @return int[]
|
||||
*/
|
||||
public function getDataboxIds();
|
||||
|
||||
/**
|
||||
* @param int $databoxId
|
||||
* @return array
|
||||
*/
|
||||
public function getDataboxGroup($databoxId);
|
||||
|
||||
/**
|
||||
* Reorder groups if needed
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function reorderGroups();
|
||||
}
|
20
lib/Alchemy/Phrasea/Record/PerDataboxRecordId.php
Normal file
20
lib/Alchemy/Phrasea/Record/PerDataboxRecordId.php
Normal file
@@ -0,0 +1,20 @@
|
||||
<?php
|
||||
/*
|
||||
* This file is part of Phraseanet
|
||||
*
|
||||
* (c) 2005-2016 Alchemy
|
||||
*
|
||||
* For the full copyright and license information, please view the LICENSE
|
||||
* file that was distributed with this source code.
|
||||
*/
|
||||
|
||||
namespace Alchemy\Phrasea\Record;
|
||||
|
||||
interface PerDataboxRecordId
|
||||
{
|
||||
/**
|
||||
* @param int $databoxId
|
||||
* @return int[]
|
||||
*/
|
||||
public function getDataboxRecordIds($databoxId);
|
||||
}
|
218
lib/Alchemy/Phrasea/Record/RecordCollection.php
Normal file
218
lib/Alchemy/Phrasea/Record/RecordCollection.php
Normal file
@@ -0,0 +1,218 @@
|
||||
<?php
|
||||
/*
|
||||
* This file is part of Phraseanet
|
||||
*
|
||||
* (c) 2005-2016 Alchemy
|
||||
*
|
||||
* For the full copyright and license information, please view the LICENSE
|
||||
* file that was distributed with this source code.
|
||||
*/
|
||||
|
||||
namespace Alchemy\Phrasea\Record;
|
||||
|
||||
use Alchemy\Phrasea\Databox\DataboxGroupable;
|
||||
use Assert\Assertion;
|
||||
|
||||
class RecordCollection implements \IteratorAggregate, \ArrayAccess, \Countable, DataboxGroupable, PerDataboxRecordId
|
||||
{
|
||||
/**
|
||||
* @var \record_adapter[]
|
||||
*/
|
||||
private $records = [];
|
||||
|
||||
/**
|
||||
* @var array<int, int|string>
|
||||
*/
|
||||
private $groups;
|
||||
|
||||
/**
|
||||
* @var bool
|
||||
*/
|
||||
private $reorderNeeded = false;
|
||||
|
||||
public function __construct($records = [])
|
||||
{
|
||||
Assertion::allIsInstanceOf($records, \record_adapter::class);
|
||||
|
||||
foreach ($records as $index => $record) {
|
||||
$this->add($record, $index);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @param \record_adapter $record
|
||||
* @param null|int|string $index
|
||||
* @return void
|
||||
*/
|
||||
public function add(\record_adapter $record, $index = null)
|
||||
{
|
||||
if (null === $index) {
|
||||
$this->addWithUnknownIndex($record);
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
if (isset($this->records[$index])){
|
||||
unset($this->groups[$this->records[$index]->getDataboxId()][$index]);
|
||||
$this->reorderNeeded = true;
|
||||
}
|
||||
|
||||
$this->records[$index] = $record;
|
||||
|
||||
$this->addIndexToGroups($record, $index);
|
||||
}
|
||||
|
||||
/**
|
||||
* @return \ArrayIterator
|
||||
*/
|
||||
public function getIterator()
|
||||
{
|
||||
return new \ArrayIterator($this->records);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param int|string $offset
|
||||
* @return bool
|
||||
*/
|
||||
public function offsetExists($offset)
|
||||
{
|
||||
return isset($this->records[$offset]);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param int|string $offset
|
||||
* @return \record_adapter
|
||||
*/
|
||||
public function offsetGet($offset)
|
||||
{
|
||||
return $this->records[$offset];
|
||||
}
|
||||
|
||||
/**
|
||||
* @param int|string $offset
|
||||
* @param \record_adapter $value
|
||||
*/
|
||||
public function offsetSet($offset, $value)
|
||||
{
|
||||
Assertion::isInstanceOf($value, \record_adapter::class);
|
||||
|
||||
$this->add($value, $offset);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param int|string $offset
|
||||
* @return void
|
||||
*/
|
||||
public function offsetUnset($offset)
|
||||
{
|
||||
if (isset($this->records[$offset])) {
|
||||
unset($this->groups[$this->records[$offset]->getDataboxId()][$offset]);
|
||||
}
|
||||
|
||||
unset($this->records[$offset]);
|
||||
}
|
||||
|
||||
/**
|
||||
* @return int
|
||||
*/
|
||||
public function count()
|
||||
{
|
||||
return count($this->records);
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns records groups by databoxId (possibly not in order)
|
||||
*
|
||||
* @return \record_adapter[][]
|
||||
*/
|
||||
public function groupByDatabox()
|
||||
{
|
||||
return $this->groups;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return int[]
|
||||
*/
|
||||
public function getDataboxIds()
|
||||
{
|
||||
return array_keys($this->groups);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param int $databoxId
|
||||
* @return \record_adapter[]
|
||||
*/
|
||||
public function getDataboxGroup($databoxId)
|
||||
{
|
||||
return isset($this->groups[$databoxId]) ? $this->groups[$databoxId] : [];
|
||||
}
|
||||
|
||||
/**
|
||||
* @return void
|
||||
*/
|
||||
public function reorderGroups()
|
||||
{
|
||||
if (!$this->reorderNeeded) {
|
||||
return;
|
||||
}
|
||||
|
||||
$groups = [];
|
||||
|
||||
foreach ($this->records as $index => $record) {
|
||||
$databoxId = $record->getDataboxId();
|
||||
|
||||
if (!isset($groups[$databoxId])) {
|
||||
$groups[$databoxId] = [];
|
||||
}
|
||||
|
||||
$groups[$databoxId][$index] = $record;
|
||||
}
|
||||
|
||||
$this->groups = $groups;
|
||||
$this->reorderNeeded = false;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param int $databoxId
|
||||
* @return int[]
|
||||
*/
|
||||
public function getDataboxRecordIds($databoxId)
|
||||
{
|
||||
$recordIds = [];
|
||||
|
||||
foreach ($this->getDataboxGroup($databoxId) as $record) {
|
||||
$recordIds[$record->getRecordId()] = true;
|
||||
}
|
||||
|
||||
return array_keys($recordIds);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param \record_adapter $record
|
||||
* @return void
|
||||
*/
|
||||
private function addWithUnknownIndex(\record_adapter $record)
|
||||
{
|
||||
$this->records[] = $record;
|
||||
|
||||
end($this->records);
|
||||
|
||||
$this->addIndexToGroups($record, key($this->records));
|
||||
}
|
||||
|
||||
/**
|
||||
* @param \record_adapter $record
|
||||
* @param int|string $index
|
||||
* @return void
|
||||
*/
|
||||
private function addIndexToGroups(\record_adapter $record, $index)
|
||||
{
|
||||
$databoxId = $record->getDataboxId();
|
||||
|
||||
if (!isset($this->groups[$databoxId])) {
|
||||
$this->groups[$databoxId] = [];
|
||||
}
|
||||
|
||||
$this->groups[$databoxId][$index] = $record;
|
||||
}
|
||||
}
|
@@ -10,10 +10,11 @@
|
||||
|
||||
namespace Alchemy\Phrasea\Record;
|
||||
|
||||
use Alchemy\Phrasea\Databox\DataboxGroupable;
|
||||
use Alchemy\Phrasea\Model\RecordReferenceInterface;
|
||||
use Assert\Assertion;
|
||||
|
||||
class RecordReferenceCollection implements \IteratorAggregate, \ArrayAccess
|
||||
class RecordReferenceCollection implements \IteratorAggregate, \ArrayAccess, \Countable, DataboxGroupable, PerDataboxRecordId
|
||||
{
|
||||
/**
|
||||
* @param array<int|string,array> $records
|
||||
@@ -192,13 +193,17 @@ class RecordReferenceCollection implements \IteratorAggregate, \ArrayAccess
|
||||
return $records;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param int|string $offset
|
||||
* @return bool
|
||||
*/
|
||||
public function offsetExists($offset)
|
||||
{
|
||||
return isset($this->references[$offset]);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param mixed $offset
|
||||
* @param int|string $offset
|
||||
* @return RecordReferenceInterface
|
||||
*/
|
||||
public function offsetGet($offset)
|
||||
@@ -206,6 +211,10 @@ class RecordReferenceCollection implements \IteratorAggregate, \ArrayAccess
|
||||
return $this->references[$offset];
|
||||
}
|
||||
|
||||
/**
|
||||
* @param int|string $offset
|
||||
* @param RecordReferenceInterface $value
|
||||
*/
|
||||
public function offsetSet($offset, $value)
|
||||
{
|
||||
Assertion::isInstanceOf($value, RecordReferenceInterface::class);
|
||||
@@ -213,9 +222,69 @@ class RecordReferenceCollection implements \IteratorAggregate, \ArrayAccess
|
||||
$this->add($value, $offset);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param int|string $offset
|
||||
* @return void
|
||||
*/
|
||||
public function offsetUnset($offset)
|
||||
{
|
||||
unset($this->references[$offset]);
|
||||
$this->groups = null;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return RecordReferenceInterface[][]
|
||||
*/
|
||||
public function groupByDatabox()
|
||||
{
|
||||
$this->reorderGroups();
|
||||
|
||||
return $this->groups;
|
||||
}
|
||||
|
||||
public function reorderGroups()
|
||||
{
|
||||
if ($this->groups) {
|
||||
return;
|
||||
}
|
||||
|
||||
$groups = [];
|
||||
|
||||
foreach ($this->references as $index => $reference) {
|
||||
if (!isset($groups[$reference->getDataboxId()])) {
|
||||
$groups[$reference->getDataboxId()] = [];
|
||||
}
|
||||
|
||||
$groups[$reference->getDataboxId()][$index] = $reference;
|
||||
}
|
||||
|
||||
$this->groups = $groups;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param int $databoxId
|
||||
* @return RecordReferenceInterface[]
|
||||
*/
|
||||
public function getDataboxGroup($databoxId)
|
||||
{
|
||||
$this->reorderGroups();
|
||||
|
||||
return isset($this->groups[$databoxId]) ? $this->groups[$databoxId] : [];
|
||||
}
|
||||
|
||||
public function getDataboxRecordIds($databoxId)
|
||||
{
|
||||
$recordsIds = [];
|
||||
|
||||
foreach ($this->getDataboxGroup($databoxId) as $references) {
|
||||
$recordsIds[$references->getRecordId()] = true;
|
||||
}
|
||||
|
||||
return array_keys($recordsIds);
|
||||
}
|
||||
|
||||
public function count()
|
||||
{
|
||||
return count($this->references);
|
||||
}
|
||||
}
|
||||
|
Reference in New Issue
Block a user