fbinfobanner.php 7.44 KB
<?php
/**
 * Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved
 *
 * This source code is licensed under the license found in the
 * LICENSE file in the root directory of this source tree.
 *
 * @package FacebookCommerce
 */

if ( ! defined( 'ABSPATH' ) ) {
	exit;
}

if ( ! class_exists( 'WC_Facebookcommerce_Utils' ) ) {
	include_once 'includes/fbutils.php';
}

if ( ! class_exists( 'WC_Facebookcommerce_Info_Banner' ) ) :

	/**
	 * FB Info Banner class
	 */
	class WC_Facebookcommerce_Info_Banner {

		const FB_NO_TIP_EXISTS           = 'No Tip Exist!';
		const DEFAULT_TIP_IMG_URL_PREFIX = 'https://www.facebook.com';
		const CHANNEL_ID                 = 2087541767986590;

		/** @var object Class Instance */
		private static $instance;

		/** @var string If the banner has been dismissed */
		private $external_merchant_settings_id;
		private $fbgraph;
		private $should_query_tip;

		/**
		 * Get the class instance
		 */
		public static function get_instance(
		$external_merchant_settings_id,
		$fbgraph,
		$should_query_tip = false ) {
			return null === self::$instance
			? ( self::$instance = new self(
				$external_merchant_settings_id,
				$fbgraph,
				$should_query_tip
			) )
			: self::$instance;
		}

		/**
		 * Constructor
		 */
		public function __construct(
		$external_merchant_settings_id,
		$fbgraph,
		$should_query_tip = false ) {
			$this->should_query_tip              = $should_query_tip;
			$this->external_merchant_settings_id = $external_merchant_settings_id;
			$this->fbgraph                       = $fbgraph;
			add_action( 'wp_ajax_ajax_woo_infobanner_post_click', array( $this, 'ajax_woo_infobanner_post_click' ) );
			add_action( 'wp_ajax_ajax_woo_infobanner_post_xout', array( $this, 'ajax_woo_infobanner_post_xout' ) );
			add_action( 'admin_notices', array( $this, 'banner' ) );
			add_action( 'admin_init', array( $this, 'dismiss_banner' ) );
		}

		/**
		 * Post click event when hit primary button.
		 */
		function ajax_woo_infobanner_post_click() {
			WC_Facebookcommerce_Utils::check_woo_ajax_permissions(
				'post tip click event',
				true
			);
			check_ajax_referer( 'wc_facebook_infobanner_jsx' );
			$tip_info = WC_Facebookcommerce_Utils::get_cached_best_tip();
			$tip_id   = isset( $tip_info->tip_id )
			? $tip_info->tip_id
			: null;
			if ( $tip_id == null ) {
				WC_Facebookcommerce_Utils::fblog(
					'Do not have tip id when click, sth went wrong',
					array( 'tip_info' => $tip_info ),
					true
				);
			} else {
				WC_Facebookcommerce_Utils::tip_events_log(
					$tip_id,
					self::CHANNEL_ID,
					'click'
				);
			}
		}

		/**
		 * Post xout event when hit dismiss button.
		 */
		function ajax_woo_infobanner_post_xout() {
			WC_Facebookcommerce_Utils::check_woo_ajax_permissions(
				'post tip xout event',
				true
			);
			check_ajax_referer( 'wc_facebook_infobanner_jsx' );
			$tip_info = WC_Facebookcommerce_Utils::get_cached_best_tip();
			$tip_id   = isset( $tip_info->tip_id )
			? $tip_info->tip_id
			: null;
			// Delete cached tip if xout.
			update_option( 'fb_info_banner_last_best_tip', '' );
			if ( $tip_id == null ) {
				WC_Facebookcommerce_Utils::fblog(
					'Do not have tip id when xout, sth went wrong',
					array( 'tip_info' => $tip_info ),
					true
				);
			} else {
				WC_Facebookcommerce_Utils::tip_events_log(
					$tip_id,
					self::CHANNEL_ID,
					'xout'
				);
			}
		}

		/**
		 * Display a info banner on Woocommerce pages.
		 */
		public function banner() {
			$screen = get_current_screen();
			if ( ! in_array(
				$screen->base,
				array(
					'woocommerce_page_wc-reports',
					'woocommerce_page_wc-settings',
					'woocommerce_page_wc-status',
				)
			) ||
			$screen->is_network || $screen->action ) {
				return;
			}

			$tip_info = null;
			if ( ! $this->should_query_tip ) {
				// If last query is less than 1 day, either has last best tip or default
				// tip pass time cap.
				$tip_info = WC_Facebookcommerce_Utils::get_cached_best_tip();
			} else {
				$tip_info = $this->fbgraph->get_tip_info(
					$this->external_merchant_settings_id
				);
				update_option( 'fb_info_banner_last_query_time', current_time( 'mysql' ) );
			}

			// Not render if no cached best tip, or no best tip returned from FB.
			if ( ! $tip_info || ( $tip_info === self::FB_NO_TIP_EXISTS ) ) {
				// Delete cached tip if should query and get no tip.
				delete_option( 'fb_info_banner_last_best_tip' );
				return;
			} else {
				// Get tip creatives via API
				if ( is_string( $tip_info ) ) {
					$tip_info = WC_Facebookcommerce_Utils::decode_json( $tip_info );
				}
				$tip_title = isset( $tip_info->tip_title->__html )
				? $tip_info->tip_title->__html
				: null;

				$tip_body = isset( $tip_info->tip_body->__html )
				? $tip_info->tip_body->__html
				: null;

				$tip_action_link = isset( $tip_info->tip_action_link )
				? $tip_info->tip_action_link
				: null;

				$tip_action = isset( $tip_info->tip_action->__html )
				? $tip_info->tip_action->__html
				: null;

				$tip_img_url = isset( $tip_info->tip_img_url )
				? self::DEFAULT_TIP_IMG_URL_PREFIX . $tip_info->tip_img_url
				: null;

				if ( $tip_title == null || $tip_body == null || $tip_action_link == null
				|| $tip_action == null || $tip_action == null ) {
					WC_Facebookcommerce_Utils::fblog(
						'Unexpected response from FB for tip info.',
						array( 'tip_info' => $tip_info ),
						true
					);
					return;
				}
				update_option(
					'fb_info_banner_last_best_tip',
					is_object( $tip_info ) || is_array( $tip_info )
					? json_encode( $tip_info ) : $tip_info
				);
			}

			$dismiss_url = $this->dismiss_url();
			echo '<div class="updated fade"><div id="fbinfobanner"><div><img src="' . $tip_img_url .
			'" class="iconDetails"></div><p class = "tipTitle">' .
			__( '<strong>' . $tip_title . '</strong>', 'facebook-for-woocommerce' ) . "\n";
			echo '<p class = "tipContent">' .
			__( $tip_body, 'facebook-for-woocommerce' ) . '</p>';
			echo '<p class = "tipButton"><a href="' . $tip_action_link . '" class = "btn" onclick="fb_woo_infobanner_post_click(); return true;" title="' .
			__( 'Click and redirect.', 'facebook-for-woocommerce' ) .
			'"> ' . __( $tip_action, 'facebook-for-woocommerce' ) . '</a>' .
			'<a href="' . esc_url( $dismiss_url ) . '" class = "btn dismiss grey" onclick="fb_woo_infobanner_post_xout(); return true;" title="' .
			__( 'Dismiss this notice.', 'facebook-for-woocommerce' ) .
			'"> ' . __( 'Dismiss', 'facebook-for-woocommerce' ) . '</a></p></div></div>';
		}

		/**
		 * Returns the url that the user clicks to remove the info banner
		 *
		 * @return (string)
		 */
		private function dismiss_url() {
			$url = admin_url( 'admin.php' );

			$url = add_query_arg(
				array(
					'page'      => 'wc-settings',
					'tab'       => 'integration',
					'wc-notice' => 'dismiss-fb-info-banner',
				),
				$url
			);

			return wp_nonce_url( $url, 'woocommerce_info_banner_dismiss' );
		}

		/**
		 * Handles the dismiss action so that the banner can be permanently hidden
		 * during time threshold
		 */
		public function dismiss_banner() {
			if ( ! isset( $_GET['wc-notice'] ) ) {
				return;
			}

			if ( 'dismiss-fb-info-banner' !== $_GET['wc-notice'] ) {
				return;
			}

			if ( ! check_admin_referer( 'woocommerce_info_banner_dismiss' ) ) {
				return;
			}

			// Delete cached tip if xout.
			delete_option( 'fb_info_banner_last_best_tip' );
			if ( wp_get_referer() ) {
				wp_safe_redirect( wp_get_referer() );
			} else {
				wp_safe_redirect( admin_url( 'admin.php?page=wc-settings&tab=integration' ) );
			}
		}
	}

endif;