<?php
/**
 * Limitation Manager
 *
 * Handles processes related to limitations.
 *
 * @package WP_Ultimo
 * @subpackage Managers/Limitation_Manager
 * @since 2.0.0
 */

namespace WP_Ultimo\Managers;

// Exit if accessed directly
defined('ABSPATH') || exit;

use \WP_Ultimo\Dependencies\Psr\Log\LogLevel;
use \WP_Ultimo\Objects\Limitations;
use \WP_Ultimo\Database\Sites\Site_Type;

/**
 * Handles processes related to limitations.
 *
 * @since 2.0.0
 */
class Limitation_Manager {

	use \WP_Ultimo\Traits\Singleton;

	/**
	 * Instantiate the necessary hooks.
	 *
	 * @since 2.0.0
	 * @return void
	 */
	public function init() {

		if (WP_Ultimo()->is_loaded() === false) {

			return;

		} // end if;

		add_filter('wu_product_options_sections', array($this, 'add_limitation_sections'), 10, 2);

		add_filter('wu_membership_options_sections', array($this, 'add_limitation_sections'), 10, 2);

		add_filter('wu_site_options_sections', array($this, 'add_limitation_sections'), 10, 2);

		add_action('plugins_loaded', array($this, 'register_forms'));

		add_action('wu_async_handle_plugins', array($this, 'async_handle_plugins'), 10, 5);

		add_action('wu_async_switch_theme', array($this, 'async_switch_theme'), 10, 2);

	} // end init;

	/**
	 * Handles async plugin activation and deactivation.
	 *
	 * @since 2.0.0
	 *
	 * @param string       $action The action to perform, can be either 'activate' or 'deactivate'.
	 * @param int          $site_id The site ID.
	 * @param string|array $plugins The plugin or list of plugins to (de)activate.
	 * @param boolean      $network_wide If we want to (de)activate it network-wide.
	 * @param boolean      $silent IF we should do the process silently - true by default.
	 * @return bool
	 */
	public function async_handle_plugins($action, $site_id, $plugins, $network_wide = false, $silent = true) {

		$results = false;

		// Avoid doing anything on the main site.
		if (wu_get_main_site_id() === $site_id) {

			return $results;

		} // end if;

		switch_to_blog($site_id);

		if ($action === 'activate') {

			$results = activate_plugins($plugins, '', $network_wide, $silent);

		} elseif ($action === 'deactivate') {

			$results = deactivate_plugins($plugins, $silent, $network_wide);

		} // end if;

		if (is_wp_error($results)) {

			wu_log_add('plugins', $results, LogLevel::ERROR);

		} // end if;

		restore_current_blog();

		return $results;

	} // end async_handle_plugins;

	/**
	 * Switch themes via Job Queue.
	 *
	 * @since 2.0.0
	 *
	 * @param int    $site_id The site ID.
	 * @param string $theme_stylesheet The theme stylesheet.
	 * @return true
	 */
	public function async_switch_theme($site_id, $theme_stylesheet): bool {

		switch_to_blog($site_id);

		switch_theme($theme_stylesheet);

		restore_current_blog();

		return true;

	} // end async_switch_theme;

	/**
	 * Register the modal windows to confirm resetting the limitations.
	 *
	 * @since 2.0.0
	 * @return void
	 */
	public function register_forms() {

		wu_register_form('confirm_limitations_reset', array(
			'render'  => array($this, 'render_confirm_limitations_reset'),
			'handler' => array($this, 'handle_confirm_limitations_reset'),
		));

	} // end register_forms;

