我正在扩展rainlab.user
插件,以允许每个用户通过一个包含以下字段的简单中间表与朋友联系:
user_id
friend_id
status
我扩展了User
模型:
use RainLab\User\Models\User as FrontUser;
FrontUser::extend(function($model) {
$model->belongsToMany['friends']=[
'RainLab\User\Models\User',
'table' => 'meysam_social_friends',
'pivot' => ['status'],
'pivotModel' => 'Meysam\Social\Models\FriendsPivot',
'timestamps' => true,
'key' => 'user_id',
'otherKey' => 'friend_id'
];
$model->addDynamicMethod('isFriendWith', function (FrontUser $user) use ($model) {
$model->friends->contains($user->id);
});
$model->addDynamicMethod('addFriend', function (FrontUser $user) use ($model) {
$model->friends()->attach($user->id);
});
$model->addDynamicMethod('removeFriend', function (FrontUser $user) use ($model) {
$model->friends()->detach($user->id);
});
});
并且还将Rainlab.User Controller
扩展为Friends
标签,其中列出了用户的所有朋友,并且可以添加和删除:
use RainLab\User\Controllers\Users as UsersController;
UsersController::extend(function($controller) {
if(!isset($controller->implement['Backend.Behaviors.RelationController'])) {
$controller->implement[] = 'Backend.Behaviors.RelationController';
}
$controller->relationConfig = '$/meysam/social/controllers/user/config_relations.yaml';
});
UsersController::extendFormFields(function($form, $model, $context) {
if(!$model instanceof FrontUser or $context != 'preview'){
// friends tab should not be displayed in update and create contexts
return;
}
$form->addTabFields([
'friends' => [
'label' => '',
'tab' => 'Friends',
'type' => 'partial',
'path' => '$/meysam/social/controllers/user/_friends.htm',
]
]);
});
现在我需要保持双向的友谊关系。即,只要将user_id
和friend_id
添加到friends
表格,我就会自动将friend_id
和user_id
添加到表格中。为实现这一目标,我在afterSave
模型中实施了beforeSave
和FriendsPivot
:
class FriendsPivot extends Pivot
{
/*
* Validation
*/
public $rules = [
'status' => 'required'
];
public $belongsTo = [
'user' => ['RainLab\User\Models\User', 'key' => 'user_id'],
'friend' => ['RainLab\User\Models\User', 'key' => 'friend_id']
];
public function getStatusOptions()
{
return [
1 => 'Pending',
2 => 'Approved',
3 => 'Blocked',
];
}
public function afterSave()
{
Log::info('Saving pivot...');
if(!$this->friend->isFriendWith($this->user)) {
$this->friend->addFriend($this->user);
}
}
public function beforeDelete()
{
Log::info('Deleting pivot...');
if($this->friend->isFriendWith($this->user)) {
$this->friend->removeFriend($this->user);
}
}
}
问题是永远不会调用beforeDelete
。 afterSave
被调用但beforeDelete
永远不会被调用,因此关系的反转不会被删除(user_id
- friend_id
会从数据库中移除friend_id
- { {1}}不会被删除)。为什么没有调用user_id
?有什么我做错了吗?有没有更好的方法来维持双向友谊关系?
答案 0 :(得分:1)
我发现这篇文章是因为我试图做与你完全相同的事情。如果你已经解决了这个问题,我想知道你是否愿意分享你的解决方案?
答案 1 :(得分:0)
这听起来很奇怪,但也许这是因为Pivot模型的特殊delete behavior。它似乎使用QueryBuilder构建一个原始查询,从而绕过任何常规的Eloquent(十月)事件。 在我看来,最好的解决方案是在delete方法中手动触发delete事件,但我不确定这是否有任何副作用。 也许你可以测试一下并在Github上准备PR如果它有效。