我正在尝试为我正在从事的项目编写元行为,该行为将允许我将自定义变量/属性分配给模型,类似于在 WordPress 帖子中创建自定义字段的方式。
我创建了一个元行为,它将元模型绑定(bind)到它所作用的模型,并且还有一个 beforeSave 回调,该回调循环遍历模型数据变量并将模型名称放入元数组中。
一切都在保存,但是当我检查数据库时,模型字段返回为空。
Meta 的数据库结构是
id - A unique if for the meta
model - The name of the model that this meta entry is associated with
foreign_id - The id of the above model that this meta entry is associated with
key - Name of the meta variable
value - Value of the meta variable
从表单传入 saveAll 函数的数据是
Array
(
[Page] => Array
(
[id] => 12
[name] => Test
[slug] => a/b/c/d
[layout] => 0
[body] => Test multilevel
)
[Meta] => Array
(
[0] => Array
(
[id] => 1
[key] => page_title
[value] => About Us
)
[1] => Array
(
[id] => 6
[key] => test4
[value] => test
)
[2] => Array
(
[key] => test3
[value] => lala
)
)
)
在它运行完 beforeSave 的行为之后,它是
Array
(
[Page] => Array
(
[id] => 12
[name] => Test
[slug] => a/b/c/d
[layout] => 0
[body] => Test multilevel
[modified] => 2010-05-04 15:56:54
)
[Meta] => Array
(
[0] => Array
(
[id] => 1
[key] => page_title
[value] => About Us
[model] => Page
)
[1] => Array
(
[id] => 6
[key] => test4
[value] => test
[model] => Page
)
[2] => Array
(
[key] => test3
[value] => lala
[model] => Page
)
)
)
行为代码是
<?php
/**
* Meta Model Behavior
*
* Adds custom variables to models
*
**/
class MetaBehavior extends ModelBehavior {
/**
* Contains configuration settings for use with individual model objects.
* Individual model settings should be stored as an associative array,
* keyed off of the model name.
*
* @var array
* @access public
* @see Model::$alias
*/
var $__settings = array();
var $__defaults = array(
'class' => 'Meta',
'foreign_key' => 'foreign_id',
'dependent' => true,
'auto_bind' => true
);
/**
* Initiate Meta Behavior
*
* @param object $model
* @param array $config
* @return void
* @access public
*/
function setup(&$model, $settings = array()) {
$default = $this->__defaults;
$default['conditions'] = array('Meta.model' => $model->alias);
if (!isset($this->__settings[$model->alias])) {
$this->__settings[$model->alias] = $default;
}
$this->__settings[$model->alias] = array_merge($this->__settings[$model->alias], ife(is_array($settings), $settings, array()));
if ($this->__settings[$model->alias]['auto_bind']) {
$hasManyMeta = array(
'Meta' => array(
'className' => $this->__settings[$model->alias]['class'],
'foreignKey' => $this->__settings[$model->alias]['foreign_key'],
'dependent' => $this->__settings[$model->alias]['dependent'],
'conditions' => $this->__settings[$model->alias]['conditions']
)
);
$metaBelongsTo = array(
$model->alias => array(
'className' => $model->alias,
'foreignKey' => $this->__settings[$model->alias]['foreign_key']
)
);
$model->bindModel(array('hasMany' => $hasManyMeta), false);
$model->Meta->bindModel(array('belongsTo' => $metaBelongsTo), false);
}
}
function beforeSave(&$model) {
foreach($model->data[$this->__settings['class']] as $key => $value) {
$model->data[$this->__settings['class']][$key]['model'] = $model->alias;
}
return true;
}
} // End of MetaBehavior
?>
我有一种感觉,这可能是因为关系的原因,并且 saveall 在保存关联时使用了传递的数据(原始)。
我想到的唯一其他方法是删除关系并将一些代码放入行为的 afterSave 函数中以处理保存,然后将一些其他代码放入 afterFind 中以检索它们。
有什么想法吗?
干杯, 院长
最佳答案
我不相信你会对此有任何运气,因为 Model::saveAll() 根本不调用 beforeSave() 。事实上,它在调用 $this->save() 之前加载关联。
Line 1652模型源的显示在任何真正调用 Model::__save() 之前加载的关联,这不仅仅是验证数据。
这意味着,乍一看,MetaBehavior 中神奇的 bindModel() 调用似乎不会产生任何影响。老实说,我真的不会为类似的事情烦恼,因为在页面模型类定义中设置关联更简单。
当然,你可以:
- 在您的 MetaBehavior 中创建一个 afterSave,以方便保存元模型数据,或者,
- 在模型中重写 saveAll() 以添加绑定(bind),然后调用parent::saveAll()
最终,我觉得使用 Model::saveAll() 并在模型类定义中设置真正的关联更为明智。利用行为来促进 DRY 实践是非常崇高的,但这并不意味着您需要让事情变得不那么琐碎,作为您或其他人遵循的开发方法的一部分
TLDR;您已将 Meta
添加到行为数组中,添加关联只需在同一文件中添加几行代码即可。
关于php - 使用saveAll时修改Behavior的beforeSave中的Model数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2763182/