id; $code['created_at'] = date('Y-m-d H:i:s'); if (!TalkRecordsCode::create($code)) { throw new Exception('插入聊天记录(代码消息)失败...'); } Db::commit(); } catch (Exception $e) { Db::rollBack(); return false; } LastMessage::getInstance()->save($insert->talk_type, $insert->user_id, $insert->receiver_id, [ 'text' => '[代码消息]', 'created_at' => date('Y-m-d H:i:s') ]); event()->dispatch(new TalkEvent(TalkEventConstant::EVENT_TALK, [ 'sender_id' => $insert->user_id, 'receiver_id' => $insert->receiver_id, 'talk_type' => $insert->talk_type, 'record_id' => $insert->id ])); return true; } /** * 创建文件类消息 * * @param array $message * @param array $file * @return bool */ public function insertFileMessage(array $message, array $file) { Db::beginTransaction(); try { $message['msg_type'] = TalkMessageType::FILE_MESSAGE; $message['created_at'] = date('Y-m-d H:i:s'); $message['updated_at'] = date('Y-m-d H:i:s'); $insert = TalkRecords::create($message); if (!$insert) { throw new Exception('插入聊天记录失败...'); } $file['record_id'] = $insert->id; $file['file_type'] = MediaTypeConstant::getMediaType($file['file_suffix']); $file['created_at'] = date('Y-m-d H:i:s'); if (!TalkRecordsFile::create($file)) { throw new Exception('插入聊天记录(代码消息)失败...'); } Db::commit(); } catch (Exception $e) { Db::rollBack(); return false; } LastMessage::getInstance()->save($insert->talk_type, $insert->user_id, $insert->receiver_id, [ 'text' => '[图片消息]', 'created_at' => date('Y-m-d H:i:s') ]); event()->dispatch(new TalkEvent(TalkEventConstant::EVENT_TALK, [ 'sender_id' => $insert->user_id, 'receiver_id' => $insert->receiver_id, 'talk_type' => $insert->talk_type, 'record_id' => $insert->id ])); return true; } /** * 添加投票消息 * * @param array $message * @param array $vote * @return bool */ public function insertVoteMessage(array $message, array $vote) { $answer_num = GroupMember::where('group_id', $message['receiver_id'])->where('is_quit', 0)->count(); Db::beginTransaction(); try { $message['msg_type'] = TalkMessageType::VOTE_MESSAGE; $message['created_at'] = date('Y-m-d H:i:s'); $message['updated_at'] = date('Y-m-d H:i:s'); $insert = TalkRecords::create($message); $options = []; foreach ($vote['answer_option'] as $k => $option) { $options[chr(65 + $k)] = $option; } $vote['record_id'] = $insert->id; $vote['answer_option'] = $options; $vote['answer_num'] = $answer_num; $vote['created_at'] = date('Y-m-d H:i:s'); $vote['updated_at'] = $vote['created_at']; if (!TalkRecordsVote::create($vote)) { throw new Exception('插入聊天记录(投票消息)失败...'); } Db::commit(); } catch (Exception $e) { Db::rollBack(); return false; } LastMessage::getInstance()->save($insert->talk_type, $insert->user_id, $insert->receiver_id, [ 'text' => '[投票消息]', 'created_at' => date('Y-m-d H:i:s') ]); event()->dispatch(new TalkEvent(TalkEventConstant::EVENT_TALK, [ 'sender_id' => $insert->user_id, 'receiver_id' => $insert->receiver_id, 'talk_type' => $insert->talk_type, 'record_id' => $insert->id ])); return true; } /** * 群投票处理方法 * * @param int $user_id * @param array $params * @return bool */ public function handleVote(int $user_id, array $params): array { $record = TalkRecords::join('talk_records_vote as vote', 'vote.record_id', '=', 'talk_records.id') ->where('talk_records.id', $params['record_id']) ->withCasts([ 'answer_option' => 'array' ]) ->first([ 'talk_records.id', 'talk_records.receiver_id', 'talk_records.talk_type', 'talk_records.msg_type', 'vote.id as vote_id', 'vote.answer_mode', 'vote.answer_option', 'vote.answer_num', 'vote.status as vote_status' ]); if (!$record) return false; if ($record->msg_type != TalkMessageType::VOTE_MESSAGE) { return [false, []]; } if (!UserRelation::isFriendOrGroupMember($user_id, $record->receiver_id, $record->talk_type)) { return [false, []]; } if (TalkRecordsVoteAnswer::where('vote_id', $record->vote_id)->where('user_id', $user_id)->exists()) { return [false, []]; } $options = $params['options']; sort($options); foreach ($options as $value) { if (!isset($record->answer_option[$value])) return [false, []]; } // 单选模式取第一个 if ($record->answer_mode == 0) { $options = [$options[0]]; } try { Db::transaction(function () use ($options, $record, $user_id) { TalkRecordsVote::where('id', $record->vote_id)->update([ 'answered_num' => Db::raw('answered_num + 1'), 'status' => Db::raw('if(answered_num >= answer_num, 1, 0)'), 'updated_at' => date('Y-m-d H:i:s'), ]); foreach ($options as $option) { TalkRecordsVoteAnswer::create([ 'vote_id' => $record->vote_id, 'user_id' => $user_id, 'option' => $option, 'created_at' => date('Y-m-d H:i:s'), ]); } }); } catch (\Exception $e) { return [false, []]; } // 更新投票缓存 VoteCache::getInstance()->updateCache($record->vote_id); $cache = VoteStatisticsCache::getInstance()->updateVoteCache($record->vote_id); // todo 推送消息 return [true, $cache]; } /** * 添加登录消息 * * @param array $message * @param array $loginParams * @return bool */ public function insertLoginMessage(array $message, array $loginParams) { Db::beginTransaction(); try { $message['receiver_id'] = RobotConstant::LOGIN_ROBOT; $message['talk_type'] = TalkModeConstant::PRIVATE_CHAT; $message['msg_type'] = TalkMessageType::USER_LOGIN_MESSAGE; $message['created_at'] = date('Y-m-d H:i:s'); $message['updated_at'] = date('Y-m-d H:i:s'); $insert = TalkRecords::create($message); if (!$insert) { throw new Exception('插入聊天记录失败...'); } $loginParams['record_id'] = $insert->id; $loginParams['created_at'] = date('Y-m-d H:i:s'); if (!TalkRecordsLogin::create($loginParams)) { throw new Exception('插入聊天记录(登录消息)失败...'); } Db::commit(); } catch (Exception $e) { Db::rollBack(); return false; } // 创建对话列表 di()->get(TalkListService::class)->create($insert->user_id, $insert->receiver_id, $insert->talk_type, true); LastMessage::getInstance()->save($insert->talk_type, $insert->user_id, $insert->receiver_id, [ 'text' => '[登录提醒]', 'created_at' => date('Y-m-d H:i:s') ]); event()->dispatch(new TalkEvent(TalkEventConstant::EVENT_TALK, [ 'sender_id' => $insert->user_id, 'receiver_id' => $insert->receiver_id, 'talk_type' => $insert->talk_type, 'record_id' => $insert->id ])); return true; } }