	/**
	 * Renders the conformation modal to reset limitations.
	 *
	 * @since 2.0.0
	 * @return void
	 */
	public function render_confirm_limitations_reset() {

		$fields = array(
			'confirm'       => array(
				'type'      => 'toggle',
				'title'     => __('Confirm Reset', 'wp-ultimo'),
				'desc'      => __('This action can not be undone.', 'wp-ultimo'),
				'html_attr' => array(
					'v-model' => 'confirmed',
				),
			),
			'submit_button' => array(
				'type'            => 'submit',
				'title'           => __('Reset Limitations', 'wp-ultimo'),
				'value'           => 'save',
				'classes'         => 'button button-primary wu-w-full',
				'wrapper_classes' => 'wu-items-end',
				'html_attr'       => array(
					'v-bind:disabled' => '!confirmed',
				),
			),
			'id'            => array(
				'type'  => 'hidden',
				'value' => wu_request('id'),
			),
			'model'         => array(
				'type'  => 'hidden',
				'value' => wu_request('model'),
			),
		);

		$form_attributes = array(
			'title'                 => __('Reset', 'wp-ultimo'),
			'views'                 => 'admin-pages/fields',
			'classes'               => 'wu-modal-form wu-widget-list wu-striped wu-m-0 wu-mt-0',
			'field_wrapper_classes' => 'wu-w-full wu-box-border wu-items-center wu-flex wu-justify-between wu-p-4 wu-m-0 wu-border-t wu-border-l-0 wu-border-r-0 wu-border-b-0 wu-border-gray-300 wu-border-solid',
			'html_attr'             => array(
				'data-wu-app' => 'reset_limitations',
				'data-state'  => json_encode(array(
					'confirmed' => false,
				)),
			),
		);

		$form = new \WP_Ultimo\UI\Form('reset_limitations', $fields, $form_attributes);

		$form->render();

	} // end render_confirm_limitations_reset;

	/**
	 * Handles the reset of permissions.
	 *
	 * @since 2.0.0
	 * @return void
	 */
	public function handle_confirm_limitations_reset() {

		$id = wu_request('id');

		$model = wu_request('model');

		if (!$id || !$model) {

			wp_send_json_error(new \WP_Error(
				'parameters-not-found',
				__('Required parameters are missing.', 'wp-ultimo')
			));

		} // end if;

		/*
		 * Remove limitations object
		 */
		Limitations::remove_limitations($model, $id);

		wp_send_json_success(array(
			'redirect_url' => wu_network_admin_url("wp-ultimo-edit-{$model}", array(
				'id'      => $id,
				'updated' => 1,
			))
		));

	} // end handle_confirm_limitations_reset;

	/**
	 * Returns the type of the object that has limitations.
	 *
	 * @since 2.0.0
	 *
	 * @param \WP_Ultimo\Models\Trait\Trait_Limitable $object Model to test.
	 * @return string
	 */
	public function get_object_type($object) {

		$model = false;

		if (is_a($object, \WP_Ultimo\Models\Site::class)) {

			$model = 'site';

		} elseif (is_a($object, WP_Ultimo\Models\Membership::class)) {

			$model = 'membership';

		} elseif (is_a($object, \WP_Ultimo\Models\Product::class)) {

			$model = 'product';

		} // end if;

		return apply_filters('wu_limitations_get_object_type', $model);

	} // end get_object_type;

