diff options
author | Anthony G. Basile <blueness@gentoo.org> | 2018-03-10 19:17:40 -0500 |
---|---|---|
committer | Anthony G. Basile <blueness@gentoo.org> | 2018-03-10 19:17:40 -0500 |
commit | 1ede1db458d07b50cfede5937958cb20752df616 (patch) | |
tree | b7484d24649fb07b8a591148ada617b14d8bbc6d /plugins/jetpack/modules/woocommerce-analytics | |
parent | Update akismet 4.0.3 (diff) | |
download | blogs-gentoo-1ede1db458d07b50cfede5937958cb20752df616.tar.gz blogs-gentoo-1ede1db458d07b50cfede5937958cb20752df616.tar.bz2 blogs-gentoo-1ede1db458d07b50cfede5937958cb20752df616.zip |
Update jetpack 5.9
Signed-off-by: Anthony G. Basile <blueness@gentoo.org>
Diffstat (limited to 'plugins/jetpack/modules/woocommerce-analytics')
3 files changed, 475 insertions, 0 deletions
diff --git a/plugins/jetpack/modules/woocommerce-analytics/classes/wp-woocommerce-analytics-universal.php b/plugins/jetpack/modules/woocommerce-analytics/classes/wp-woocommerce-analytics-universal.php new file mode 100644 index 00000000..9745b91c --- /dev/null +++ b/plugins/jetpack/modules/woocommerce-analytics/classes/wp-woocommerce-analytics-universal.php @@ -0,0 +1,328 @@ +<?php +/** + * Jetpack_WooCommerce_Analytics_Universal + * + * @package Jetpack + * @author Automattic + */ + +/** + * Bail if accessed directly + */ +if ( ! defined( 'ABSPATH' ) ) { + exit; +} + +/** + * Class Jetpack_WooCommerce_Analytics_Universal + * Filters and Actions added to Store pages to perform analytics + */ +class Jetpack_WooCommerce_Analytics_Universal { + /** + * Jetpack_WooCommerce_Analytics_Universal constructor. + */ + public function __construct() { + // loading _wca + add_action( 'wp_head', array( $this, 'wp_head_top' ), 1 ); + + // loading s.js + add_action( 'wp_head', array( $this, 'wp_head_bottom' ), 999999 ); + + // single product page view + add_action( 'woocommerce_after_single_product', array( $this, 'product_detail' ) ); + + // add to cart on single product page + add_action( 'woocommerce_after_add_to_cart_button', array( $this, 'add_to_cart' ) ); + + // add to carts from list views (search, store etc.) + add_action( 'wp_footer', array( $this, 'loop_add_to_cart' ) ); + + + add_action( 'woocommerce_after_cart', array( $this, 'remove_from_cart' ) ); + add_action( 'woocommerce_after_mini_cart', array( $this, 'remove_from_cart' ) ); + add_action( 'wcct_before_cart_widget', array( $this, 'remove_from_cart' ) ); + add_filter( 'woocommerce_cart_item_remove_link', array( $this, 'remove_from_cart_attributes' ), 10, 2 ); + + // cart checkout + add_action( 'woocommerce_after_checkout_form', array( $this, 'checkout_process' ) ); + + // order confirmed + add_action( 'woocommerce_thankyou', array( $this, 'order_process' ), 10, 1 ); + add_action( 'woocommerce_after_cart', array( $this, 'remove_from_cart_via_quantity' ), 10, 1 ); + } + + /** + * Make _wca available to queue events + */ + public function wp_head_top() { + if ( is_cart() || is_checkout() || is_checkout_pay_page() || is_order_received_page() || is_add_payment_method_page() ) { + $prevent_referrer_code = "<script>window._wca_prevent_referrer = true;</script>"; + echo "$prevent_referrer_code\r\n"; + } + $wca_code = "<script>window._wca = window._wca || [];</script>"; + echo "$wca_code\r\n"; + } + + + /** + * Place script to call s.js, Store Analytics + */ + public function wp_head_bottom() { + $filename = 's-' . gmdate( 'YW' ) . '.js'; + $async_code = "<script async src='https://stats.wp.com/" . $filename . "'></script>"; + echo "$async_code\r\n"; + } + + /** + * On a product page, add a click event listener to "Add to Cart" button click + */ + public function add_to_cart() { + + if ( ! is_single() ) { + return; + } + + $blogid = Jetpack::get_option( 'id' ); + global $product; + + wc_enqueue_js( + "jQuery( '" . esc_js( '.single_add_to_cart_button' ) . "' ).click( function() { + _wca.push( { + '_en': 'woocommerceanalytics_add_to_cart', + 'blog_id': " . esc_js( $blogid ) . ", + 'pi': '" . esc_js( $product->get_id() ) . "', + 'pn' : '" . esc_js( $product->get_title() ) . "', + 'pq': jQuery( 'input.qty' ).val() ? jQuery( 'input.qty' ).val() : '1', + 'ui': '" . esc_js( $this->get_user_id() ) . "', + } ); + } );" + ); + } + + /** + * On product lists or other non-product pages, add an event listener to "Add to Cart" button click + */ + public function loop_add_to_cart() { + $blogid = Jetpack::get_option( 'id' ); + $selector = '.add_to_cart_button:not(.product_type_variable, .product_type_grouped)'; + + wc_enqueue_js( + "jQuery( '" . esc_js( $selector ) . "' ).click( function() { + var productID = jQuery( this ).data( 'product_id' ); + var productDetails = { + 'id': productID, + 'quantity': jQuery( this ).data( 'quantity' ), + }; + _wca.push( { + '_en': 'woocommerceanalytics_product_view', + 'blog_id': '" . esc_js( $blogid ) . "', + 'pi': productDetails.id, + 'ui': '" . esc_js( $this->get_user_id() ) . "', + } ); + _wca.push( { + '_en': 'woocommerceanalytics_add_to_cart', + 'blog_id': " . esc_js( $blogid ) . ", + 'pi': productDetails.id, + 'pq': productDetails.quantity, + 'ui': '" . esc_js( $this->get_user_id() ) . "', + } ); + } );" + ); + } + + /** + * On the cart page, add an event listener for removal of product click + */ + public function remove_from_cart() { + + // We listen at div.woocommerce because the cart 'form' contents get forcibly + // updated and subsequent removals from cart would then not have this click + // handler attached. + $blogid = Jetpack::get_option( 'id' ); + wc_enqueue_js( + "jQuery( 'div.woocommerce' ).on( 'click', 'a.remove', function() { + var productID = jQuery( this ).data( 'product_id' ); + var quantity = jQuery( this ).parent().parent().find( '.qty' ).val() + var productDetails = { + 'id': productID, + 'quantity': quantity ? quantity : '1', + }; + _wca.push( { + '_en': 'woocommerceanalytics_remove_from_cart', + 'blog_id': '" . esc_js( $blogid ) . "', + 'pi': productDetails.id, + 'pq': productDetails.quantity, + 'ui': '" . esc_js( $this->get_user_id() ) . "', + } ); + } );" + ); + } + + /** + * Adds the product ID to the remove product link (for use by remove_from_cart above) if not present + * + * @param string $url url. + * @param string $key key. + * @return mixed. + */ + public function remove_from_cart_attributes( $url, $key ) { + if ( false !== strpos( $url, 'data-product_id' ) ) { + return $url; + } + + $item = WC()->cart->get_cart_item( $key ); + $product = $item['data']; + + $new_attributes = sprintf( + 'href="%s" data-product_id="%s" data-product_sku="%s"', + esc_attr( $url ), + esc_attr( $product->get_id() ), + esc_attr( $product->get_sku() ) + ); + $url = str_replace( 'href=', $new_attributes ); + return $url; + } + + /** + * Gather relevant product information + * + * @param array $product product + * @return array + */ + public function get_item_details( $product ) { + return array( + 'id' => $product->get_id(), + 'name' => $product->get_title(), + 'category' => Jetpack_WooCommerce_Analytics_Utils::get_product_categories_concatenated( $product ), + 'price' => $product->get_price(), + ); + } + + /** + * Track a product page view + */ + public function product_detail() { + + global $product; + $blogid = Jetpack::get_option( 'id' ); + + $item_details = $this->get_item_details( $product ); + + wc_enqueue_js( + "_wca.push( { + '_en': 'woocommerceanalytics_product_view', + 'blog_id': '" . esc_js( $blogid ) . "', + 'pi': '" . esc_js( $item_details['id'] ) . "', + 'pn': '" . esc_js( $item_details['name'] ) . "', + 'pc': '" . esc_js( $item_details['category'] ) . "', + 'pp': '" . esc_js( $item_details['price'] ) . "', + 'ui': '" . esc_js( $this->get_user_id() ) . "', + } );" + ); + } + + /** + * On the Checkout page, trigger an event for each product in the cart + */ + public function checkout_process() { + + $universal_commands = array(); + $cart = WC()->cart->get_cart(); + $blogid = Jetpack::get_option( 'id' ); + + foreach ( $cart as $cart_item_key => $cart_item ) { + /** + * This filter is already documented in woocommerce/templates/cart/cart.php + */ + $product = apply_filters( 'woocommerce_cart_item_product', $cart_item['data'], $cart_item, $cart_item_key ); + + $item_details = $this->get_item_details( $product ); + + $universal_commands[] = "_wca.push( { + '_en': 'woocommerceanalytics_product_checkout', + 'blog_id': '" . esc_js( $blogid ) . "', + 'pi': '" . esc_js( $item_details['id'] ) . "', + 'pn': '" . esc_js( $item_details['name'] ) . "', + 'pc': '" . esc_js( $item_details['category'] ) . "', + 'pp': '" . esc_js( $item_details['price'] ) . "', + 'pq': '" . esc_js( $cart_item['quantity'] ) . "', + 'ui': '" . esc_js( $this->get_user_id() ) . "', + } );"; + } + + wc_enqueue_js( implode( "\r\n", $universal_commands ) ); + } + + /** + * After the checkout process, fire an event for each item in the order + * + * @param string $order_id Order Id. + */ + public function order_process( $order_id ) { + $order = wc_get_order( $order_id ); + $universal_commands = array(); + $blogid = Jetpack::get_option( 'id' ); + + // loop through products in the order and queue a purchase event. + foreach ( $order->get_items() as $order_item_id => $order_item ) { + $product = $order->get_product_from_item( $order_item ); + + $item_details = $this->get_item_details( $product ); + + $universal_commands[] = "_wca.push( { + '_en': 'woocommerceanalytics_product_purchase', + 'blog_id': '" . esc_js( $blogid ) . "', + 'pi': '" . esc_js( $item_details['id'] ) . "', + 'pn': '" . esc_js( $item_details['name'] ) . "', + 'pc': '" . esc_js( $item_details['category'] ) . "', + 'pp': '" . esc_js( $item_details['price'] ) . "', + 'pq': '" . esc_js( $order_item->get_quantity() ) . "', + 'oi': '" . esc_js( $order->get_order_number() ) . "', + 'ui': '" . esc_js( $this->get_user_id() ) . "', + } );"; + } + + wc_enqueue_js( implode( "\r\n", $universal_commands ) ); + } + + /** + * Listen for clicks on the "Update Cart" button to know if an item has been removed by + * updating its quantity to zero + */ + public function remove_from_cart_via_quantity() { + $blogid = Jetpack::get_option( 'id' ); + + wc_enqueue_js( " + jQuery( 'button[name=update_cart]' ).on( 'click', function() { + var cartItems = jQuery( '.cart_item' ); + cartItems.each( function( item ) { + var qty = jQuery( this ).find( 'input.qty' ); + if ( qty && qty.val() === '0' ) { + var productID = jQuery( this ).find( '.product-remove a' ).data( 'product_id' ); + _wca.push( { + '_en': 'woocommerceanalytics_remove_from_cart', + 'blog_id': '" . esc_js( $blogid ) . "', + 'pi': productID, + 'ui': '" . esc_js( $this->get_user_id() ) . "', + } ); + } + } ); + } ); + " ); + } + + /** + * Get the current user id + * + * @return int + */ + public function get_user_id() { + if ( is_user_logged_in() ) { + $blogid = Jetpack::get_option( 'id' ); + $userid = get_current_user_id(); + return $blogid . ":" . $userid; + } + return 'null'; + } + +} diff --git a/plugins/jetpack/modules/woocommerce-analytics/classes/wp-woocommerce-analytics-utils.php b/plugins/jetpack/modules/woocommerce-analytics/classes/wp-woocommerce-analytics-utils.php new file mode 100644 index 00000000..b27dc043 --- /dev/null +++ b/plugins/jetpack/modules/woocommerce-analytics/classes/wp-woocommerce-analytics-utils.php @@ -0,0 +1,51 @@ +<?php +/** + * Jetpack_WooCommerce_Analytics_Options provides a single interface to module options + * + * @package Jetpack + * @author Automattic + */ + +/** + * Bail if accessed directly + */ +if ( ! defined( 'ABSPATH' ) ) { + exit; +} + +/** + * Class Jetpack_WooCommerce_Analytics_Utils + * Utility function for WooCommerce Analytics + */ +class Jetpack_WooCommerce_Analytics_Utils { + /** + * Gets product categories or varation attributes as a formatted concatenated string + * + * @param array $product WC_Product. + * @return string + */ + public static function get_product_categories_concatenated( $product ) { + if ( ! class_exists( 'WooCommerce' ) ) { + return ''; + } + + if ( ! $product ) { + return ''; + } + + $variation_data = $product->is_type( 'variation' ) ? wc_get_product_variation_attributes( $product->get_id() ) : ''; + if ( is_array( $variation_data ) && ! empty( $variation_data ) ) { + $line = wc_get_formatted_variation( $variation_data, true ); + } else { + $out = array(); + $categories = get_the_terms( $product->get_id(), 'product_cat' ); + if ( $categories ) { + foreach ( $categories as $category ) { + $out[] = $category->name; + } + } + $line = join( '/', $out ); + } + return $line; + } +} diff --git a/plugins/jetpack/modules/woocommerce-analytics/wp-woocommerce-analytics.php b/plugins/jetpack/modules/woocommerce-analytics/wp-woocommerce-analytics.php new file mode 100644 index 00000000..cc1c5ad0 --- /dev/null +++ b/plugins/jetpack/modules/woocommerce-analytics/wp-woocommerce-analytics.php @@ -0,0 +1,96 @@ +<?php +/** + * Jetpack_WooCommerce_Analytics is ported from the Jetpack_Google_Analytics code. + * + * @package Jetpack + */ + +if ( ! defined( 'ABSPATH' ) ) { + exit; +} + +require_once plugin_basename( 'classes/wp-woocommerce-analytics-utils.php' ); +require_once plugin_basename( 'classes/wp-woocommerce-analytics-universal.php' ); + +/** + * Class Jetpack_WooCommerce_Analytics + * Instantiate WooCommerce Analytics + */ +class Jetpack_WooCommerce_Analytics { + + /** + * Instance of this class + * + * @var Jetpack_WooCommerce_Analytics - Static property to hold our singleton instance + */ + private static $instance = false; + + /** + * Instance of the Universal functions + * + * @var Static property to hold concrete analytics impl that does the work (universal or legacy) + */ + private static $analytics = false; + + /** + * WooCommerce Analytics is only available to Jetpack connected WooCommerce stores with both plugins set to active + * and WooCommerce version 3.0 or higher + * + * @return bool + */ + public static function shouldTrackStore() { + // Tracking only Site pages + if ( is_admin() ) { + return false; + } + // Don't track site admins + if ( is_user_logged_in() && in_array( 'administrator', wp_get_current_user()->roles ) ) { + return false; + } + // Make sure Jetpack is installed and active + if ( ! Jetpack::is_active() ) { + return false; + } + /** + * Make sure WooCommerce is installed and active + * + * This action is documented in https://docs.woocommerce.com/document/create-a-plugin + */ + if ( ! in_array( 'woocommerce/woocommerce.php', apply_filters( 'active_plugins', get_option( 'active_plugins' ) ) ) ) { + return false; + } + + // Ensure the WooCommerce class exists and is a valid version + $minimum_woocommerce_active = class_exists( 'WooCommerce' ) && version_compare( WC_VERSION, '3.0', '>=' ); + if ( ! $minimum_woocommerce_active ) { + return false; + } + return true; + } + + /** + * This is our constructor, which is private to force the use of get_instance() + * + * @return void + */ + private function __construct() { + $analytics = new Jetpack_WooCommerce_Analytics_Universal(); + } + + /** + * Function to instantiate our class and make it a singleton + */ + public static function get_instance() { + if ( ! Jetpack_WooCommerce_Analytics::shouldTrackStore() ) { + return; + } + if ( ! self::$instance ) { + self::$instance = new self(); + } + + return self::$instance; + } +} + +global $jetpack_woocommerce_analytics; +$jetpack_woocommerce_analytics = Jetpack_WooCommerce_Analytics::get_instance(); |