0 ) { add_submenu_page( 'woocommerce-marketing', 'TikTok', 'TikTok', 'manage_woocommerce', 'tiktok', array( 'Tt4b_Menu_Class', 'tt4b_admin_menu_main' ), 5 ); } else { add_menu_page( 'TikTok', 'TikTok', 'manage_options', 'tiktok', array( 'Tt4b_Menu_Class', 'tt4b_admin_menu_main' ), '', 40 ); } add_action( 'admin_enqueue_scripts', array( 'tt4b_menu_class', 'load_styles' ) ); } /** * Loads the plug-in page. */ public static function tt4b_admin_menu_main() { $logger = new Logger(); $mapi = new Tt4b_Mapi_Class( $logger ); $request_uri = ''; if ( isset( $_SERVER['REQUEST_URI'] ) ) { $request_uri = esc_url_raw( wp_unslash( $_SERVER['REQUEST_URI'] ) ); } // business profile data, to be replaced with response from business profile endpoint. $is_connected = false; $business_platform = 'WOO_COMMERCE'; $external_business_id = get_option( 'tt4b_external_business_id' ); if ( false === $external_business_id ) { $external_business_id = uniqid( 'tt4b_woocommerce_' ); update_option( 'tt4b_external_business_id', $external_business_id ); } // to ensure non-null. To be populated with real data. $advertiser_id = ''; $bc_id = ''; $catalog_id = ''; $pixel_code = ''; $processing = 0; $approved = 0; $rejected = 0; $advanced_matching = false; $shop_name = get_bloginfo( 'name' ); $redirect_uri = admin_url(); $app_id = get_option( 'tt4b_app_id' ); $secret = get_option( 'tt4b_secret' ); $external_data_key = get_option( 'tt4b_external_data_key' ); if ( false === $app_id || false === $secret || false === $external_data_key ) { // create open source app. $cleaned_redirect = preg_replace( '/[^A-Za-z0-9\-]/', '', admin_url() ); $smb_id = $external_business_id . $cleaned_redirect; $app_rsp = $mapi->create_open_source_app( $smb_id, $business_platform, $redirect_uri ); if ( false !== $app_rsp ) { $open_source_app_rsp = json_decode( $app_rsp, true ); $app_id = isset( $open_source_app_rsp['data']['app_id'] ) ? $open_source_app_rsp['data']['app_id'] : ''; $secret = isset( $open_source_app_rsp['data']['app_secret'] ) ? $open_source_app_rsp['data']['app_secret'] : ''; $external_data_key = isset( $open_source_app_rsp['data']['external_data_key'] ) ? $open_source_app_rsp['data']['external_data_key'] : ''; $redirect_uri = isset( $open_source_app_rsp['data']['redirect_uri'] ) ? $open_source_app_rsp['data']['redirect_uri'] : ''; update_option( 'tt4b_app_id', $app_id ); update_option( 'tt4b_secret', $secret ); update_option( 'tt4b_external_data_key', $external_data_key ); } else { wp_die( "An error occurred generating credentials needed for the integration. Please reach out to support here." ); } } $current_tiktok_for_woocommerce_version = get_option( 'tt4b_version' ); $version = '1.5'; $industry_id = '291408'; $locale = strtok( get_locale(), '_' ); $now = new DateTime(); $timestamp = (string) ( $now->getTimestamp() * 1000 ); $timezone = 'GMT+00:00'; $gmt_offset = get_option( 'gmt_offset' ); if ( $gmt_offset > 0 ) { $timezone = 'GMT+' . $gmt_offset . ':00'; } elseif ( $gmt_offset < 0 ) { $timezone = 'GMT' . $gmt_offset . ':00'; } $email = get_option( 'admin_email' ); $shop_url = get_site_url(); $shop_domain = get_site_url(); $hmac_str = 'version=' . $version . '×tamp=' . $timestamp . '&locale=' . $locale . '&business_platform=' . $business_platform . '&external_business_id=' . $external_business_id; $hmac = hash_hmac( 'sha256', $hmac_str, $external_data_key ); $obj = array( 'external_business_id' => $external_business_id, 'business_platform' => $business_platform, 'locale' => $locale, 'version' => $version, 'timestamp' => $timestamp, 'timezone' => $timezone, 'email' => $email, 'industry_id' => $industry_id, 'store_name' => $shop_name, 'website_url' => $shop_url, 'domain' => $shop_domain, 'app_id' => $app_id, 'redirect_uri' => $redirect_uri, 'hmac' => $hmac, 'close_method' => 'redirect_inside_tiktok', 'extra_data' => $current_tiktok_for_woocommerce_version, ); $obj = self::get_country_and_currency_for_external_data( $obj ); $obj = self::get_eligibility_info_for_external_data( $obj ); $obj = self::get_address_info_for_external_data( $obj ); $obj = self::get_extra_data_for_external_data( $obj ); $external_data = base64_encode( json_encode( $obj, JSON_UNESCAPED_SLASHES ) ); update_option( 'tt4b_external_data', $external_data ); // log the external_data for ease of debugging. $logger->log( __METHOD__, 'external_data: ' . $external_data ); $access_token = get_option( 'tt4b_access_token' ); if ( false !== $access_token ) { $is_connected = true; // get business profile information to pass into external data. $business_profile_rsp = $mapi->get_business_profile( $access_token, $external_business_id ); $business_profile = json_decode( $business_profile_rsp, true ); // Check connection status against profile status if ( array() === $business_profile['data'] ) { $is_connected = false; delete_option( 'tt4b_access_token' ); } elseif ( ! is_null( $business_profile['data']['status'] ) && 2 !== $business_profile['data']['status'] ) { $is_connected = false; } else { if ( ! is_null( $business_profile['data']['adv_id'] ) ) { $advertiser_id = $business_profile['data']['adv_id']; update_option( 'tt4b_advertiser_id', $advertiser_id ); } if ( ! is_null( $business_profile['data']['bc_id'] ) ) { $bc_id = $business_profile['data']['bc_id']; update_option( 'tt4b_bc_id', $bc_id ); } if ( ! is_null( $business_profile['data']['pixel_code'] ) ) { $pixel_code = $business_profile['data']['pixel_code']; update_option( 'tt4b_pixel_code', $pixel_code ); } if ( ! is_null( $business_profile['data'] ) && array_key_exists( 'catalog_id', $business_profile['data'] ) && ! is_null( $business_profile['data']['catalog_id'] ) ) { $catalog_id = $business_profile['data']['catalog_id']; update_option( 'tt4b_catalog_id', $catalog_id ); } if ( ! is_null( $business_profile['data'] ) && array_key_exists( 'catalog_id', $business_profile['data'] ) && ! is_null( $business_profile['data']['catalog_id'] ) && array_key_exists( 'bc_id', $business_profile['data'] ) && ! is_null( $business_profile['data']['bc_id'] ) ) { if ( did_action( 'woocommerce_loaded' ) > 0 ) { $catalog_obj = new Tt4b_Catalog_Class( $mapi, $logger ); $logger->log( __METHOD__, 'initiate catalog sync started' ); $catalog_obj->initiate_catalog_sync( $catalog_id, $bc_id, $shop_name, $access_token ); $product_review_status = $catalog_obj->get_catalog_processing_status( $access_token, $bc_id, $catalog_id ); $processing = $product_review_status['processing']; $approved = $product_review_status['approved']; $rejected = $product_review_status['rejected']; } } if ( is_null( $advertiser_id ) || is_null( $pixel_code ) || is_null( $access_token ) ) { // set advanced matching to false if the pixel cannot be found. $advanced_matching = false; } else { $pixel_obj = new Tt4b_Pixel_Class(); $pixel_rsp = $pixel_obj->get_pixels( $access_token, $advertiser_id, $pixel_code ); $pixel = json_decode( $pixel_rsp, true ); if ( '' !== $pixel ) { $connected_pixel = $pixel['data']['pixels'][0]; $advanced_matching = $connected_pixel['advanced_matching_fields']['email']; update_option( 'tt4b_advanced_matching', $advanced_matching ); } } // update eligibility $total_gmv = intval( get_option( 'tt4b_mapi_total_gmv' ) ); $total_orders = intval( get_option( 'tt4b_mapi_total_orders' ) ); $days_since_first_order = intval( get_option( 'tt4b_mapi_tenure' ) ); $mapi->update_business_profile( $access_token, $external_business_id, $total_gmv, $total_orders, $days_since_first_order, $current_tiktok_for_woocommerce_version ); } } // enqueue js. echo '
'; wp_register_script( 'tt4b_cdn', 'https://sf-ttmp.ttcdn-row.com/obj/ttastatic-sg/tiktok-business-plugin/tbp_external_platform-v2.3.11.js', '', 'v1', false ); wp_register_script( 'tt4b_script', plugins_url( '/admin/js/localJs.js', dirname( __DIR__ ) . '/Tiktokforbusiness.php' ), array( 'tt4b_cdn' ), 'v1', false ); wp_enqueue_script( 'tt4b_script' ); wp_localize_script( 'tt4b_script', 'tt4b_script_vars', array( 'is_connected' => $is_connected, 'external_business_id' => $external_business_id, 'business_platform' => $business_platform, 'base_uri' => $request_uri, 'external_data' => $external_data, 'bc_id' => $bc_id, 'adv_id' => $advertiser_id, 'store_name' => $shop_name, 'pixel_code' => $pixel_code, 'catalog_id' => $catalog_id, 'advanced_matching' => $advanced_matching, 'approved' => $approved, 'processing' => $processing, 'rejected' => $rejected, 'country' => get_option( 'tt4b_user_country' ), ) ); } /** * Populates eligibility for woocommerce merchants. */ public static function get_eligibility_info_for_external_data( $external_data ) { // pull eligibility // pull eligibility metrics for tt shopping. if ( ! did_action( 'woocommerce_loaded' ) > 0 ) { return $external_data; } $logger = new Logger(); $mapi = new Tt4b_Mapi_Class( $logger ); $mapi->fetch_eligibility(); $total_gmv = intval( get_option( 'tt4b_mapi_total_gmv' ) ); $total_orders = intval( get_option( 'tt4b_mapi_total_orders' ) ); $days_since_first_order = intval( get_option( 'tt4b_mapi_tenure' ) ); $net_gmv = array( array( 'interval' => 'LIFETIME', 'min' => $total_gmv, 'max' => $total_gmv, 'unit' => 'CURRENCY', ), ); $net_order_count = array( array( 'interval' => 'LIFETIME', 'min' => $total_orders, 'max' => $total_orders, 'unit' => 'COUNT', ), ); $tenure = array( 'min' => $days_since_first_order, 'unit' => 'DAYS', ); $external_data['is_email_verified'] = true; $external_data['is_verified'] = true; $external_data['net_gmv'] = $net_gmv; $external_data['net_order_count'] = $net_order_count; $external_data['tenure'] = $tenure; return $external_data; } /** * Populates the extra data for external data. */ public static function get_extra_data_for_external_data( $external_data ) { $current_tiktok_for_woocommerce_version = get_option( 'tt4b_version' ); $extra_data = array( 'version' => $current_tiktok_for_woocommerce_version, ); if ( ! did_action( 'woocommerce_loaded' ) > 0 ) { $extra_data['plugin_type'] = 'LEAD_GEN'; } $external_data['extra_data'] = json_encode( $extra_data ); return $external_data; } /** * Populates address info for woocommerce merchants. */ public static function get_address_info_for_external_data( $external_data ) { if ( did_action( 'woocommerce_loaded' ) > 0 ) { $store_address = get_option( 'woocommerce_store_address' ); $store_address_2 = get_option( 'woocommerce_store_address_2' ); $store_city = get_option( 'woocommerce_store_city' ); $store_postcode = get_option( 'woocommerce_store_postcode' ); // country and state separated. $store_raw_country = get_option( 'woocommerce_default_country' ); $split_country = explode( ':', $store_raw_country ); $store_state = ''; if ( count( $split_country ) > 1 ) { $store_state = $split_country[1]; $menu_obj = new Tt4b_Menu_Class(); $store_state = $menu_obj->convert_state( $store_state ); } $external_data['address_1'] = $store_address; $external_data['address_2'] = $store_address_2; $external_data['city'] = $store_city; $external_data['state'] = $store_state; $external_data['zip_code'] = $store_postcode; } return $external_data; } /** * Populates currency and country_iso for woocommerce merchants. */ public static function get_country_and_currency_for_external_data( $external_data ) { if ( did_action( 'woocommerce_loaded' ) > 0 ) { $external_data['currency'] = get_woocommerce_currency(); $external_data['country_region'] = WC()->countries->get_base_country(); update_option( 'tt4b_user_country', WC()->countries->get_base_country() ); return $external_data; } $country_region = self::get_user_country(); if ( '' !== $country_region ) { $external_data['country_region'] = $country_region; } update_option( 'tt4b_user_country', $country_region ); return $external_data; } /** * Gets the user country. * * @return string */ private static function get_user_country() { $logger = new Logger(); $mapi = new Tt4b_Mapi_Class( $logger ); $resp = $mapi->get_user_location(); $decoded_resp = json_decode( $resp, true ); if ( ! is_null( $decoded_resp['data'] && ! is_null( $decoded_resp['data']['country_code'] ) ) ) { return $decoded_resp['data']['country_code']; } return ''; } /** * Loads content from TikTok CDN into the page head. * * @param $hook_suffix string The hook used * * @return void */ public static function load_styles( $hook_suffix = '' ) { if ( 'marketing_page_tiktok' !== $hook_suffix ) { return; } wp_enqueue_style( 'tt4b_localCss', plugins_url( '/admin/css/main.css', dirname( __DIR__ ) . '/tiktok-for-woocommerce.php' ), '', 'v1' ); } /** * Stores the mapi issued access token. * * @return void */ public static function tt4b_store_access_token() { $logger = new Logger(); $mapi = new Tt4b_Mapi_Class( $logger ); $url = ''; if ( isset( $_SERVER['HTTP_HOST'] ) && isset( $_SERVER['REQUEST_URI'] ) ) { $url = esc_url_raw( wp_unslash( $_SERVER['HTTP_HOST'] ) . wp_unslash( $_SERVER['REQUEST_URI'] ) ); } $auth_code = ''; $split_url = explode( '&', $url ); $split_url_params = count( $split_url ); for ( $i = 0; $i < $split_url_params; $i++ ) { if ( false !== strpos( $split_url[ $i ], 'auth_code' ) ) { $auth_code = substr( $split_url[ $i ], strpos( $split_url[ $i ], '=' ) + 1 ); $logger->log( __METHOD__, "auth_code retrieved: $auth_code" ); } } if ( '' !== $auth_code ) { $app_id = get_option( 'tt4b_app_id' ); $secret = get_option( 'tt4b_secret' ); $access_token_rsp = $mapi->get_access_token( $app_id, $secret, $auth_code, 'v1.2' ); $results = json_decode( $access_token_rsp, true ); if ( 'OK' === $results['message'] ) { // status OK. $access_token = $results['data']['access_token']; update_option( 'tt4b_access_token', $access_token ); wp_safe_redirect( get_admin_url() . 'admin.php?page=tiktok' ); } } } /** * Converts state abbreviation to full state name. * * @param string $name The state abbreviation or name * * @return string */ public static function convert_state( $name ) { $states = array( array( 'name' => 'Alabama', 'abbr' => 'AL', ), array( 'name' => 'Alaska', 'abbr' => 'AK', ), array( 'name' => 'Arizona', 'abbr' => 'AZ', ), array( 'name' => 'Arkansas', 'abbr' => 'AR', ), array( 'name' => 'California', 'abbr' => 'CA', ), array( 'name' => 'Colorado', 'abbr' => 'CO', ), array( 'name' => 'Connecticut', 'abbr' => 'CT', ), array( 'name' => 'District of Columbia', 'abbr' => 'DC', ), array( 'name' => 'Delaware', 'abbr' => 'DE', ), array( 'name' => 'Florida', 'abbr' => 'FL', ), array( 'name' => 'Georgia', 'abbr' => 'GA', ), array( 'name' => 'Hawaii', 'abbr' => 'HI', ), array( 'name' => 'Idaho', 'abbr' => 'ID', ), array( 'name' => 'Illinois', 'abbr' => 'IL', ), array( 'name' => 'Indiana', 'abbr' => 'IN', ), array( 'name' => 'Iowa', 'abbr' => 'IA', ), array( 'name' => 'Kansas', 'abbr' => 'KS', ), array( 'name' => 'Kentucky', 'abbr' => 'KY', ), array( 'name' => 'Louisiana', 'abbr' => 'LA', ), array( 'name' => 'Maine', 'abbr' => 'ME', ), array( 'name' => 'Maryland', 'abbr' => 'MD', ), array( 'name' => 'Massachusetts', 'abbr' => 'MA', ), array( 'name' => 'Michigan', 'abbr' => 'MI', ), array( 'name' => 'Minnesota', 'abbr' => 'MN', ), array( 'name' => 'Mississippi', 'abbr' => 'MS', ), array( 'name' => 'Missouri', 'abbr' => 'MO', ), array( 'name' => 'Montana', 'abbr' => 'MT', ), array( 'name' => 'Nebraska', 'abbr' => 'NE', ), array( 'name' => 'Nevada', 'abbr' => 'NV', ), array( 'name' => 'New Hampshire', 'abbr' => 'NH', ), array( 'name' => 'New Jersey', 'abbr' => 'NJ', ), array( 'name' => 'New Mexico', 'abbr' => 'NM', ), array( 'name' => 'New York', 'abbr' => 'NY', ), array( 'name' => 'North Carolina', 'abbr' => 'NC', ), array( 'name' => 'North Dakota', 'abbr' => 'ND', ), array( 'name' => 'Ohio', 'abbr' => 'OH', ), array( 'name' => 'Oklahoma', 'abbr' => 'OK', ), array( 'name' => 'Oregon', 'abbr' => 'OR', ), array( 'name' => 'Pennsylvania', 'abbr' => 'PA', ), array( 'name' => 'Rhode Island', 'abbr' => 'RI', ), array( 'name' => 'South Carolina', 'abbr' => 'SC', ), array( 'name' => 'South Dakota', 'abbr' => 'SD', ), array( 'name' => 'Tennessee', 'abbr' => 'TN', ), array( 'name' => 'Texas', 'abbr' => 'TX', ), array( 'name' => 'Utah', 'abbr' => 'UT', ), array( 'name' => 'Vermont', 'abbr' => 'VT', ), array( 'name' => 'Virginia', 'abbr' => 'VA', ), array( 'name' => 'Washington', 'abbr' => 'WA', ), array( 'name' => 'West Virginia', 'abbr' => 'WV', ), array( 'name' => 'Wisconsin', 'abbr' => 'WI', ), array( 'name' => 'Wyoming', 'abbr' => 'WY', ), array( 'name' => 'Virgin Islands', 'abbr' => 'V.I.', ), array( 'name' => 'Guam', 'abbr' => 'GU', ), array( 'name' => 'Puerto Rico', 'abbr' => 'PR', ), ); $return = ''; $str_len = strlen( $name ); foreach ( $states as $state ) : if ( $str_len < 2 ) { return ''; } elseif ( 2 === $str_len ) { if ( strtolower( $state['abbr'] ) === strtolower( $name ) ) { $return = $state['name']; break; } } elseif ( strtolower( $state['name'] ) === strtolower( $name ) ) { $return = strtoupper( $state['abbr'] ); break; } endforeach; return $return; } }