	/**
	 * Injects the limitations panels when necessary.
	 *
	 * @since 2.0.0
	 *
	 * @param array                                   $sections List of tabbed widget sections.
	 * @param \WP_Ultimo\Models\Trait\Trait_Limitable $object The model being edited.
	 * @return array
	 */
	public function add_limitation_sections($sections, $object) {

		if ($this->get_object_type($object) === 'site' && $object->get_type() !== Site_Type::CUSTOMER_OWNED) {

			$html = sprintf('<span class="wu--mt-4 wu-p-2 wu-bg-blue-100 wu-text-blue-600 wu-rounded wu-block">%s</span>', __('Limitations are only available for customer-owned sites. You need to change the type to Customer-owned and save this site before the options are shown.', 'wp-ultimo'));

			$sections['sites'] = array(
				'title'  => __('Limits', 'wp-ultimo'),
				'desc'   => __('Only customer-owned sites have limitations.', 'wp-ultimo'),
				'icon'   => 'dashicons-wu-browser',
				'fields' => array(
					'note' => array(
						'type'    => 'html',
						'content' => $html,
					)
				)
			);

			return $sections;

		} // end if;

		if ($this->get_object_type($object) !== 'site') {

			$sections['sites'] = array(
				'title'  => __('Sites', 'wp-ultimo'),
				'desc'   => __('Control limitations imposed to the number of sites allowed for memberships attached to this product.', 'wp-ultimo'),
				'icon'   => 'dashicons-wu-browser',
				'fields' => $this->get_sites_fields($object),
				'v-show' => "get_state_value('product_type', 'none') !== 'service'",
				'state'  => array(
					'limit_sites' => $object->get_limitations()->sites->is_enabled(),
				),
			);

		} // end if;

		/*
		 * Add Visits limitation control
		 */
		if ((bool) wu_get_setting('enable_visits_limiting', true)) {

			$sections['visits'] = array(
				'title'  => __('Visits', 'wp-ultimo'),
				'desc'   => __('Control limitations imposed to the number of unique visitors allowed for memberships attached to this product.', 'wp-ultimo'),
				'icon'   => 'dashicons-wu-man',
				'v-show' => "get_state_value('product_type', 'none') !== 'service'",
				'state'  => array(
					'limit_visits' => $object->get_limitations()->visits->is_enabled(),
				),
				'fields' => array(
					'modules[visits][enabled]' => array(
						'type'      => 'toggle',
						'title'     => __('Limit Unique Visits', 'wp-ultimo'),
						'desc'      => __('Toggle this option to enable unique visits limitation.', 'wp-ultimo'),
						'value'     => 10,
						'html_attr' => array(
							'v-model' => 'limit_visits'
						),
					),
				),
			);

			if ($object->model !== 'product') {

				$sections['visits']['fields']['modules_visits_overwrite'] = $this->override_notice($object->get_limitations(false)->visits->has_own_enabled());

			} // end if;

			$sections['visits']['fields']['modules[visits][limit]'] = array(
				'type'              => 'number',
				'title'             => __('Unique Visits Quota', 'wp-ultimo'),
				'desc'              => __('Set a top limit for the number of monthly unique visits. Leave empty or 0 to allow for unlimited visits.', 'wp-ultimo'),
				'placeholder'       => __('e.g. 10000', 'wp-ultimo'),
				'value'             => $object->get_limitations()->visits->get_limit(),
				'wrapper_html_attr' => array(
					'v-show'  => 'limit_visits',
					'v-cloak' => '1',
				),
				'html_attr'         => array(
					':min' => 'limit_visits ? 1 : -999',
				),
			);

			if ($object->model !== 'product') {

				$sections['visits']['fields']['allowed_visits_overwrite'] = $this->override_notice($object->get_limitations(false)->visits->has_own_limit(), array('limit_visits'));

			} // end if;

			/*
			 * If this is a site edit screen, show the current values
			 * for visits and the reset date
			 */
			if ($this->get_object_type($object) === 'site') {

				$sections['visits']['fields']['visits_count'] = array(
					'type'              => 'text-display',
					'title'             => __('Current Unique Visits Count this Month', 'wp-ultimo'),
					'desc'              => __('Current visits count for this particular site.', 'wp-ultimo'),
					'display_value'     => sprintf('%s visit(s)', $object->get_visits_count()),
					'wrapper_html_attr' => array(
						'v-show'  => 'limit_visits',
						'v-cloak' => '1',
					),
				);

			} // end if;

		} // end if;

		$sections['users'] = array(
			'title'  => __('Users', 'wp-ultimo'),
			'desc'   => __('Control limitations imposed to the number of user allowed for memberships attached to this product.', 'wp-ultimo'),
			'icon'   => 'dashicons-wu-users',
			'v-show' => "get_state_value('product_type', 'none') !== 'service'",
			'state'  => array(
				'limit_users' => $object->get_limitations()->users->is_enabled(),
			),
			'fields' => array(
				'modules[users][enabled]' => array(
					'type'      => 'toggle',
					'title'     => __('Limit User', 'wp-ultimo'),
					'desc'      => __('Enable user limitations for this product.', 'wp-ultimo'),
					'html_attr' => array(
						'v-model' => 'limit_users'
					),
				),
			),
		);

		if ($object->model !== 'product') {

			$sections['users']['fields']['modules_user_overwrite'] = $this->override_notice($object->get_limitations(false)->users->has_own_enabled());

		} // end if;

		$this->register_user_fields($sections, $object);

		$sections['post_types'] = array(
			'title'  => __('Post Types', 'wp-ultimo'),
			'desc'   => __('Control limitations imposed to the number of posts allowed for memberships attached to this product.', 'wp-ultimo'),
			'icon'   => 'dashicons-wu-book',
			'v-show' => "get_state_value('product_type', 'none') !== 'service'",
			'state'  => array(
				'limit_post_types' => $object->get_limitations()->post_types->is_enabled(),
			),
			'fields' => array(
				'modules[post_types][enabled]' => array(
					'type'      => 'toggle',
					'title'     => __('Limit Post Types', 'wp-ultimo'),
					'desc'      => __('Toggle this option to set limits to each post type.', 'wp-ultimo'),
					'value'     => false,
					'html_attr' => array(
						'v-model' => 'limit_post_types',
					),
				),
			),
		);

		if ($object->model !== 'product') {

			$sections['post_types']['fields']['post_quota_overwrite'] = $this->override_notice($object->get_limitations(false)->post_types->has_own_enabled());

		} // end if;

		$sections['post_types']['post_quota_note'] = array(
			'type'              => 'note',
			'desc'              => __('<strong>Note:</strong> Using the fields below you can set a post limit for each of the post types activated. <br>Toggle the switch to <strong>deactivate</strong> the post type altogether. Leave 0 or blank for unlimited posts.', 'wp-ultimo'),
			'wrapper_html_attr' => array(
				'v-show'  => 'limit_post_types',
				'v-cloak' => '1',
			),
		);

		$this->register_post_type_fields($sections, $object);

		$sections['limit_disk_space'] = array(
			'title'  => __('Disk Space', 'wp-ultimo'),
			'desc'   => __('Control limitations imposed to the disk space allowed for memberships attached to this entity.', 'wp-ultimo'),
			'icon'   => 'dashicons-wu-drive',
			'v-show' => "get_state_value('product_type', 'none') !== 'service'",
			'state'  => array(
				'limit_disk_space' => $object->get_limitations()->disk_space->is_enabled(),
			),
			'fields' => array(
				'modules[disk_space][enabled]' => array(
					'type'      => 'toggle',
					'title'     => __('Limit Disk Space per Site', 'wp-ultimo'),
					'desc'      => __('Enable disk space limitations for this entity.', 'wp-ultimo'),
					'value'     => true,
					'html_attr' => array(
						'v-model' => 'limit_disk_space',
					),
				),
			),
		);

		if ($object->model !== 'product') {

			$sections['limit_disk_space']['fields']['disk_space_modules_overwrite'] = $this->override_notice($object->get_limitations(false)->disk_space->has_own_enabled());

		} // end if;

		$sections['limit_disk_space']['fields']['modules[disk_space][limit]'] = array(
			'type'              => 'number',
			'title'             => __('Disk Space Allowance', 'wp-ultimo'),
			'desc'              => __('Set a limit in MBs for the disk space for <strong>each</strong> individual site.', 'wp-ultimo'),
			'min'               => 0,
			'placeholder'       => 100,
			'value'             => $object->get_limitations()->disk_space->get_limit(),
			'wrapper_html_attr' => array(
				'v-show'  => "get_state_value('product_type', 'none') !== 'service' && limit_disk_space",
				'v-cloak' => '1',
			),
		);

		if ($object->model !== 'product') {

			$sections['limit_disk_space']['fields']['disk_space_override'] = $this->override_notice($object->get_limitations(false)->disk_space->has_own_limit(), array('limit_disk_space'));

		} // end if;

		$sections['custom_domain'] = array(
			'title'  => __('Custom Domains', 'wp-ultimo'),
			'desc'   => __('Limit the number of users on each role, posts, pages, and more.', 'wp-ultimo'),
			'icon'   => 'dashicons-wu-link1',
			'v-show' => "get_state_value('product_type', 'none') !== 'service'",
			'state'  => array(
				'allow_domain_mapping' => $object->get_limitations()->domain_mapping->is_enabled(),
			),
			'fields' => array(
				'modules[domain_mapping][enabled]' => array(
					'type'              => 'toggle',
					'title'             => __('Allow Custom Domains', 'wp-ultimo'),
					'desc'              => __('Toggle this option on to allow this plan to enable custom domains for sign-ups on this plan.', 'wp-ultimo'),
					'value'             => $object->get_limitations()->domain_mapping->is_enabled(),
					'wrapper_html_attr' => array(
						'v-cloak' => '1',
					),
					'html_attr'         => array(
						'v-model' => 'allow_domain_mapping',
					),
				),
			),
		);

		if ($object->model !== 'product') {

			$sections['custom_domain']['fields']['custom_domain_override'] = $this->override_notice($object->get_limitations(false)->domain_mapping->has_own_enabled(), array('allow_domain_mapping'));

		} // end if;

		$sections['allowed_themes'] = array(
			'title'  => __('Themes', 'wp-ultimo'),
			'desc'   => __('Limit the number of users on each role, posts, pages, and more.', 'wp-ultimo'),
			'icon'   => 'dashicons-wu-palette',
			'v-show' => "get_state_value('product_type', 'none') !== 'service'",
			'state'  => array(
				'force_active_theme' => '',
			),
			'fields' => array(
				'themes' => array(
					'type'    => 'html',
					'title'   => __('Themes', 'wp-ultimo'),
					'desc'    => __('Select how the themes installed on the network should behave.', 'wp-ultimo'),
					'content' => fn() => $this->get_theme_selection_list($object, $sections['allowed_themes']),
				),
			),
		);

		$sections['allowed_plugins'] = array(
			'title'  => __('Plugins', 'wp-ultimo'),
			'desc'   => __('You can choose the behavior of each plugin installed on the platform.', 'wp-ultimo'),
			'icon'   => 'dashicons-wu-power-plug',
			'v-show' => "get_state_value('product_type', 'none') !== 'service'",
			'fields' => array(
				'plugins' => array(
					'type'    => 'html',
					'title'   => __('Plugins', 'wp-ultimo'),
					'desc'    => __('Select how the plugins installed on the network should behave.', 'wp-ultimo'),
					'content' => fn() => $this->get_plugin_selection_list($object),
				),
			),
		);

		$reset_url = wu_get_form_url('confirm_limitations_reset', array(
			'id'    => $object->get_id(),
			'model' => $object->model,
		));

		$sections['reset_limitations'] = array(
			'title'  => __('Reset Limitations', 'wp-ultimo'),
			'desc'   => __('Reset the limitations applied to this element.', 'wp-ultimo'),
			'icon'   => 'dashicons-wu-back-in-time',
			'fields' => array(
				'reset_permissions' => array(
					'type'  => 'note',
					'title' => sprintf("%s<span class='wu-normal-case wu-block wu-text-xs wu-font-normal wu-mt-1'>%s</span>", __('Reset Limitations', 'wp-ultimo'), __('Use this option to reset the custom limitations applied to this object.', 'wp-ultimo')),
					'desc'  => sprintf('<a href="%s" title="%s" class="wubox button-primary">%s</a>', $reset_url, __('Reset Limitations', 'wp-ultimo'), __('Reset Limitations', 'wp-ultimo')),
				),
			),
		);

		return $sections;

	} // end add_limitation_sections;

