mirror of
https://github.com/flarum/core.git
synced 2025-07-17 14:51:19 +02:00
392 lines
10 KiB
PHP
Executable File
392 lines
10 KiB
PHP
Executable File
<?php namespace Flarum\Core\Discussions;
|
|
|
|
use Flarum\Core\Model;
|
|
use Flarum\Core\Discussions\Events\DiscussionWasDeleted;
|
|
use Flarum\Core\Discussions\Events\DiscussionWasStarted;
|
|
use Flarum\Core\Discussions\Events\DiscussionWasRenamed;
|
|
use Flarum\Core\Posts\Events\PostWasDeleted;
|
|
use Flarum\Core\Posts\Post;
|
|
use Flarum\Core\Posts\MergeablePost;
|
|
use Flarum\Core\Users\Guest;
|
|
use Flarum\Core\Users\User;
|
|
use Flarum\Core\Support\EventGenerator;
|
|
use Flarum\Core\Support\Locked;
|
|
use Flarum\Core\Support\VisibleScope;
|
|
use Flarum\Core\Support\ValidatesBeforeSave;
|
|
|
|
class Discussion extends Model
|
|
{
|
|
use EventGenerator;
|
|
use Locked;
|
|
use VisibleScope;
|
|
use ValidatesBeforeSave;
|
|
|
|
/**
|
|
* The table associated with the model.
|
|
*
|
|
* @var string
|
|
*/
|
|
protected $table = 'discussions';
|
|
|
|
/**
|
|
* The validation rules for this model.
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $rules = [
|
|
'title' => 'required',
|
|
'start_time' => 'required|date',
|
|
'comments_count' => 'integer',
|
|
'participants_count' => 'integer',
|
|
'start_user_id' => 'integer',
|
|
'start_post_id' => 'integer',
|
|
'last_time' => 'date',
|
|
'last_user_id' => 'integer',
|
|
'last_post_id' => 'integer',
|
|
'last_post_number' => 'integer'
|
|
];
|
|
|
|
/**
|
|
* The attributes that should be mutated to dates.
|
|
*
|
|
* @var array
|
|
*/
|
|
protected static $dateAttributes = ['start_time', 'last_time'];
|
|
|
|
/**
|
|
* The user for which the state relationship should be loaded.
|
|
*
|
|
* @var User
|
|
*/
|
|
protected static $stateUser;
|
|
|
|
/**
|
|
* An array of callables that apply constraints to the postsVisibleTo query.
|
|
*
|
|
* @var callable[]
|
|
*/
|
|
protected static $postVisibilityScopes = [];
|
|
|
|
/**
|
|
* Boot the model.
|
|
*
|
|
* @return void
|
|
*/
|
|
public static function boot()
|
|
{
|
|
parent::boot();
|
|
|
|
static::deleted(function ($discussion) {
|
|
$discussion->raise(new DiscussionWasDeleted($discussion));
|
|
|
|
// Delete all of the posts in the discussion. Before we delete them
|
|
// in a big batch query, we will loop through them and raise a
|
|
// PostWasDeleted event for each post.
|
|
$posts = $discussion->posts()->allTypes();
|
|
|
|
foreach ($posts->get() as $post) {
|
|
$discussion->raise(new PostWasDeleted($post));
|
|
}
|
|
|
|
$posts->delete();
|
|
|
|
// Delete all of the 'state' records for all of the users who have
|
|
// read the discussion.
|
|
$discussion->readers()->detach();
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Start a new discussion. Raises the DiscussionWasStarted event.
|
|
*
|
|
* @param string $title
|
|
* @param User $user
|
|
* @return static
|
|
*/
|
|
public static function start($title, User $user)
|
|
{
|
|
$discussion = new static;
|
|
|
|
$discussion->title = $title;
|
|
$discussion->start_time = time();
|
|
$discussion->start_user_id = $user->id;
|
|
|
|
$discussion->setRelation('startUser', $user);
|
|
|
|
$discussion->raise(new DiscussionWasStarted($discussion));
|
|
|
|
return $discussion;
|
|
}
|
|
|
|
/**
|
|
* Rename the discussion. Raises the DiscussionWasRenamed event.
|
|
*
|
|
* @param string $title
|
|
* @param User $user
|
|
* @return $this
|
|
*/
|
|
public function rename($title, User $user)
|
|
{
|
|
if ($this->title !== $title) {
|
|
$oldTitle = $this->title;
|
|
$this->title = $title;
|
|
|
|
$this->raise(new DiscussionWasRenamed($this, $user, $oldTitle));
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Set the discussion's start post details.
|
|
*
|
|
* @param \Flarum\Core\Posts\Post $post
|
|
* @return $this
|
|
*/
|
|
public function setStartPost(Post $post)
|
|
{
|
|
$this->start_time = $post->time;
|
|
$this->start_user_id = $post->user_id;
|
|
$this->start_post_id = $post->id;
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Set the discussion's last post details.
|
|
*
|
|
* @param \Flarum\Core\Posts\Post $post
|
|
* @return $this
|
|
*/
|
|
public function setLastPost(Post $post)
|
|
{
|
|
$this->last_time = $post->time;
|
|
$this->last_user_id = $post->user_id;
|
|
$this->last_post_id = $post->id;
|
|
$this->last_post_number = $post->number;
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Refresh a discussion's last post details.
|
|
*
|
|
* @return $this
|
|
*/
|
|
public function refreshLastPost()
|
|
{
|
|
if ($lastPost = $this->comments()->latest('time')->first()) {
|
|
$this->setLastPost($lastPost);
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Refresh the discussion's comments count.
|
|
*
|
|
* @return $this
|
|
*/
|
|
public function refreshCommentsCount()
|
|
{
|
|
$this->comments_count = $this->comments()->count();
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Refresh the discussion's participants count.
|
|
*
|
|
* @return $this
|
|
*/
|
|
public function refreshParticipantsCount()
|
|
{
|
|
$this->participants_count = $this->participants()->count('users.id');
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Save a post, attempting to merge it with the discussion's last post.
|
|
*
|
|
* The merge logic is delegated to the new post. (As an example, a
|
|
* DiscussionRenamedPost will merge if adjacent to another
|
|
* DiscussionRenamedPost, and delete if the title has been reverted
|
|
* completely.)
|
|
*
|
|
* @param \Flarum\Core\Posts\Post $post The post to save.
|
|
* @return \Flarum\Core\Posts\Post The resulting post. It may or may not be
|
|
* the same post as was originally intended to be saved. It also may not
|
|
* exist, if the merge logic resulted in deletion.
|
|
*/
|
|
public function mergePost(MergeablePost $post)
|
|
{
|
|
$lastPost = $this->posts()->latest('time')->first();
|
|
|
|
return $post->saveAfter($lastPost);
|
|
}
|
|
|
|
/**
|
|
* Define the relationship with the discussion's posts.
|
|
*
|
|
* @return \Illuminate\Database\Eloquent\Relations\HasMany
|
|
*/
|
|
public function posts()
|
|
{
|
|
return $this->hasMany('Flarum\Core\Posts\Post');
|
|
}
|
|
|
|
/**
|
|
* Define the relationship with the discussion's posts, but only ones which
|
|
* are visible to the given user.
|
|
*
|
|
* @param User $user
|
|
* @return \Illuminate\Database\Eloquent\Relations\HasMany
|
|
*/
|
|
public function postsVisibleTo(User $user)
|
|
{
|
|
$query = $this->posts();
|
|
|
|
foreach (static::$postVisibilityScopes as $scope) {
|
|
$scope($query, $user, $this);
|
|
}
|
|
|
|
return $query;
|
|
}
|
|
|
|
/**
|
|
* Define the relationship with the discussion's publicly-visible comments.
|
|
*
|
|
* @return \Illuminate\Database\Eloquent\Relations\HasMany
|
|
*/
|
|
public function comments()
|
|
{
|
|
return $this->postsVisibleTo(new Guest)->where('type', 'comment');
|
|
}
|
|
|
|
/**
|
|
* Query the discussion's participants (a list of unique users who have
|
|
* posted in the discussion).
|
|
*
|
|
* @return \Illuminate\Database\Eloquent\Builder
|
|
*/
|
|
public function participants()
|
|
{
|
|
return User::join('posts', 'posts.user_id', '=', 'users.id')
|
|
->where('posts.discussion_id', $this->id)
|
|
->select('users.*')
|
|
->distinct();
|
|
}
|
|
|
|
/**
|
|
* Define the relationship with the discussion's first post.
|
|
*
|
|
* @return \Illuminate\Database\Eloquent\Relations\BelongsTo
|
|
*/
|
|
public function startPost()
|
|
{
|
|
return $this->belongsTo('Flarum\Core\Posts\Post', 'start_post_id');
|
|
}
|
|
|
|
/**
|
|
* Define the relationship with the discussion's author.
|
|
*
|
|
* @return \Illuminate\Database\Eloquent\Relations\BelongsTo
|
|
*/
|
|
public function startUser()
|
|
{
|
|
return $this->belongsTo('Flarum\Core\Users\User', 'start_user_id');
|
|
}
|
|
|
|
/**
|
|
* Define the relationship with the discussion's last post.
|
|
*
|
|
* @return \Illuminate\Database\Eloquent\Relations\BelongsTo
|
|
*/
|
|
public function lastPost()
|
|
{
|
|
return $this->belongsTo('Flarum\Core\Posts\Post', 'last_post_id');
|
|
}
|
|
|
|
/**
|
|
* Define the relationship with the discussion's most recent author.
|
|
*
|
|
* @return \Illuminate\Database\Eloquent\Relations\BelongsTo
|
|
*/
|
|
public function lastUser()
|
|
{
|
|
return $this->belongsTo('Flarum\Core\Users\User', 'last_user_id');
|
|
}
|
|
|
|
/**
|
|
* Define the relationship with the discussion's readers.
|
|
*
|
|
* @return \Illuminate\Database\Eloquent\Relations\BelongsTo
|
|
*/
|
|
public function readers()
|
|
{
|
|
return $this->belongsToMany('Flarum\Core\Users\User', 'users_discussions');
|
|
}
|
|
|
|
/**
|
|
* Define the relationship with the discussion's state for a particular
|
|
* user.
|
|
*
|
|
* If no user is passed (i.e. in the case of eager loading the 'state'
|
|
* relation), then the static `$stateUser` property is used.
|
|
*
|
|
* @see Discussion::setStateUser()
|
|
*
|
|
* @param User|null $user
|
|
* @return \Illuminate\Database\Eloquent\Relations\HasOne
|
|
*/
|
|
public function state(User $user = null)
|
|
{
|
|
$user = $user ?: static::$stateUser;
|
|
|
|
return $this->hasOne('Flarum\Core\Discussions\DiscussionState')->where('user_id', $user ? $user->id : null);
|
|
}
|
|
|
|
/**
|
|
* Get the state model for a user, or instantiate a new one if it does not
|
|
* exist.
|
|
*
|
|
* @param User $user
|
|
* @return \Flarum\Core\Discussions\DiscussionState
|
|
*/
|
|
public function stateFor(User $user)
|
|
{
|
|
$state = $this->state($user)->first();
|
|
|
|
if (! $state) {
|
|
$state = new DiscussionState;
|
|
$state->discussion_id = $this->id;
|
|
$state->user_id = $user->id;
|
|
}
|
|
|
|
return $state;
|
|
}
|
|
|
|
/**
|
|
* Set the user for which the state relationship should be loaded.
|
|
*
|
|
* @param User $user
|
|
*/
|
|
public static function setStateUser(User $user)
|
|
{
|
|
static::$stateUser = $user;
|
|
}
|
|
|
|
/**
|
|
* Constrain which posts are visible to a user.
|
|
*
|
|
* @param callable $scope A callback that applies constraints to the posts
|
|
* query. It is passed three parameters: the query builder object, the
|
|
* user to constrain posts for, and the discussion instance.
|
|
*/
|
|
public static function addPostVisibilityScope(callable $scope)
|
|
{
|
|
static::$postVisibilityScopes[] = $scope;
|
|
}
|
|
}
|