/home/arranoyd/magicraft/wp-content/plugins/popup-maker/includes/class-pum-popup.php
<?php

// Exit if accessed directly
if ( ! defined( 'ABSPATH' ) ) {
	exit;
}

if ( ! class_exists( 'PUM_Popup' ) ) {

	/**
	 * Class PUM_Popup
	 *
	 * @since 1.4
	 */
	class PUM_Popup extends PUM_Post {

		/**
		 * @var string
		 */
		protected $required_post_type = 'popup';

		/**
		 * @var null
		 */
		public $cookies = null;

		/**
		 * @var null
		 */
		public $triggers = null;

		/**
		 * @var null
		 */
		public $conditions = null;

		/**
		 * @var array
		 */
		public $conditions_filtered = array();


		/**
		 * @var null
		 */
		public $display = null;

		/**
		 * @var null
		 */
		public $close = null;

		/**
		 * @var null
		 */
		public $theme_id = null;

		/**
		 * @var null
		 */
		public $data_attr = null;

		/**
		 * @var null
		 */
		public $title = null;

		/**
		 * @var null
		 */
		public $content = null;

		/**
		 * If no id is passed this will check for the current global id.
		 *
		 * @todo replace usage of popmake_get_the_popup_ID.
		 *
		 * @uses function `popmake_get_the_popup_ID`
		 *
		 * @param bool $_id
		 * @param array $_args
		 *
		 * return boolean $valid
		 */
		public function __construct( $_id = null, $_args = array() ) {
			if ( ! $_id ) {
				$_id = popmake_get_the_popup_ID();
			}

			return parent::__construct( $_id, $_args );
		}

		/**
		 * Returns the title of a popup.
		 *
		 * @uses filter `popmake_get_the_popup_title`
		 * @uses filter `pum_popup_get_title`
		 *
		 * @return string
		 */
		public function get_title() {
			if ( ! $this->title ) {
				$title = get_post_meta( $this->ID, 'popup_title', true );

				// Deprecated
				$this->title = apply_filters( 'popmake_get_the_popup_title', $title, $this->ID );
			}

			return apply_filters( 'pum_popup_get_title', $this->title, $this->ID );
		}

		/**
		 * Returns the content of a popup.
		 *
		 * @uses filter `the_popup_content`
		 * @uses filter `pum_popup_content`
		 *
		 * @return string
		 */
		public function get_content() {
			if ( ! $this->content ) {
				// Deprecated Filter
				$this->content = apply_filters( 'the_popup_content', $this->post_content, $this->ID );
			}

			return $this->content = apply_filters( 'pum_popup_content', $this->content, $this->ID );
		}

		/**
		 * Returns this popups theme id or the default id.
		 *
		 * @todo replace usage of popmake_get_default_popup_theme.
		 *
		 * @uses filter `popmake_get_the_popup_theme`
		 * @uses filter `pum_popup_get_theme_id`
		 *
		 * @return int $theme_id
		 */
		public function get_theme_id() {
			if ( ! $this->theme_id ) {
				$theme_id = get_post_meta( $this->ID, 'popup_theme', true );

				if ( ! $theme_id ) {
					$theme_id = popmake_get_default_popup_theme();
				}

				// Deprecated filter
				$this->theme_id = apply_filters( 'popmake_get_the_popup_theme', $theme_id, $this->ID );
			}

			return (int) apply_filters( 'pum_popup_get_theme_id', $this->theme_id, $this->ID );
		}

		/**
		 * Returns array of classes for this popup.
		 *
		 * @uses filter `popmake_get_the_popup_classes`
		 * @uses filter `pum_popup_get_classes`
		 *
		 * @param string $element The key or html element identifier.
		 *
		 * @return array $classes
		 */
		function get_classes( $element = 'overlay' ) {
			$classes = array(
				'overlay'   => array(
					'pum',
					'pum-overlay',
					'pum-theme-' . $this->get_theme_id(),
					'popmake-overlay', // Backward Compatibility
				),
				'container' => array(
					'pum-container',
					'popmake', // Backward Compatibility
					'theme-' . $this->get_theme_id(), // Backward Compatibility
				),
				'title'     => array(
					'pum-title',
					'popmake-title', // Backward Compatibility
				),
				'content'   => array(
					'pum-content',
					'popmake-content', // Backward Compatibility
				),
				'close'     => array(
					'pum-close',
					'popmake-close' // Backward Compatibility
				),
			);


			$size = $this->get_display( 'size' );
			if ( in_array( $size, array( 'nano', 'micro', 'tiny', 'small', 'medium', 'normal', 'large', 'xlarge' ) ) ) {
				$classes['container'][] = 'pum-responsive';
				$classes['container'][] = 'pum-responsive-' . $size;
				$classes['container'][] = 'responsive'; // Backward Compatibility
				$classes['container'][] = 'size-' . $size; // Backward Compatibility
			} elseif ( $size == 'custom' ) {
				$classes['container'][] = 'size-custom'; // Backward Compatibility
			}

			if ( ! $this->get_display( 'custom_height_auto' ) && $this->get_display( 'scrollable_content' ) ) {
				$classes['container'][] = 'pum-scrollable';
				$classes['container'][] = 'scrollable'; // Backward Compatibility
			}

			if ( $this->get_display( 'position_fixed' ) ) {
				$classes['container'][] = 'pum-position-fixed';
			}

			if ( $this->get_display( 'overlay_disabled' ) ) {
				$classes['overlay'][] = 'pum-overlay-disabled';
			}

			// Add a class for each trigger type.
			foreach ( $this->get_triggers() as $trigger ) {
				if ( ! in_array( $trigger['type'], $classes['overlay'] ) ) {
					$classes['overlay'][] = $trigger['type'];
				}
			}

			if ( is_singular( 'popup' ) ) {
				$classes['overlay'][] = 'pum-preview';
			}

			// Deprecated: applies old classes to the new overlay
			if ( $element == 'container' ) {
				$classes['container'] = apply_filters( 'popmake_get_the_popup_classes', $classes['container'], $this->ID );
			}

			$classes = apply_filters( 'pum_popup_get_classes', $classes, $this->ID );

			if ( ! isset( $classes[ $element ] ) ) {
				return array();
			}

			return apply_filters( "pum_popup_get_{$element}_classes", $classes[ $element ], $this->ID );
		}

		/**
		 * @return mixed|void
		 */
		function get_cookies() {
			if ( ! $this->cookies ) {
				$this->cookies = get_post_meta( $this->ID, 'popup_cookies', true );

				if ( ! $this->cookies ) {
					$this->cookies = array();
				}
			}

			return apply_filters( 'pum_popup_get_cookies', $this->cookies, $this->ID );
		}

		/**
		 * @return mixed|void
		 */
		function get_triggers() {
			if ( ! $this->triggers ) {
				$this->triggers = get_post_meta( $this->ID, 'popup_triggers', true );

				if ( ! $this->triggers ) {
					$this->triggers = array();
				}

				if ( ! is_admin() && ! ( defined( 'DOING_AJAX' ) && DOING_AJAX ) ) {
					$has_click_trigger = false;

					foreach ( $this->triggers as $trigger ) {
						if ( $trigger['type'] == 'click_open' ) {
							$has_click_trigger = true;
						}
					}

					if ( ! $has_click_trigger ) {
						$this->triggers[] = array(
							'type'     => 'click_open',
							'settings' => array(
								'extra_selectors' => '',
								'cookie'          => array(
									'name' => null,
								),
							),
						);
					}
				}

			}

			return apply_filters( 'pum_popup_get_triggers', $this->triggers, $this->ID );
		}

		/**
		 * Returns all or single display settings.
		 *
		 * @param null $key
		 *
		 * @return mixed
		 */
		function get_display( $key = null ) {
			if ( ! $this->display ) {
				$display_values = get_post_meta( $this->ID, 'popup_display', true );

				if ( ! $display_values ) {
					$display_values = apply_filters( "pum_popup_display_defaults", array() );
				}

				$this->display = $display_values;
			}

			$values = apply_filters( 'pum_popup_get_display', $this->display, $this->ID );

			if ( ! $key ) {
				return $values;
			}

			$value = isset ( $values[ $key ] ) ? $values[ $key ] : null;

			return apply_filters( 'pum_popup_get_display_' . $key, $value, $this->ID );
		}

		/**
		 * Returns all or single close settings.
		 *
		 * @param null $key
		 *
		 * @return mixed
		 */
		function get_close( $key = null ) {
			if ( ! $this->close ) {
				$close_values = get_post_meta( $this->ID, 'popup_close', true );

				if ( ! $close_values ) {
					$close_values = apply_filters( "pum_popup_close_defaults", array() );
				}

				$this->close = $close_values;
			}

			$values = apply_filters( 'pum_popup_get_close', $this->close, $this->ID );

			if ( ! $key ) {
				return $values;
			}

			$value = isset ( $values[ $key ] ) ? $values[ $key ] : null;

			return apply_filters( 'pum_popup_get_close_' . $key, $value, $this->ID );

		}

		/**
		 * Returns array for data attribute of this popup.
		 *
		 * @todo integrate popmake_clean_popup_data_attr
		 *
		 * @uses deprecated filter `popmake_get_the_popup_data_attr`
		 * @uses filter `pum_popup_get_data_attr`
		 *
		 * @return array
		 */
		function get_data_attr() {
			if ( ! $this->data_attr ) {

				$data_attr = array(
					'id'              => $this->ID,
					'slug'            => $this->post_name,
					'theme_id'        => $this->get_theme_id(),
					'cookies'         => $this->get_cookies(),
					'triggers'        => $this->get_triggers(),
					'mobile_disabled' => $this->mobile_disabled() ? true : null,
					'tablet_disabled' => $this->tablet_disabled() ? true : null,
					'meta'            => array(
						'display'    => $this->get_display(),
						'close'      => $this->get_close(),
						// Added here for backward compatibility in extensions.
						'click_open' => popmake_get_popup_click_open( $this->ID ),
					),
				);

				$filters = array( 'js_only' => true );

				if ( $this->has_conditions( $filters ) ) {
					$data_attr['conditions'] = $this->get_conditions( $filters );
				}

				// Deprecated
				$this->data_attr = apply_filters( 'popmake_get_the_popup_data_attr', $data_attr, $this->ID );
			}

			return apply_filters( 'pum_popup_get_data_attr', $this->data_attr, $this->ID );
		}

		/**
		 * Returns the close button text.
		 * @return mixed|void
		 */
		public function close_text() {
			$text = '&#215;';

			/** @deprecated */
			$text = apply_filters( 'popmake_popup_default_close_text', $text, $this->ID );

			// Check to see if popup has close text to over ride default.
			$popup_close_text = $this->get_close( 'text' );
			if ( $popup_close_text && $popup_close_text != '' ) {
				$text = $popup_close_text;
			} else {
				// todo replace this with PUM_Theme class in the future.
				$theme_text = popmake_get_popup_theme_close( $this->get_theme_id(), 'text', false );
				if ( $theme_text && $theme_text != '' ) {
					$text = $theme_text;
				}
			}

			return apply_filters( 'pum_popup_close_text', $text, $this->ID );
		}

		/**
		 * Returns true if the close button should be rendered.
		 *
		 * @uses apply_filters `popmake_show_close_button`
		 * @uses apply_filters `pum_popup_show_close_button`
		 *
		 * @return bool
		 */
		public function show_close_button() {
			// Deprecated filter.
			$show = apply_filters( 'popmake_show_close_button', true, $this->ID );

			return boolval( apply_filters( 'pum_popup_show_close_button', $show, $this->ID ) );
		}

		public function mobile_disabled() {
			$mobile_disabled = $this->get_meta( 'popup_mobile_disabled', true );

			return (bool) apply_filters( 'pum_popup_mobile_disabled', $mobile_disabled, $this->ID );
		}

		public function tablet_disabled() {
			$tablet_disabled = $this->get_meta( 'popup_tablet_disabled', true );

			return (bool) apply_filters( 'pum_popup_tablet_disabled', $tablet_disabled, $this->ID );
		}

		/**
		 * @param array $filters
		 *
		 * @return mixed|void
		 */
		public function get_conditions( $filters = array() ) {

			$filters = wp_parse_args( $filters, array(
				'php_only' => null,
				'js_only'  => null,
			) );

			$cache_key = hash( 'md5', json_encode( $filters ) );

			if ( ! $this->conditions ) {
				$this->conditions = get_post_meta( $this->ID, 'popup_conditions', true );

				if ( ! $this->conditions ) {
					$this->conditions = array();
				}
			}

			// Check if these exclusion filters have already been applied and prevent extra processing.
			$conditions = isset( $this->conditions_filtered[ $cache_key ] ) ? $this->conditions_filtered[ $cache_key ] : false;

			if ( ! $conditions ) {
				$conditions = $this->conditions;
				// Sanity Check on the values not operand value.
				foreach ( $conditions as $group_key => $group ) {

					foreach ( $group as $key => $condition ) {

						if ( $this->exclude_condition( $condition, $filters ) ) {
							unset( $conditions[ $group_key ][ $key ] );
							if ( empty( $conditions[ $group_key ] ) ) {
								unset( $conditions[ $group_key ] );
								break;
							}
							continue;
						}

						$conditions[ $group_key ][ $key ] = $this->parse_condition( $condition );
					}

					if ( ! empty( $conditions[ $group_key ] ) ) {
						// Renumber each subarray.
						$conditions[ $group_key ] = array_values( $conditions[ $group_key ] );
					}

				}

				// Renumber top arrays.
				$conditions = array_values( $conditions );

				$this->conditions_filtered[ $cache_key ] = $conditions;
			}


			return apply_filters( 'pum_popup_get_conditions', $conditions, $this->ID, $filters );
		}

		public function parse_condition( $condition ) {
			// The not operand value is missing, set it to false.
			if ( ! isset ( $condition['not_operand'] ) ) {
				$condition['not_operand'] = false;
			}

			return $condition;
		}

		public function exclude_condition( $condition, $filters = array() ) {

			$exclude = false;

			// The condition target doesn't exist. Lets ignore this condition.
			if ( empty( $condition['target'] ) ) {
				return true;
			}

			$condition_args = PUM_Conditions::instance()->get_condition( $condition['target'] );

			// The condition target doesn't exist. Lets ignore this condition.
			if ( ! $condition_args ) {
				return true;
			}


			if ( $filters['js_only'] && ! $condition_args->is_advanced() ) {
				return true;
			} elseif ( $filters['php_only'] && $condition_args->is_advanced() ) {
				return true;
			}

			return $exclude;
		}

		/**
		 * @return mixed|void
		 */
		public function has_conditions( $filters = array() ) {
			return boolval( count( $this->get_conditions( $filters ) ) );
		}

		/**
		 * Returns whether or not the popup is visible in the loop.
		 *
		 * @return bool
		 */
		public function is_loadable() {

			// Loadable defaults to true if no conditions. Making the popup available everywhere.
			$loadable = true;

			if ( ! $this->ID ) {
				return false;
				// Published/private
			}

			$filters = array( 'php_only' => true );

			if ( $this->has_conditions( $filters ) ) {

				// All Groups Must Return True. Break if any is false and set $loadable to false.
				foreach ( $this->get_conditions( $filters ) as $group => $conditions ) {

					// Groups are false until a condition proves true.
					$group_check = false;

					// At least one group condition must be true. Break this loop if any condition is true.
					foreach ( $conditions as $condition ) {

						// If any condition passes, set $group_check true and break.
						if ( ! $condition['not_operand'] && $this->check_condition( $condition ) ) {
							$group_check = true;
							break;
						} elseif ( $condition['not_operand'] && ! $this->check_condition( $condition ) ) {
							$group_check = true;
							break;
						}

					}

					// If any group of conditions doesn't pass, popup is not loadable.
					if ( ! $group_check ) {
						$loadable = false;
					}

				}

			}

			return apply_filters( 'pum_popup_is_loadable', $loadable, $this->ID );
		}

		public function check_condition( $settings = array() ) {
			$condition = PUM_Conditions::instance()->get_condition( $settings['target'] );

			if ( ! $condition ) {
				return false;
			}

			return call_user_func( $condition->get_callback(), $settings, $this );
		}

		public function get_meta( $key = null, $single = false ) {
			return get_post_meta( $this->ID, $key, $single );
		}

		/**
		 * Returns the correct open count meta key based on $which.
		 *
		 * Can be current or total.
		 *
		 * - 'current' since last reset
		 * - 'total' since creation
		 *
		 * @param string $which
		 *
		 * @return integer
		 */
		public function get_open_count( $which = 'current' ) {
			switch ( $which ) {
				case 'current' :
					return absint( $this->get_meta( 'popup_open_count', true ) );
				case 'total'   :
					return absint( $this->get_meta( 'popup_open_count_total', true ) );
			}

			return 0;
		}

		public function increase_open_count() {

			// Set the current open count
			$current = $this->get_open_count();
			if ( ! $current ) {
				$current = 0;
			};
			$current = $current + 1;

			// Set the total open count since creation.
			$total = $this->get_open_count( 'total' );
			if ( ! $total ) {
				$total = 0;
			}
			$total = $total + 1;

			update_post_meta( $this->ID, 'popup_open_count', absint( $current ) );
			update_post_meta( $this->ID, 'popup_open_count_total', absint( $total ) );
			update_post_meta( $this->ID, 'popup_last_opened', current_time( 'timestamp', 0 ) );

			$total_opens = get_option( 'pum_total_open_count', 0 );

			$total_opens ++;

			update_option( 'pum_total_open_count', $total_opens );

			// If is multisite add this blogs total to the site totals.
			if ( function_exists( 'is_multisite' ) && is_multisite() ) {
				$site_total_open_count = get_site_option( 'pum_site_total_open_count', false );
				if ( ! $site_total_open_count ) {
					$site_total_open_count = $total_opens;
				} else {
					$site_total_open_count ++;
				}
				update_site_option( 'pum_site_total_open_count', $site_total_open_count );
			}

		}

		/**
		 * Log and reset popup open count to 0.
		 */
		public function reset_open_count() {

			// Log the reset time and count.
			add_post_meta( $this->ID, 'popup_open_count_reset', array(
				'timestamp' => current_time( 'timestamp', 0 ),
				'count'     => absint( $this->get_open_count( 'current' ) ),
			) );

			update_post_meta( $this->ID, 'popup_open_count', 0 );
			update_post_meta( $this->ID, 'popup_last_opened', 0 );

		}

		/**
		 * Returns the last reset information.
		 *
		 * @return mixed
		 */
		public function get_last_open_count_reset() {
			$resets = $this->get_meta( 'popup_open_count_reset' );

			if ( empty ( $resets ) ) {
				return false;
			}

			/**
			 * Compare function for reset timestamps. Sorts Newest First.
			 *
			 * @param $a
			 * @param $b
			 *
			 * @return bool
			 */
			function cmp( $a, $b ) {
				return ( float ) $a['timestamp'] < ( float ) $b['timestamp'];
			}

			usort( $resets, "cmp" );

			return $resets[0];
		}

	}

}