274 lines
7.8 KiB
PHP
274 lines
7.8 KiB
PHP
<?php
|
|
// +----------------------------------------------------------------------
|
|
// | ThinkPHP [ WE CAN DO IT JUST THINK ]
|
|
// +----------------------------------------------------------------------
|
|
// | Copyright (c) 2006~2018 http://thinkphp.cn All rights reserved.
|
|
// +----------------------------------------------------------------------
|
|
// | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 )
|
|
// +----------------------------------------------------------------------
|
|
// | Author: liu21st <liu21st@gmail.com>
|
|
// +----------------------------------------------------------------------
|
|
|
|
namespace think\model\concern;
|
|
|
|
use think\Collection;
|
|
use think\Exception;
|
|
use think\Loader;
|
|
use think\Model;
|
|
use think\model\Collection as ModelCollection;
|
|
|
|
/**
|
|
* 模型数据转换处理
|
|
*/
|
|
trait Conversion
|
|
{
|
|
/**
|
|
* 数据输出显示的属性
|
|
* @var array
|
|
*/
|
|
protected $visible = [];
|
|
|
|
/**
|
|
* 数据输出隐藏的属性
|
|
* @var array
|
|
*/
|
|
protected $hidden = [];
|
|
|
|
/**
|
|
* 数据输出需要追加的属性
|
|
* @var array
|
|
*/
|
|
protected $append = [];
|
|
|
|
/**
|
|
* 数据集对象名
|
|
* @var string
|
|
*/
|
|
protected $resultSetType;
|
|
|
|
/**
|
|
* 设置需要附加的输出属性
|
|
* @access public
|
|
* @param array $append 属性列表
|
|
* @param bool $override 是否覆盖
|
|
* @return $this
|
|
*/
|
|
public function append(array $append = [], $override = false)
|
|
{
|
|
$this->append = $override ? $append : array_merge($this->append, $append);
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* 设置附加关联对象的属性
|
|
* @access public
|
|
* @param string $attr 关联属性
|
|
* @param string|array $append 追加属性名
|
|
* @return $this
|
|
* @throws Exception
|
|
*/
|
|
public function appendRelationAttr($attr, $append)
|
|
{
|
|
if (is_string($append)) {
|
|
$append = explode(',', $append);
|
|
}
|
|
|
|
$relation = Loader::parseName($attr, 1, false);
|
|
if (isset($this->relation[$relation])) {
|
|
$model = $this->relation[$relation];
|
|
} else {
|
|
$model = $this->getRelationData($this->$relation());
|
|
}
|
|
|
|
if ($model instanceof Model) {
|
|
foreach ($append as $key => $attr) {
|
|
$key = is_numeric($key) ? $attr : $key;
|
|
if (isset($this->data[$key])) {
|
|
throw new Exception('bind attr has exists:' . $key);
|
|
} else {
|
|
$this->data[$key] = $model->$attr;
|
|
}
|
|
}
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* 设置需要隐藏的输出属性
|
|
* @access public
|
|
* @param array $hidden 属性列表
|
|
* @param bool $override 是否覆盖
|
|
* @return $this
|
|
*/
|
|
public function hidden(array $hidden = [], $override = false)
|
|
{
|
|
$this->hidden = $override ? $hidden : array_merge($this->hidden, $hidden);
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* 设置需要输出的属性
|
|
* @access public
|
|
* @param array $visible
|
|
* @param bool $override 是否覆盖
|
|
* @return $this
|
|
*/
|
|
public function visible(array $visible = [], $override = false)
|
|
{
|
|
$this->visible = $override ? $visible : array_merge($this->visible, $visible);
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* 转换当前模型对象为数组
|
|
* @access public
|
|
* @return array
|
|
*/
|
|
public function toArray()
|
|
{
|
|
$item = [];
|
|
$hasVisible = false;
|
|
|
|
foreach ($this->visible as $key => $val) {
|
|
if (is_string($val)) {
|
|
if (strpos($val, '.')) {
|
|
list($relation, $name) = explode('.', $val);
|
|
$this->visible[$relation][] = $name;
|
|
} else {
|
|
$this->visible[$val] = true;
|
|
$hasVisible = true;
|
|
}
|
|
unset($this->visible[$key]);
|
|
}
|
|
}
|
|
|
|
foreach ($this->hidden as $key => $val) {
|
|
if (is_string($val)) {
|
|
if (strpos($val, '.')) {
|
|
list($relation, $name) = explode('.', $val);
|
|
$this->hidden[$relation][] = $name;
|
|
} else {
|
|
$this->hidden[$val] = true;
|
|
}
|
|
unset($this->hidden[$key]);
|
|
}
|
|
}
|
|
|
|
// 合并关联数据
|
|
$data = array_merge($this->data, $this->relation);
|
|
|
|
foreach ($data as $key => $val) {
|
|
if ($val instanceof Model || $val instanceof ModelCollection) {
|
|
// 关联模型对象
|
|
if (isset($this->visible[$key]) && is_array($this->visible[$key])) {
|
|
$val->visible($this->visible[$key]);
|
|
} elseif (isset($this->hidden[$key]) && is_array($this->hidden[$key])) {
|
|
$val->hidden($this->hidden[$key]);
|
|
}
|
|
// 关联模型对象
|
|
if (!isset($this->hidden[$key]) || true !== $this->hidden[$key]) {
|
|
$item[$key] = $val->toArray();
|
|
}
|
|
} elseif (isset($this->visible[$key])) {
|
|
$item[$key] = $this->getAttr($key);
|
|
} elseif (!isset($this->hidden[$key]) && !$hasVisible) {
|
|
$item[$key] = $this->getAttr($key);
|
|
}
|
|
}
|
|
|
|
// 追加属性(必须定义获取器)
|
|
if (!empty($this->append)) {
|
|
foreach ($this->append as $key => $name) {
|
|
if (is_array($name)) {
|
|
// 追加关联对象属性
|
|
$relation = $this->getRelation($key);
|
|
|
|
if (!$relation) {
|
|
$relation = $this->getAttr($key);
|
|
if ($relation) {
|
|
$relation->visible($name);
|
|
}
|
|
}
|
|
|
|
$item[$key] = $relation ? $relation->append($name)->toArray() : [];
|
|
} elseif (strpos($name, '.')) {
|
|
list($key, $attr) = explode('.', $name);
|
|
// 追加关联对象属性
|
|
$relation = $this->getRelation($key);
|
|
|
|
if (!$relation) {
|
|
$relation = $this->getAttr($key);
|
|
if ($relation) {
|
|
$relation->visible([$attr]);
|
|
}
|
|
}
|
|
|
|
$item[$key] = $relation ? $relation->append([$attr])->toArray() : [];
|
|
} else {
|
|
$item[$name] = $this->getAttr($name, $item);
|
|
}
|
|
}
|
|
}
|
|
|
|
return $item;
|
|
}
|
|
|
|
/**
|
|
* 转换当前模型对象为JSON字符串
|
|
* @access public
|
|
* @param integer $options json参数
|
|
* @return string
|
|
*/
|
|
public function toJson($options = JSON_UNESCAPED_UNICODE)
|
|
{
|
|
return json_encode($this->toArray(), $options);
|
|
}
|
|
|
|
/**
|
|
* 移除当前模型的关联属性
|
|
* @access public
|
|
* @return $this
|
|
*/
|
|
public function removeRelation()
|
|
{
|
|
$this->relation = [];
|
|
return $this;
|
|
}
|
|
|
|
public function __toString()
|
|
{
|
|
return $this->toJson();
|
|
}
|
|
|
|
// JsonSerializable
|
|
public function jsonSerialize()
|
|
{
|
|
return $this->toArray();
|
|
}
|
|
|
|
/**
|
|
* 转换数据集为数据集对象
|
|
* @access public
|
|
* @param array|Collection $collection 数据集
|
|
* @param string $resultSetType 数据集类
|
|
* @return Collection
|
|
*/
|
|
public function toCollection($collection, $resultSetType = null)
|
|
{
|
|
$resultSetType = $resultSetType ?: $this->resultSetType;
|
|
|
|
if ($resultSetType && false !== strpos($resultSetType, '\\')) {
|
|
$collection = new $resultSetType($collection);
|
|
} else {
|
|
$collection = new ModelCollection($collection);
|
|
}
|
|
|
|
return $collection;
|
|
}
|
|
|
|
}
|