oont-contents/plugins/tiktok-for-business/admin/Tt4b_Menu_Class.php
2025-03-31 21:42:48 +02:00

668 lines
22 KiB
PHP

<?php
/**
* Copyright (c) Bytedance, 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 TikTok
*/
class Tt4b_Menu_Class {
/**
* Loads the menu.
*
* @return void
*/
public static function tt4b_admin_menu() {
// if on woocommerce, load into woo marketing submenu. Else load into admin dashboard
if ( did_action( 'woocommerce_loaded' ) > 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 <a href='https://ads.tiktok.com/athena/user-feedback/form?identify_key=6a1e079024806640c5e1e695d13db80949525168a052299b4970f9c99cb5ac78&board_id=1705151646989423'>here</a>." );
}
}
$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 . '&timestamp=' . $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 '<div class="tt4b_wrap" id="tiktok-for-business-root"></div>';
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;
}
}