????
Your IP : 18.222.112.64
<?php
namespace ElementorPro\Modules\Notes\Database\Models;
use ElementorPro\Core\Database\Model_Base;
use ElementorPro\Core\Database\Query_Builder;
use ElementorPro\Core\Utils\Collection;
use ElementorPro\Modules\Notes\Admin_Page;
use ElementorPro\Modules\Notes\Database\Query\Note_Query_Builder;
use ElementorPro\Modules\Notes\Module;
use ElementorPro\Modules\Notes\User\Capabilities;
use ElementorPro\Modules\Notes\Utils;
if ( ! defined( 'ABSPATH' ) ) {
exit; // Exit if accessed directly
}
class Note extends Model_Base {
// Note statuses.
const STATUS_PUBLISH = 'publish';
const STATUS_DRAFT = 'draft';
const STATUS_TRASH = 'trash';
// Note user relations.
const USER_RELATION_READ = 'read';
const USER_RELATION_MENTION = 'mention';
/**
* Note ID.
*
* @var int
*/
public $id;
/**
* Note's post ID.
*
* @var null|int
*/
public $post_id = null;
/**
* Note's element ID.
*
* @var null|int
*/
public $element_id = null;
/**
* Note's parent ID.
*
* @var int
*/
public $parent_id = 0;
/**
* Note's author ID.
*
* @var null|int
*/
public $author_id = null;
/**
* @var null|string
*/
public $author_display_name = null;
/**
* Note's route post ID.
*
* @var null|integer
*/
public $route_post_id = null;
/**
* @var string
*/
public $route_url = null;
/**
* @var string
*/
public $route_title = null;
/**
* Note's status.
*
* @var string
*/
public $status = self::STATUS_PUBLISH;
/**
* Note's position in the element.
*
* @var array{x: int, y: int}
*/
public $position = [
'x' => 0,
'y' => 0,
];
/**
* Note's content.
*
* @var null|string
*/
public $content = null;
/**
* Note's resolve status.
*
* @var bool
*/
public $is_resolved = false;
/**
* Note's public status.
*
* @var bool
*/
public $is_public = true;
/**
* Is the note read by the user.
*
* @var boolean
*/
public $is_read = false;
/**
* Note's replies.
*
* @var Collection <Note>
*/
public $replies;
/**
* Note's mentions.
*
* @var Collection<User>
*/
public $mentions;
/**
* Note's author.
*
* @var User
*/
public $author;
/**
* Note's document
*
* @var Document
*/
public $document;
/**
* Note's replies count.
*
* @var int
*/
public $replies_count = 0;
/**
* Note's unread replies count.
*
* @var int
*/
public $unread_replies_count = 0;
/**
* Note's readers.
*
* @var Collection <User>
*/
public $readers;
/**
* Note's creation time.
*
* @var \DateTime
*/
public $created_at;
/**
* Note's last update time.
*
* @var \DateTime
*/
public $updated_at;
/**
* Note's last activity time.
*
* @var \DateTime
*/
public $last_activity_at;
/**
* User's capabilities for the current note.
* [
* 'edit' => boolean,
* 'delete' => boolean,
* ]
*
* @var array
*/
public $user_can = [];
/**
* Casts array.
*
* @var array
*/
protected static $casts = [
'id' => self::TYPE_INTEGER,
'post_id' => self::TYPE_INTEGER,
'route_post_id' => self::TYPE_INTEGER,
'parent_id' => self::TYPE_INTEGER,
'author_id' => self::TYPE_INTEGER,
'position' => self::TYPE_JSON,
'is_resolved' => self::TYPE_BOOLEAN,
'is_public' => self::TYPE_BOOLEAN,
'is_read' => self::TYPE_BOOLEAN,
'replies' => self::TYPE_COLLECTION,
'mentions' => self::TYPE_COLLECTION,
'readers' => self::TYPE_COLLECTION,
'replies_count' => self::TYPE_INTEGER,
'unread_replies_count' => self::TYPE_INTEGER,
'created_at' => self::TYPE_DATETIME_GMT,
'updated_at' => self::TYPE_DATETIME_GMT,
'last_activity_at' => self::TYPE_DATETIME_GMT,
];
public function __construct( array $fields ) {
// Defaults must be empty collection, when there is no replies or mentions it should remain empty.
$this->replies = new Collection( [] );
$this->mentions = new Collection( [] );
$this->readers = new Collection( [] );
parent::__construct( $fields );
}
/**
* Override the default Query Builder.
*
* @param \wpdb|null $connection
*
* @return Note_Query_Builder
*/
public static function query( \wpdb $connection = null ) {
return ( new Note_Query_Builder( $connection ) )->from( static::get_table() );
}
/**
* Get the notes table name.
*
* @return string
*/
public static function get_table() {
return Module::TABLE_NOTES;
}
/**
* Is the current note is top level note.
*
* @return bool
*/
public function is_thread() {
return 0 === $this->parent_id;
}
/**
* Determine if the current note is a reply.
*
* @return bool
*/
public function is_reply() {
return ! $this->is_thread();
}
/**
* Get the thread ID of the current note.
*
* @return int
*/
public function get_thread_id() {
return $this->is_thread() ? $this->id : $this->parent_id;
}
/**
* Get the note deep link.
*
* @param bool $force_auth - Whether to force authentication. Defaults to `true`.
*
* @return string
*/
public function get_url( $force_auth = true ) {
// NOTICE: Make sure that the returned URL is not dynamic!
return static::generate_url(
$this->get_thread_id(),
$this->route_url,
$force_auth
);
}
/**
* Generate a note deep link URL.
*
* @param string|int $id - Note ID.
* @param string $route_url - Note route URL. Required if `$force_auth = false`.
* @param bool $force_auth - Whether to force authentication. Defaults to `true`. Used in cases where the user
* should be passed through the proxy in order to force their authentication (since the
* "Notes" feature and the Web-CLI are available only for logged-in users).
*
* @return string
*/
public static function generate_url( $id = null, $route_url = '', $force_auth = true ) {
// Add a placeholder if ID doesn't exist (used for passing the note URL as a pattern).
$id = ( null === $id ) ? '{{NOTE_ID}}' : (int) $id;
if ( $force_auth ) {
$page = sprintf( 'admin.php?page=%s¬e-id=%s', Admin_Page::PAGE_ID, $id );
return admin_url( $page );
}
$command = sprintf( '#e:run:notes/open?{"id":%s}', $id );
$base_url = get_site_url( null, Utils::clean_url( $route_url ) );
return $base_url . $command;
}
/**
* @shortcut `$this->add_user_relation()`
*/
public function add_readers( $user_ids = [] ) {
$this->add_user_relation( static::USER_RELATION_READ, $user_ids );
}
/**
* @shortcut `$this->remove_user_relation()`
*/
public function remove_readers( $user_ids = [] ) {
$this->remove_user_relation( static::USER_RELATION_READ, $user_ids );
}
/**
* @shortcut `$this->sync_user_relation()`
*/
public function sync_mentions( $user_keys = [], $key = 'ID' ) {
return $this->sync_user_relation( static::USER_RELATION_MENTION, $user_keys, $key );
}
/**
* @shortcut `$this->add_user_relation()`
*/
public function add_mentions( $user_ids = [] ) {
$this->add_user_relation( static::USER_RELATION_MENTION, $user_ids );
}
/**
* Remove old relations and add new ones.
*
* @param $type
* @param array $user_keys
* @param string $key
*
* @return Collection Only users with a newly created relation (excluding the existing ones).
*/
public function sync_user_relation( $type, array $user_keys, $key = 'ID' ) {
$users = User::query()
->where_in( $key, $user_keys )
->get();
$already_has_relation = ( new Query_Builder() )
->select( [ 'user_id' ] )
->from( Module::TABLE_NOTES_USERS_RELATIONS )
->where( 'type', '=', $type )
->where( 'note_id', '=', $this->id )
->get()
->pluck( 'user_id' );
$should_have_relation = $users
->pluck( 'ID' )
->unique();
$should_remove = $already_has_relation->diff( $should_have_relation )->values();
$should_insert = $should_have_relation->diff( $already_has_relation )->values();
// Delete all the previous relations.
$this->remove_user_relation( $type, $should_remove );
// Only the users that not already in relation.
$this->add_user_relation( $type, $should_insert );
// Return only the users that were inserted in the users_relations DB table.
return $users->filter( function ( User $user ) use ( $should_insert ) {
return in_array( $user->ID, $should_insert, true );
} );
}
/**
* Remove user relation.
*
* @param $type
* @param array $user_ids
*/
public function remove_user_relation( $type, array $user_ids ) {
( new Query_Builder() )
->table( Module::TABLE_NOTES_USERS_RELATIONS )
->where( 'note_id', '=', $this->id )
->where( 'type', '=', $type )
->where_in( 'user_id', $user_ids )
->delete();
}
/**
* Add user relation.
*
* @param $type
* @param array $user_ids
*
* @throws \Exception
*/
public function add_user_relation( $type, array $user_ids ) {
$now = gmdate( 'Y-m-d H:i:s' );
foreach ( $user_ids as $user_id ) {
( new Query_Builder() )
->table( Module::TABLE_NOTES_USERS_RELATIONS )
->insert( [
'note_id' => $this->id,
'user_id' => $user_id,
'type' => $type,
'created_at' => $now,
'updated_at' => $now,
] );
}
}
/**
* Add user capabilities to the Note and its replies.
*
* @param integer $user_id - User ID to use.
* @param bool $recursive - Whether to add the capabilities also to the replies.
*
* @return Note
*/
public function attach_user_capabilities( $user_id, $recursive = true ) {
$this->user_can = [
'edit' => user_can( $user_id, Capabilities::EDIT_NOTES, $this ),
'delete' => user_can( $user_id, Capabilities::DELETE_NOTES, $this ),
];
// Add the capabilities also to the replies.
if ( $recursive ) {
$this->replies = $this->replies->map( function ( Note $reply ) use ( $user_id ) {
$reply->user_can = [
'edit' => user_can( $user_id, Capabilities::EDIT_NOTES, $reply ),
'delete' => user_can( $user_id, Capabilities::DELETE_NOTES, $reply ),
];
return $reply;
} );
}
return $this;
}
}