File "class-wc-connect-nux.php"
Full Path: /home/vantageo/public_html/cache/cache/cache/cache/.wp-cli/wp-content/plugins/woocommerce-services/classes/class-wc-connect-nux.php
File size: 21.55 KB
MIME-type: text/x-php
Charset: utf-8
<?php
if ( ! class_exists( 'WC_Connect_Nux' ) ) {
class WC_Connect_Nux {
/**
* Jetpack status constants.
*/
const JETPACK_NOT_CONNECTED = 'not-connected';
const JETPACK_OFFLINE_MODE = 'offline-mode';
const JETPACK_CONNECTED = 'connected';
const IS_NEW_LABEL_USER = 'wcc_is_new_label_user';
/**
* Option name for dismissing success banner
* after the JP connection flow
*/
const SHOULD_SHOW_AFTER_CXN_BANNER = 'should_display_nux_after_jp_cxn_banner';
/**
* @var WC_Connect_Tracks
*/
protected $tracks;
/**
* @var WC_Connect_Shipping_Label
*/
private $shipping_label;
function __construct( WC_Connect_Tracks $tracks, WC_Connect_Shipping_Label $shipping_label ) {
$this->tracks = $tracks;
$this->shipping_label = $shipping_label;
$this->init_pointers();
}
private function get_notice_states() {
$states = get_user_meta( get_current_user_id(), 'wc_connect_nux_notices', true );
if ( ! is_array( $states ) ) {
return array();
}
return $states;
}
public function is_notice_dismissed( $notice ) {
$notices = $this->get_notice_states();
return isset( $notices[ $notice ] ) && $notices[ $notice ];
}
public function dismiss_notice( $notice ) {
$notices = $this->get_notice_states();
$notices[ $notice ] = true;
update_user_meta( get_current_user_id(), 'wc_connect_nux_notices', $notices );
}
public function ajax_dismiss_notice() {
if ( empty( $_POST['dismissible_id'] ) ) {
return;
}
check_ajax_referer( 'wc_connect_dismiss_notice', 'nonce' );
$this->dismiss_notice( sanitize_key( $_POST['dismissible_id'] ) );
wp_die();
}
private function init_pointers() {
add_filter( 'wc_services_pointer_post.php', array( $this, 'register_order_page_labels_pointer' ) );
add_filter( 'wc_services_pointer_post.php', array( $this, 'register_new_carrier_dhl_pointer' ) );
}
public function show_pointers( $hook ) {
/*
Get admin pointers for the current admin page.
*
* @since 0.9.6
*
* @param array $pointers Array of pointers.
*/
$pointers = apply_filters( 'wc_services_pointer_' . $hook, array() );
if ( ! $pointers || ! is_array( $pointers ) ) {
return;
}
$dismissed_pointers = $this->get_dismissed_pointers();
$valid_pointers = array();
foreach ( $pointers as $pointer ) {
if ( ! in_array( $pointer['id'], $dismissed_pointers, true ) ) {
$valid_pointers[] = $pointer;
}
}
if ( empty( $valid_pointers ) ) {
return;
}
wp_enqueue_style( 'wp-pointer' );
wp_localize_script( 'wc_services_admin_pointers', 'wcServicesAdminPointers', $valid_pointers );
wp_enqueue_script( 'wc_services_admin_pointers' );
}
public function get_dismissed_pointers() {
$data = get_user_meta( get_current_user_id(), 'dismissed_wp_pointers', true );
if ( is_string( $data ) && 0 < strlen( $data ) ) {
return explode( ',', $data );
}
return array();
}
/**
* Dismiss a WP pointer for the current user.
*
* @param string $pointer_to_dismiss Pointer ID to dismiss for the current user
*/
public function dismiss_pointer( $pointer_to_dismiss ) {
$dismissed_pointers = $this->get_dismissed_pointers();
if ( in_array( $pointer_to_dismiss, $dismissed_pointers, true ) ) {
return;
}
$dismissed_pointers[] = $pointer_to_dismiss;
$dismissed_data = implode( ',', $dismissed_pointers );
update_user_meta( get_current_user_id(), 'dismissed_wp_pointers', $dismissed_data );
}
public function is_new_labels_user() {
$is_new_user = get_transient( self::IS_NEW_LABEL_USER );
if ( false === $is_new_user ) {
global $wpdb;
$query = "SELECT meta_key FROM {$wpdb->postmeta} WHERE meta_key = 'wc_connect_labels' LIMIT 1";
$results = $wpdb->get_results( $query );
$is_new_user = 0 === count( $results ) ? 'yes' : 'no';
set_transient( self::IS_NEW_LABEL_USER, $is_new_user );
}
return 'yes' === $is_new_user;
}
public function register_order_page_labels_pointer( $pointers ) {
// If the user is not new to labels, we should just dismiss this pointer
if ( ! $this->is_new_labels_user() ) {
$this->dismiss_pointer( 'wc_services_labels_metabox' );
return $pointers;
}
global $post;
if ( ! $this->shipping_label->should_show_meta_box( $post ) ) {
return $pointers;
}
$supported_carriers = array( 'USPS' );
if ( $this->shipping_label->is_dhl_express_available() ) {
$supported_carriers[] = 'DHL';
}
$pointers[] = array(
'id' => 'wc_services_labels_metabox',
'target' => '#woocommerce-order-label .button',
'options' => array(
'content' => sprintf(
'<h3>%s</h3><p>%s</p>',
__( 'Discounted Shipping Labels', 'woocommerce-services' ),
sprintf( __( "When you're ready, purchase and print discounted labels from %s right here.", 'woocommerce-services' ), implode( ' or ', $supported_carriers ) )
),
'position' => array(
'edge' => 'top',
'align' => 'left',
),
),
'dim' => true,
);
return $pointers;
}
public function register_new_carrier_dhl_pointer( $pointers ) {
// new user? no need to show this alert, `wc_services_labels_metabox` will take care of communicating about DHL
if ( $this->is_new_labels_user() ) {
$this->dismiss_pointer( 'wc_services_new_carrier_dhl_express' );
return $pointers;
}
// existing user? figure out if the order supports DHL, then let them know DHL is a new carrier!
if ( ! $this->shipping_label->is_order_dhl_express_eligible() ) {
return $pointers;
}
$pointers[] = array(
'id' => 'wc_services_new_carrier_dhl_express',
'target' => '#woocommerce-order-label .button',
'options' => array(
'content' => sprintf(
'<h3>%s</h3><p>%s</p>',
__( 'Discounted DHL Shipping Labels', 'woocommerce-services' ),
__( 'WooCommerce Shipping now supports DHL labels for international shipments. Purchase and print discounted labels from DHL and USPS right here.', 'woocommerce-services' )
),
'position' => array(
'edge' => 'top',
'align' => 'left',
),
),
'dim' => true,
);
return $pointers;
}
public static function get_banner_type_to_display( $status = array() ) {
if ( ! isset( $status['jetpack_connection_status'] ) ) {
return false;
}
/*
The NUX Flow:
- Case 1: Jetpack not connected (with TOS or no TOS accepted):
1. show_banner_before_connection()
2. connect to JP
3. show_banner_after_connection(), which sets the TOS acceptance in options
- Case 2: Jetpack connected, no TOS
1. show_tos_only_banner(), which accepts TOS on button click
- Case 3: Jetpack connected, and TOS accepted
This is an existing user. Do nothing.
*/
switch ( $status['jetpack_connection_status'] ) {
case self::JETPACK_NOT_CONNECTED:
return 'before_jetpack_connection';
case self::JETPACK_CONNECTED:
case self::JETPACK_OFFLINE_MODE:
// Has the user just gone through our NUX connection flow?
if ( isset( $status['should_display_after_cxn_banner'] ) && $status['should_display_after_cxn_banner'] ) {
return 'after_jetpack_connection';
}
// Has the user already accepted our TOS? Then do nothing.
// Note: TOS is accepted during the after_connection banner
if (
isset( $status['tos_accepted'] )
&& ! $status['tos_accepted']
&& isset( $status['can_accept_tos'] )
&& $status['can_accept_tos']
) {
return 'tos_only_banner';
}
return false;
default:
return false;
}
}
public function get_jetpack_install_status() {
if ( WC_Connect_Jetpack::is_offline_mode() ) {
// activated, and dev mode on
return self::JETPACK_OFFLINE_MODE;
}
// dev mode off, check if connected
if ( ! WC_Connect_Jetpack::is_connected() ) {
return self::JETPACK_NOT_CONNECTED;
}
return self::JETPACK_CONNECTED;
}
public function should_display_nux_notice_on_screen( $screen ) {
if ( // Display if on any of these admin pages.
( // Products list.
'product' === $screen->post_type
&& 'edit' === $screen->base
)
|| ( // Orders list and edit order page when not using HPOS.
'shop_order' === $screen->post_type
&& in_array( $screen->base, array( 'edit', 'post' ), true )
)
|| ( // Orders list and edit order page when using HPOS.
wc_get_page_screen_id( 'shop_order' ) === $screen->id
)
|| ( // WooCommerce settings.
'woocommerce_page_wc-settings' === $screen->base
)
|| ( // WooCommerce featured extension page
'woocommerce_page_wc-addons' === $screen->base
&& isset( $_GET['section'] ) && 'featured' === $_GET['section']
)
|| ( // WooCommerce shipping extension page
'woocommerce_page_wc-addons' === $screen->base
&& isset( $_GET['section'] ) && 'shipping_methods' === $_GET['section']
)
|| 'plugins' === $screen->base
) {
return true;
}
return false;
}
/**
* https://developers.taxjar.com/api/reference/#countries
*/
public function is_taxjar_supported_country( $country_code ) {
$taxjar_supported_countries = array_merge(
array(
'US',
'CA',
'AU',
),
WC()->countries->get_european_union_countries()
);
return in_array( $country_code, $taxjar_supported_countries );
}
public function should_display_nux_notice_for_current_store_locale() {
$store_country = WC()->countries->get_base_country();
$supports_taxes = $this->is_taxjar_supported_country( $store_country );
$supports_shipping = in_array( $store_country, array( 'US', 'CA' ) );
return $supports_shipping || $supports_taxes;
}
public function get_feature_list_for_country( $country ) {
$feature_list = false;
$supports_taxes = $this->is_taxjar_supported_country( $country );
$supports_labels = ( 'US' === $country );
$is_ppec_active = is_plugin_active( 'woocommerce-gateway-paypal-express-checkout/woocommerce-gateway-paypal-express-checkout.php' );
$ppec_settings = get_option( 'woocommerce_ppec_paypal_settings', array() );
$supports_payments = $is_ppec_active && ( ! isset( $ppec_settings['enabled'] ) || 'yes' === $ppec_settings['enabled'] );
if ( $supports_payments && $supports_taxes && $supports_labels ) {
$feature_list = __( 'automated tax calculation, shipping label printing, and smoother payment setup', 'woocommerce-services' );
} elseif ( $supports_payments && $supports_taxes ) {
$feature_list = __( 'automated tax calculation and smoother payment setup', 'woocommerce-services' );
} elseif ( $supports_taxes && $supports_labels ) {
$feature_list = __( 'automated tax calculation and shipping label printing', 'woocommerce-services' );
} elseif ( $supports_payments && $supports_labels ) {
$feature_list = __( 'shipping label printing and smoother payment setup', 'woocommerce-services' );
} elseif ( $supports_payments ) {
$feature_list = __( 'smoother payment setup', 'woocommerce-services' );
} elseif ( $supports_taxes ) {
$feature_list = __( 'automated tax calculation', 'woocommerce-services' );
} elseif ( $supports_labels ) {
$feature_list = __( 'shipping label printing', 'woocommerce-services' );
}
return $feature_list;
}
public function get_jetpack_redirect_url() {
$full_path = add_query_arg( array() );
// Remove [...]/wp-admin so we can use admin_url().
$new_index = strpos( $full_path, '/wp-admin' ) + strlen( '/wp-admin' );
$path = substr( $full_path, $new_index );
return esc_url( admin_url( $path ) );
}
public function set_up_nux_notices() {
if ( ! current_user_can( 'manage_woocommerce' ) ) {
return;
}
// Check for plugin install and activate permissions to handle Jetpack on multisites:
// Admins might not be able to install or activate plugins, but Jetpack might already have been installed by a superadmin.
// If this is the case, the admin can connect the site on their own, and should be able to use WCS as ususal
$jetpack_install_status = $this->get_jetpack_install_status();
$banner_to_display = self::get_banner_type_to_display(
array(
'jetpack_connection_status' => $jetpack_install_status,
'tos_accepted' => WC_Connect_Options::get_option( 'tos_accepted' ),
'can_accept_tos' => WC_Connect_Jetpack::is_current_user_connection_owner() || WC_Connect_Jetpack::is_offline_mode(),
'should_display_after_cxn_banner' => WC_Connect_Options::get_option( self::SHOULD_SHOW_AFTER_CXN_BANNER ),
)
);
switch ( $banner_to_display ) {
case 'before_jetpack_connection':
wp_enqueue_script( 'wc_connect_banner' );
add_action(
'admin_post_register_woocommerce_services_jetpack',
array( $this, 'register_woocommerce_services_jetpack' )
);
wp_enqueue_style( 'wc_connect_banner' );
add_action( 'admin_notices', array( $this, 'show_banner_before_connection' ), 9 );
break;
case 'tos_only_banner':
wp_enqueue_style( 'wc_connect_banner' );
add_action( 'admin_notices', array( $this, 'show_tos_banner' ) );
break;
case 'after_jetpack_connection':
wp_enqueue_style( 'wc_connect_banner' );
add_action( 'admin_notices', array( $this, 'show_banner_after_connection' ) );
break;
}
add_action( 'wp_ajax_wc_connect_dismiss_notice', array( $this, 'ajax_dismiss_notice' ) );
}
public function show_banner_before_connection() {
if ( ! $this->should_display_nux_notice_for_current_store_locale() ) {
return;
}
if ( ! $this->should_display_nux_notice_on_screen( get_current_screen() ) ) {
return;
}
// Remove Jetpack's connect banners since we're showing our own.
if ( class_exists( 'Jetpack_Connection_Banner' ) ) {
$jetpack_banner = Jetpack_Connection_Banner::init();
remove_action( 'admin_notices', array( $jetpack_banner, 'render_banner' ) );
remove_action( 'admin_notices', array( $jetpack_banner, 'render_connect_prompt_full_screen' ) );
}
// Make sure that we wait until the button is clicked before displaying
// the after_connection banner
// so that we don't accept the TOS pre-maturely
WC_Connect_Options::delete_option( self::SHOULD_SHOW_AFTER_CXN_BANNER );
$country = WC()->countries->get_base_country();
/* translators: %s: list of features, potentially comma separated */
$description_base = __( "WooCommerce Shipping & Tax is almost ready to go! Once you connect your site to WordPress.com you'll have access to %s.", 'woocommerce-services' );
$feature_list = $this->get_feature_list_for_country( $country );
$banner_content = array(
'title' => __( 'Connect your site to activate WooCommerce Shipping & Tax', 'woocommerce-services' ),
'description' => sprintf( $description_base, $feature_list ),
'button_text' => __( 'Connect', 'woocommerce-services' ),
'image_url' => plugins_url( 'images/wcs-notice.png', dirname( __FILE__ ) ),
'should_show_terms' => true,
);
$this->show_nux_banner( $banner_content );
}
public function show_banner_after_connection() {
if ( ! $this->should_display_nux_notice_for_current_store_locale() ) {
return;
}
if ( ! $this->should_display_nux_notice_on_screen( get_current_screen() ) ) {
return;
}
// Did the user just dismiss?
if ( isset( $_GET['wcs-nux-notice'] ) && 'dismiss' === $_GET['wcs-nux-notice'] ) {
// No longer need to keep track of whether the before connection banner was displayed.
WC_Connect_Options::delete_option( self::SHOULD_SHOW_AFTER_CXN_BANNER );
wp_safe_redirect( remove_query_arg( 'wcs-nux-notice' ) );
exit;
}
// By going through the connection process, the user has accepted our TOS
WC_Connect_Options::update_option( 'tos_accepted', true );
$this->tracks->opted_in( 'connection_banner' );
$country = WC()->countries->get_base_country();
/* translators: %s: list of features, potentially comma separated */
$description_base = __( 'You can now enjoy %s.', 'woocommerce-services' );
$feature_list = $this->get_feature_list_for_country( $country );
$this->show_nux_banner(
array(
'title' => __( 'Setup complete.', 'woocommerce-services' ),
'description' => esc_html( sprintf( $description_base, $feature_list ) ),
'button_text' => __( 'Got it, thanks!', 'woocommerce-services' ),
'button_link' => add_query_arg(
array(
'wcs-nux-notice' => 'dismiss',
)
),
'image_url' => plugins_url(
'images/wcs-notice.png',
dirname( __FILE__ )
),
'should_show_terms' => false,
)
);
}
public function show_tos_banner() {
if ( ! $this->should_display_nux_notice_for_current_store_locale() ) {
return;
}
if ( ! $this->should_display_nux_notice_on_screen( get_current_screen() ) ) {
return;
}
if ( isset( $_GET['wcs-nux-tos'] ) && 'accept' === $_GET['wcs-nux-tos'] ) {
WC_Connect_Options::update_option( 'tos_accepted', true );
$this->tracks->opted_in( 'tos_banner' );
wp_safe_redirect( remove_query_arg( 'wcs-nux-tos' ) );
exit;
}
$country = WC()->countries->get_base_country();
/* translators: %s: list of features, potentially comma separated */
$description_base = __( "WooCommerce Shipping & Tax is almost ready to go! Once you connect your site to WordPress.com you'll have access to %s.", 'woocommerce-services' );
$feature_list = $this->get_feature_list_for_country( $country );
$this->show_nux_banner(
array(
'title' => __( 'Connect your site to activate WooCommerce Shipping & Tax', 'woocommerce-services' ),
'description' => esc_html( sprintf( $description_base, $feature_list ) ),
'button_text' => __( 'Connect', 'woocommerce-services' ),
'button_link' => add_query_arg(
array(
'wcs-nux-tos' => 'accept',
)
),
'image_url' => plugins_url(
'images/wcs-notice.png',
dirname( __FILE__ )
),
'should_show_terms' => true,
)
);
}
public function show_nux_banner( $content ) {
if ( isset( $content['dismissible_id'] ) && $this->is_notice_dismissed( sanitize_key( $content['dismissible_id'] ) ) ) {
return;
}
?>
<div class="notice wcs-nux__notice <?php echo isset( $content['dismissible_id'] ) ? 'is-dismissible' : ''; ?>">
<div class="wcs-nux__notice-logo <?php echo isset( $content['compact_logo'] ) && $content['compact_logo'] ? 'is-compact' : ''; ?>">
<img class="wcs-nux__notice-logo-graphic" src="<?php echo esc_url( $content['image_url'] ); ?>">
</div>
<div class="wcs-nux__notice-content">
<h1 class="wcs-nux__notice-content-title">
<?php echo esc_html( $content['title'] ); ?>
</h1>
<p class="wcs-nux__notice-content-text">
<?php echo esc_html( $content['description'] ); ?>
</p>
<?php if ( isset( $content['should_show_terms'] ) && $content['should_show_terms'] ) : ?>
<p class="wcs-nux__notice-content-tos">
<?php
/* translators: %1$s example values include "Install Jetpack and CONNECT >", "Activate Jetpack and CONNECT >", "CONNECT >" */
printf(
wp_kses(
__( 'By clicking "%1$s", you agree to our <a href="%2$s">Terms of Service</a> and have read our <a href="%3$s">Privacy Policy</a>.', 'woocommerce-services' ),
array(
'a' => array(
'href' => array(),
),
)
),
esc_html( $content['button_text'] ),
'https://wordpress.com/tos/',
'https://automattic.com/privacy/'
);
?>
</p>
<?php endif; ?>
<?php if ( isset( $content['button_link'] ) ) : ?>
<a
class="wcs-nux__notice-content-button button button-primary"
href="<?php echo esc_url( $content['button_link'] ); ?>"
>
<?php echo esc_html( $content['button_text'] ); ?>
</a>
<?php else : ?>
<form action="<?php echo esc_attr( admin_url( 'admin-post.php' ) ); ?>" method="post">
<input type="hidden" name="action" value="register_woocommerce_services_jetpack"/>
<input type="hidden" name="redirect_url"
value="<?php echo esc_url( $this->get_jetpack_redirect_url() ); ?>"/>
<?php wp_nonce_field( 'wcs_nux_notice' ); ?>
<button
type="submit"
class="woocommerce-services__connect-jetpack wcs-nux__notice-content-button button button-primary"
>
<?php echo esc_html( $content['button_text'] ); ?>
</button>
</form>
<?php endif; ?>
</div>
</div>
<?php
if ( isset( $content['dismissible_id'] ) ) :
// Add handler for dismissing banner. Only supports a single banner at a time
wp_enqueue_script( 'wp-util' );
?>
<script>
(
function ($) {
$('.wcs-nux__notice').on('click', '.notice-dismiss', function () {
wp.ajax.post({
action: 'wc_connect_dismiss_notice',
dismissible_id: "<?php echo esc_js( $content['dismissible_id'] ); ?>",
nonce: "<?php echo esc_js( wp_create_nonce( 'wc_connect_dismiss_notice' ) ); ?>"
})
})
}
)(jQuery)
</script>
<?php
endif;
}
/**
* Connects the site to Jetpack.
*/
public function register_woocommerce_services_jetpack() {
check_admin_referer( 'wcs_nux_notice' );
$redirect_url = '';
if ( isset( $_POST['redirect_url'] ) ) {
$redirect_url = esc_url_raw( wp_unslash( $_POST['redirect_url'] ) );
}
// Make sure we always display the after-connection banner
// after the before_connection button is clicked
WC_Connect_Options::update_option( self::SHOULD_SHOW_AFTER_CXN_BANNER, true );
WC_Connect_Jetpack::connect_site( $redirect_url );
}
}
}