Laravel 按最后一条消息对对话进行排序

Posted

技术标签:

【中文标题】Laravel 按最后一条消息对对话进行排序【英文标题】:Laravel sort conversations by last message 【发布时间】:2019-05-10 12:29:29 【问题描述】:

我有一个 Conversation 模型,其中包含许多 ConversationMessage 模型。

现在我想根据对话的最后一条消息对对话进行排序。基本上就像WhatsApp。如何构建查询?

作为我的代码的旁注:对话包含一个用户 (user_id) 和一家关联公司 (company_id)。公司与用户有belongsToMany 关系。

Conversation::whereHas('company', function ($q) use ($userId) 
    $q->whereHas('users', function ($q1) use ($userId) 
        $q1->where('users.id', $userId);
    );
)->orWhereHas('user', function($q) use ($userId) 
    $q->where('user.id', $userId);
)->with(array('conversationMessage' => function($q) 
    $q->orderBy('created_at', 'DESC');
)->get();

ConversationMessage 关系模型进行排序,而不是对话。如何根据最后一条消息对Conversations 进行排序?

【问题讨论】:

在任何事情之前,我不确定,但我认为您可以在一行中写下您的第一个 whereHasConversation::whereHas('company.users', function ($q) use ($userId) $q->where('users.id', $userId); ) 没错,谢谢! 【参考方案1】:

您可以将last_message_at 之类的timestamp 字段添加到您的Conversations 模型中,当有新消息添加到该Conversation 时更新该字段,然后您可以根据last_message_at 对结果进行排序。

或者您可以像下面这样在查询中过滤结果(测试它是否有效):

    Conversation::whereHas('company', function ($q) use ($userId) 
        $q->whereHas('users', function ($q1) use ($userId) 
            $q1->where('users.id', $userId);
        );
    )->orWhereHas('user', function($q) use ($userId) 
        $q->where('user.id', $userId);
    )->with(array('conversationMessage' => function($q) 
        $q->orderBy('created_at', 'DESC');
    )
    ->selectRaw("conversations.*, (SELECT MAX(created_at) from conversation_messages WHERE conversation_messages.conversation_id=conversations.id) as latest_message_on")
    ->orderBy("latest_message_on", "DESC")
    ->get();

【讨论】:

这也是我在考虑的一个选项,但这将是每条新消息的更新查询。就我个人而言,如果可能的话,我会发现对关系进行排序会更好 这是对答案的更新,测试它是否有效。

以上是关于Laravel 按最后一条消息对对话进行排序的主要内容,如果未能解决你的问题,请参考以下文章

如何在 Laravel 中使用 Eloquent 对 NULL 值进行最后排序

如何按最新消息对聊天列表进行排序?

Laravel,按最新创建的子关系排序模型

如何在 Laravel 中按字母顺序对记录进行排序

如何在 Laravel 5.1 中按多列对 Illuminate Collection 进行排序?

laravel 按日期对数组进行排序