	/**
	 * Generates the override notice.
	 *
	 * @since 2.0.0
	 *
	 * @param boolean $show Wether or not to show the field.
	 * @param array   $additional_checks Array containing javascript conditions that need to be met.
	 * @return array
	 */
	protected function override_notice($show = false, $additional_checks = array()) {

		$text = sprintf('<p class="wu-m-0 wu-p-2 wu-bg-blue-100 wu-text-blue-600 wu-rounded">%s</p>', __('This value is being applied only to this entity. Changes made to the membership or product permissions will not affect this particular value.', 'wp-ultimo'));

		return array(
			'desc'              => $text,
			'type'              => 'note',
			'wrapper_classes'   => 'wu-pt-0',
			'wrapper_html_attr' => array(
				'v-show'  => ($additional_checks ? (implode(' && ', $additional_checks) . ' && ') : '') . var_export((bool) $show, true),
				'v-cloak' => '1',
				'style'   => 'border-top-width: 0 !important',
			),
		);

	} // end override_notice;

	/**
	 * Register the user roles fields
	 *
	 * @since 2.0.0
	 *
	 * @param array                                   $sections Sections and fields.
	 * @param \WP_Ultimo\Models\Trait\Trait_Limitable $object The object being edit.
	 * @return void
	 */
	public function register_user_fields(&$sections, $object) {

		$user_roles = get_editable_roles();

		$sections['users']['state']['roles'] = array();

		foreach ($user_roles as $user_role_slug => $user_role) {

			$sections['users']['state']['roles'][$user_role_slug] = $object->get_limitations()->users->{$user_role_slug};

			$sections['users']['fields']["control_{$user_role_slug}"] = array(
				'type'              => 'group',
				'title'             => sprintf(__('Limit %s Role', 'wp-ultimo'), $user_role['name']),
				'desc'              => sprintf(__('The customer will be able to create %s users(s) of this user role.', 'wp-ultimo'), "{{ roles['{$user_role_slug}'].enabled ? ( parseInt(roles['{$user_role_slug}'].number, 10) ? roles['{$user_role_slug}'].number : '" . __('unlimited', 'wp-ultimo') . "' ) : '" . __('no', 'wp-ultimo') . "' }}"),
				'tooltip'           => '',
				'wrapper_html_attr' => array(
					'v-bind:class' => "!roles['{$user_role_slug}'].enabled ? 'wu-opacity-75' : ''",
					'v-show'       => 'limit_users',
					'v-cloak'      => '1',
				),
				'fields'            => array(
					"modules[users][limit][{$user_role_slug}][number]" => array(
						'type'            => 'number',
						'placeholder'     => sprintf(__('%s Role Quota. e.g. 10', 'wp-ultimo'), $user_role['name']),
						'min'             => 0,
						'wrapper_classes' => 'wu-w-full',
						'html_attr'       => array(
							'v-model'         => "roles['{$user_role_slug}'].number",
							'v-bind:readonly' => "!roles['{$user_role_slug}'].enabled",
						),
					),
					"modules[users][limit][{$user_role_slug}][enabled]" => array(
						'type'            => 'toggle',
						'wrapper_classes' => 'wu-mt-1',
						'html_attr'       => array(
							'v-model' => "roles['{$user_role_slug}'].enabled",
						),
					),
				),
			);

			/*
			 * Add override notice.
			 */
			if ($object->model !== 'product') {

				$sections['users']['fields']["override_{$user_role_slug}"] = $this->override_notice($object->get_limitations(false)->users->exists($user_role_slug), array('limit_users'));

			} // end if;

		} // end foreach;

	} // end register_user_fields;

