OctoberCMS:如何保持双向友谊关系?

时间:2017-03-11 20:38:07

标签: mysql laravel octobercms octobercms-plugins

我正在扩展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_idfriend_id添加到friends表格,我就会自动将friend_iduser_id添加到表格中。为实现这一目标,我在afterSave模型中实施了beforeSaveFriendsPivot

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);
        }
    }
}

问题是永远不会调用beforeDeleteafterSave被调用但beforeDelete永远不会被调用,因此关系的反转不会被删除(user_id - friend_id会从数据库中移除friend_id - { {1}}不会被删除)。为什么没有调用user_id?有什么我做错了吗?有没有更好的方法来维持双向友谊关系?

2 个答案:

答案 0 :(得分:1)

我发现这篇文章是因为我试图做与你完全相同的事情。如果你已经解决了这个问题,我想知道你是否愿意分享你的解决方案?

答案 1 :(得分:0)

这听起来很奇怪,但也许这是因为Pivot模型的特殊delete behavior。它似乎使用QueryBuilder构建一个原始查询,从而绕过任何常规的Eloquent(十月)事件。 在我看来,最好的解决方案是在delete方法中手动触发delete事件,但我不确定这是否有任何副作用。 也许你可以测试一下并在Github上准备PR如果它有效。