mirror of
https://github.com/processwire/processwire.git
synced 2025-08-13 18:24:57 +02:00
Initial commit to new repo (carried over from: https://github.com/ryancramerdesign/ProcessWire/tree/devns)
This commit is contained in:
169
wire/core/WireSaveableItemsLookup.php
Normal file
169
wire/core/WireSaveableItemsLookup.php
Normal file
@@ -0,0 +1,169 @@
|
||||
<?php namespace ProcessWire;
|
||||
|
||||
/**
|
||||
* ProcessWire WireSaveableItemsLookup
|
||||
*
|
||||
* Provides same functionality as WireSaveableItems except that this class includes joining/modification of a related lookup table
|
||||
*
|
||||
* ProcessWire 3.x, Copyright 2016 by Ryan Cramer
|
||||
* https://processwire.com
|
||||
*
|
||||
*/
|
||||
|
||||
abstract class WireSaveableItemsLookup extends WireSaveableItems {
|
||||
|
||||
/**
|
||||
* If a lookup table should be left joined, this method should return the table name
|
||||
*
|
||||
*/
|
||||
abstract public function getLookupTable();
|
||||
|
||||
/**
|
||||
* If a lookup table should be left joined, this method returns the name of the array field in $data that contains multiple values
|
||||
*
|
||||
* i.e. roles_permissions becomes permissions_id if getTable() returns roles
|
||||
* Does not need to be overridden unless the table naming structure doesn't follow existing logic.
|
||||
*
|
||||
*/
|
||||
public function getLookupField() {
|
||||
$lookupTable = $this->getLookupTable();
|
||||
if(!$lookupTable) return '';
|
||||
return preg_replace('/_?' . $this->getTable() . '_?/', '', $lookupTable) . '_id';
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the DatabaseQuerySelect to perform the load operation of items
|
||||
*
|
||||
* @param Selectors|string|null $selectors Selectors or a selector string to find, or NULL to load all.
|
||||
* @return DatabaseQuerySelect
|
||||
*
|
||||
*/
|
||||
protected function getLoadQuery($selectors = null) {
|
||||
$query = parent::getLoadQuery($selectors);
|
||||
$database = $this->wire('database');
|
||||
$table = $database->escapeTable($this->getTable());
|
||||
$lookupTable = $database->escapeTable($this->getLookupTable());
|
||||
$lookupField = $database->escapeCol($this->getLookupField());
|
||||
$query->select("$lookupTable.$lookupField"); // QA
|
||||
$query->leftjoin("$lookupTable ON $lookupTable.{$table}_id=$table.id ")->orderby("sort"); // QA
|
||||
return $query;
|
||||
}
|
||||
|
||||
/**
|
||||
* Load items from the database table and return them in the same type class that getAll() returns
|
||||
*
|
||||
* A selector string or Selectors may be provided so that this can be used as a find() by descending classes that don't load all items at once.
|
||||
*
|
||||
* @param WireArray $items
|
||||
* @param Selectors|string|null $selectors Selectors or a selector string to find, or NULL to load all.
|
||||
* @return WireArray Returns the same type as specified in the getAll() method.
|
||||
*
|
||||
*/
|
||||
protected function ___load(WireArray $items, $selectors = null) {
|
||||
|
||||
$query = $this->getLoadQuery($selectors);
|
||||
$database = $this->wire('database');
|
||||
$sql = $query->getQuery();
|
||||
$stmt = $database->prepare($sql);
|
||||
$stmt->execute();
|
||||
$lookupField = $this->getLookupField();
|
||||
|
||||
while($row = $stmt->fetch(\PDO::FETCH_ASSOC)) {
|
||||
|
||||
$item = $this->makeBlankItem();
|
||||
$lookupValue = $row[$lookupField];
|
||||
unset($row[$lookupField]);
|
||||
$item->addLookupItem($lookupValue, $row);
|
||||
|
||||
foreach($row as $field => $value) {
|
||||
$item->$field = $value;
|
||||
}
|
||||
|
||||
if($items->has($item)) {
|
||||
// LEFT JOIN is adding more elements of the same item, i.e. from lookup table
|
||||
// if the item is already present in $items, then use the existing one rather
|
||||
// and throw out the one we just created
|
||||
$item = $items->get($item);
|
||||
$item->addLookupItem($lookupValue, $row);
|
||||
} else {
|
||||
// add a new item
|
||||
$items->add($item);
|
||||
}
|
||||
}
|
||||
|
||||
$stmt->closeCursor();
|
||||
$items->setTrackChanges(true);
|
||||
return $items;
|
||||
}
|
||||
|
||||
/**
|
||||
* Should the given item key/field be saved in the database?
|
||||
*
|
||||
* Template method used by ___save()
|
||||
*
|
||||
*/
|
||||
protected function saveItemKey($key) {
|
||||
if($key == $this->getLookupField()) return false;
|
||||
return parent::saveItemKey($key);
|
||||
}
|
||||
|
||||
/**
|
||||
* Save the provided item to database
|
||||
*
|
||||
* @param Saveable $item
|
||||
* @return bool
|
||||
* @throws WireException
|
||||
*
|
||||
*/
|
||||
public function ___save(Saveable $item) {
|
||||
|
||||
if(!$item instanceof HasLookupItems) throw new WireException($this->className() . "::save() requires an item that implements HasLookupItems interface");
|
||||
|
||||
$database = $this->wire('database');
|
||||
$lookupTable = $database->escapeTable($this->getLookupTable());
|
||||
$lookupField = $database->escapeCol($this->getLookupField());
|
||||
$table = $database->escapeTable($this->getTable());
|
||||
$item_id = (int) $item->id;
|
||||
|
||||
if($item_id) {
|
||||
$query = $database->prepare("DELETE FROM $lookupTable WHERE {$table}_id=:item_id");
|
||||
$query->bindValue(":item_id", $item_id, \PDO::PARAM_INT);
|
||||
$query->execute();
|
||||
}
|
||||
|
||||
$result = parent::___save($item);
|
||||
$item_id = (int) $item->id; // reload, in case it was 0 before
|
||||
|
||||
$sort = 0;
|
||||
if($item_id) foreach($item->getLookupItems() as $key => $value) {
|
||||
$value_id = (int) $value->id;
|
||||
$query = $database->prepare("INSERT INTO $lookupTable SET {$table}_id=:item_id, $lookupField=:value_id, sort=:sort");
|
||||
$query->bindValue(":item_id", $item_id);
|
||||
$query->bindValue(":value_id", $value_id);
|
||||
$query->bindValue(":sort", $sort);
|
||||
$query->execute();
|
||||
$this->resetTrackChanges();
|
||||
$sort++;
|
||||
}
|
||||
|
||||
return $result;
|
||||
}
|
||||
|
||||
/**
|
||||
* Delete the provided item from the database
|
||||
*
|
||||
* @param Saveable $item
|
||||
* @return bool
|
||||
*
|
||||
*/
|
||||
public function ___delete(Saveable $item) {
|
||||
$database = $this->wire('database');
|
||||
$lookupTable = $database->escapeTable($this->getLookupTable());
|
||||
$table = $database->escapeTable($this->getTable());
|
||||
$item_id = (int) $item->id;
|
||||
$query = $database->prepare("DELETE FROM $lookupTable WHERE {$table}_id=:item_id"); // QA
|
||||
$query->bindValue(":item_id", $item_id, \PDO::PARAM_INT);
|
||||
$query->execute();
|
||||
return parent::___delete($item);
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user