	/**
	 * Register the post type fields
	 *
	 * @since 2.0.0
	 *
	 * @param array                                   $sections Sections and fields.
	 * @param \WP_Ultimo\Models\Trait\Trait_Limitable $object The object being edit.
	 * @return void
	 */
	public function register_post_type_fields(&$sections, $object) {

		$post_types = get_post_types(array(), 'objects');

		$sections['post_types']['state']['types'] = array();

		foreach ($post_types as $post_type_slug => $post_type) {

			$sections['post_types']['state']['types'][$post_type_slug] = $object->get_limitations()->post_types->{$post_type_slug};

			$sections['post_types']['fields']["control_{$post_type_slug}"] = array(
				'type'              => 'group',
				'title'             => sprintf(__('Limit %s', 'wp-ultimo'), $post_type->label),
				'desc'              => sprintf(__('The customer will be able to create %s post(s) of this post type.', 'wp-ultimo'), "{{ types['{$post_type_slug}'].enabled ? ( parseInt(types['{$post_type_slug}'].number, 10) ? types['{$post_type_slug}'].number : '" . __('unlimited', 'wp-ultimo') . "' ) : '" . __('no', 'wp-ultimo') . "' }}"),
				'tooltip'           => '',
				'wrapper_html_attr' => array(
					'v-bind:class' => "!types['{$post_type_slug}'].enabled ? 'wu-opacity-75' : ''",
					'v-show'       => 'limit_post_types',
					'v-cloak'      => '1',
				),
				'fields'            => array(
					"modules[post_types][limit][{$post_type_slug}][number]" => array(
						'type'            => 'number',
						'placeholder'     => sprintf(__('%s Quota. e.g. 200', 'wp-ultimo'), $post_type->label),
						'min'             => 0,
						'wrapper_classes' => 'wu-w-full',
						'html_attr'       => array(
							'v-model'         => "types['{$post_type_slug}'].number",
							'v-bind:readonly' => "!types['{$post_type_slug}'].enabled",
						),
					),
					"modules[post_types][limit][{$post_type_slug}][enabled]" => array(
						'type'            => 'toggle',
						'wrapper_classes' => 'wu-mt-1',
						'html_attr'       => array(
							'v-model' => "types['{$post_type_slug}'].enabled",
						),
					),
				),
			);

			/*
			 * Add override notice.
			 */
			if ($object->model !== 'product') {

				$sections['post_types']['fields']["override_{$post_type_slug}"] = $this->override_notice($object->get_limitations(false)->post_types->exists($post_type_slug), array(
					'limit_post_types'
				));

			} // end if;

		} // end foreach;

	} // end register_post_type_fields;

