我正在设置一个表单,用于更新person
和相关的person_to_role
表,person_to_role
是一个中间表,可将person
与role
相关联n..n关系。 role
有一个预定义的角色列表,不应该从某个人的范围进行修改。
我只需更新role_id
表格中的description
和person_to_role
,然后添加/删除记录。
--person
+-------------+-------------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+-------------+-------------+------+-----+---------+----------------+
| ID | int(11) | NO | PRI | NULL | auto_increment |
| NICK_NAME | varchar(16) | YES | | NULL | |
| FIRST_NAME | varchar(24) | NO | | NULL | |
| MIDDLE_NAME | varchar(8) | YES | | NULL | |
| LAST_NAME | varchar(24) | NO | | NULL | |
| BIRTH_DATE | date | NO | | NULL | |
| GENDER | varchar(8) | NO | | NULL | |
+-------------+-------------+------+-----+---------+----------------+
-- role
+-------------+--------------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+-------------+--------------+------+-----+---------+----------------+
| ID | int(11) | NO | PRI | NULL | auto_increment |
| NAME | varchar(24) | NO | | NULL | |
| DESCRIPTION | varchar(100) | YES | | NULL | |
+-------------+--------------+------+-----+---------+----------------+
--person_to_role
+-------------+--------------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+-------------+--------------+------+-----+---------+----------------+
| ID | int(11) | NO | PRI | NULL | auto_increment |
| PERSON_ID | int(11) | NO | MUL | NULL | |
| ROLE_ID | int(11) | NO | MUL | NULL | |
| DESCRIPTION | varchar(100) | YES | | NULL | |
+-------------+--------------+------+-----+---------+----------------+
public function initialize(array $ config) { 父::初始化($配置);
$this->setTable('person');
$this->setDisplayField('ID');
$this->setPrimaryKey('ID');
// define relations
$this->belongsToMany('Role', [
'joinTable'=> 'person_to_role',
'foreignKey' => 'PERSON_ID'
]);
}
public function edit($id = null)
{
$person = $this->Person->get($id, [
'contain' => ['role']
]);
$this->set('roles', $this->Person->Role->find('list'));
if ($this->request->is(['patch', 'post', 'put'])) {
$person = $this->Person->patchEntity($person, $this->request->getData());
if ($this->Person->save($person)) {
$this->Flash->success(__('The person has been saved.'));
return $this->redirect(['action' => 'index']);
}
$errors = print_r($person->errors(),1);
$this->Flash->error(__('The person could not be saved. Please, try again.<br><pre>'. $errors .'</pre>'),['escape'=> false]);
}
$this->set(compact('person'));
$this->set('_serialize', ['person']);
}
echo $this->Form->control('NICK_NAME');
echo $this->Form->control('FIRST_NAME');
echo $this->Form->control('MIDDLE_NAME');
echo $this->Form->control('LAST_NAME');
echo $this->Form->control('BIRTH_DATE');
echo $this->Form->control('GENDER',['type'=>'select','options'=>[''=> '-
Please Select -', 'Male'=>'Male','Female'=>'Female']]);
Assigned roles
...
<?php foreach ($person->role as $k=>$role) { ?>
<tr><td><?php
echo $this->Form->input("role.$k._joinData.ID",['type'=>'hidden','value'=>$person->role[$k]['_joinData']['ID']]);
echo $this->Form->input("role.$k._joinData.ROLE_ID",['value'=>$role-
>ID,'options'=>$roles,'templates'=>['formGroup' =>'{{input}}']])
?></td><td>
<?php echo $this->Form->input("role.$k._joinData.DESCRIPTION", ['value'=>$person->role[$k]['_joinData']['DESCRIPTION'],'templates'=>
['formGroup' =>'{{input}}']]); ?></td></tr>
<?php } ?>
...
这给了我以下POST数据:
'NICK_NAME' => 'Johny',
'FIRST_NAME' => 'John',
'MIDDLE_NAME' => 'J.',
'LAST_NAME' => 'Smith',
'BIRTH_DATE' => '1961-01-01',
'GENDER' => 'Male',
'role' => [
(int) 0 => [
'_joinData' => [
'ID' => '1',
'ROLE_ID' => '5',
'DESCRIPTION' => 'person role description'
]
]
]
]
但更新失败并出现以下错误:
该人无法得救。请再试一次。
Array
(
[role] => Array
(
[0] => Array
(
[NAME] => Array
(
[_required] => This field is required
)
[_joinData] => Array
(
[PERSON_ID] => Array
(
[_required] => This field is required
)
)
)
)
)
它要求提供person_to_role.person_id
哪个是foreignKey(它应该知道当前的人ID),并且需要role.name
的值。
我的关联设置错了吗?任何帮助表示赞赏。
仍然没有去,尝试了所有可能的文档和其他Internet资源的变化。目前,我可以将验证传递给save
操作,但会生成INSERT
个查询而不是UPDATE
,并且会在唯一约束违规时出错。
我可以访问person.ID
和person_to_role.ID
:
protected $_accessible = [
'*' => true,
'ID' => true
];
我的POST数据如下所示:
[
'NICK_NAME' => '',
'FIRST_NAME' => 'test',
'MIDDLE_NAME' => '',
'LAST_NAME' => 'user',
'BIRTH_DATE' => '1996-10-01',
'GENDER' => 'Male',
'role' => [
(int) 0 => [
'_joinData' => [
'ID' => '153',
'DESCRIPTION' => 'test edited text'
],
'ID' => '2'
]
]
]
我在person_to_role
中同时使用_joinData
个记录ID尝试了两个,结果相同:
INSERT INTO person_to_role (PERSON_ID, ROLE_ID, DESCRIPTION)
VALUES (129, 2, 'test edited text')
答案 0 :(得分:0)
在尝试使用'through' association的不同方法时,我在PersonToRoleTable.php
模型中添加了以下行:
$this->belongsTo('Person');
$this->belongsTo('Role');
在对这些内容进行评论后,一切都按预期工作,我可以保存更新,并在连接表中添加新的和删除现有记录。