	/**
	 * Returns the list of fields for the site tab.
	 *
	 * @since 2.0.0
	 *
	 * @param \WP_Ultimo\Models\Trait\Trait_Limitable $object The model being edited.
	 * @return array
	 */
	protected function get_sites_fields($object) {

		$fields = array(
			'modules[sites][enabled]' => array(
				'type'      => 'toggle',
				'title'     => __('Limit Sites', 'wp-ultimo'),
				'desc'      => __('Enable site limitations for this product.', 'wp-ultimo'),
				'value'     => $object->get_limitations()->sites->is_enabled(),
				'html_attr' => array(
					'v-model' => 'limit_sites'
				),
			),
		);

		if ($object->model !== 'product') {

			$fields['sites_overwrite'] = $this->override_notice($object->get_limitations(false)->sites->has_own_enabled());

		} // end if;

		/*
		 * Sites not supported on this type
		 */
		$fields['site_not_allowed_note'] = array(
			'type'              => 'note',
			'desc'              => __('The product type selection does not support allowing for the creating of extra sites.', 'wp-ultimo'),
			'tooltip'           => '',
			'wrapper_html_attr' => array(
				'v-show'  => "get_state_value('product_type', 'none') === 'service' && limit_sites",
				'v-cloak' => '1',
			),
		);

		$fields['modules[sites][limit]'] = array(
			'type'              => 'number',
			'min'               => 1,
			'title'             => __('Site Allowance', 'wp-ultimo'),
			'desc'              => __('This is the number of sites the customer will be able to create under this membership.', 'wp-ultimo'),
			'placeholder'       => 1,
			'value'             => $object->get_limitations()->sites->get_limit(),
			'wrapper_html_attr' => array(
				'v-show'  => "get_state_value('product_type', 'none') !== 'service' && limit_sites",
				'v-cloak' => '1',
			),
		);

		if ($object->model !== 'product') {

			$fields['sites_overwrite_2'] = $this->override_notice($object->get_limitations(false)->sites->has_own_limit(), array("get_state_value('product_type', 'none') !== 'service' && limit_sites"));

		} // end if;

		return apply_filters('wu_limitations_get_sites_fields', $fields, $object, $this);

	} // end get_sites_fields;

	/**
	 * Returns the HTML markup for the plugin selector list.
	 *
	 * @since 2.0.0
	 *
	 * @param \WP_Ultimo\Models\Trait\Trait_Limitable $object The model being edited.
	 * @return string
	 */
	public function get_plugin_selection_list($object) {

		$all_plugins = $this->get_all_plugins();

		return wu_get_template_contents('limitations/plugin-selector', array(
			'plugins' => $all_plugins,
			'object'  => $object,
		));

	} // end get_plugin_selection_list;

	/**
	 * Returns the HTML markup for the plugin selector list.
	 *
	 * @since 2.0.0
	 *
	 * @param \WP_Ultimo\Models\Trait\Trait_Limitable $object The model being edited.
	 * @param array                                   $section The section array.
	 * @return string
	 */
	public function get_theme_selection_list($object, &$section) {

		$all_themes = $this->get_all_themes();

		return wu_get_template_contents('limitations/theme-selector', array(
			'section' => $section,
			'themes'  => $all_themes,
			'object'  => $object,
		));

	} // end get_theme_selection_list;

	/**
	 * Returns a list of all plugins available as options, excluding WP Multisite WaaS.
	 *
	 * We also exclude a couple more.
	 *
	 * @since 2.0.0
	 * @return array
	 */
	public function get_all_plugins() {

		$all_plugins = get_plugins();

		$listed_plugins = array();

		foreach ($all_plugins as $plugin_path => $plugin_info) {

			if (wu_get_isset($plugin_info, 'Network') === true) {

				continue;

			} // end if;

			if (in_array($plugin_path, $this->plugin_exclusion_list(), true)) {

				continue;

			} // end if;

			$listed_plugins[$plugin_path] = $plugin_info;

		} // end foreach;

		return $listed_plugins;

	} // end get_all_plugins;
	/**
	 * Returns a list of all themes available as options, after filtering.
	 *
	 * @since 2.0.0
	 */
	public function get_all_themes(): array {

		$all_plugins = wp_get_themes();

		return array_filter($all_plugins, fn($path) => !in_array($path, $this->theme_exclusion_list(), true), ARRAY_FILTER_USE_KEY);

	} // end get_all_themes;

	/**
	 * Returns the exclusion list for plugins.
	 *
	 * We don't want people forcing WP Multisite WaaS to be deactivated, do we?
	 *
	 * @since 2.0.0
	 * @return array
	 */
	protected function plugin_exclusion_list() {

		$exclusion_list = array(
			'wp-ultimo/wp-ultimo.php',
			'user-switching/user-switching.php',
		);

		return apply_filters('wu_limitations_plugin_exclusion_list', $exclusion_list);

	} // end plugin_exclusion_list;

	/**
	 * Returns the exclusion list for themes.
	 *
	 * @since 2.0.0
	 * @return array
	 */
	protected function theme_exclusion_list() {

		$exclusion_list = array();

		return apply_filters('wu_limitations_theme_exclusion_list', $exclusion_list);

	} // end theme_exclusion_list;

} // end class Limitation_Manager;