| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479 |
- <?php
- /**
- * Class WC_Customer_Data_Store file.
- *
- * @package WooCommerce\DataStores
- */
- if ( ! defined( 'ABSPATH' ) ) {
- exit;
- }
- /**
- * WC Customer Data Store.
- *
- * @version 3.0.0
- */
- class WC_Customer_Data_Store extends WC_Data_Store_WP implements WC_Customer_Data_Store_Interface, WC_Object_Data_Store_Interface {
- /**
- * Data stored in meta keys, but not considered "meta".
- *
- * @since 3.0.0
- * @var array
- */
- protected $internal_meta_keys = array(
- 'locale',
- 'billing_postcode',
- 'billing_city',
- 'billing_address_1',
- 'billing_address_2',
- 'billing_state',
- 'billing_country',
- 'shipping_postcode',
- 'shipping_city',
- 'shipping_address_1',
- 'shipping_address_2',
- 'shipping_state',
- 'shipping_country',
- 'paying_customer',
- 'last_update',
- 'first_name',
- 'last_name',
- 'display_name',
- 'show_admin_bar_front',
- 'use_ssl',
- 'admin_color',
- 'rich_editing',
- 'comment_shortcuts',
- 'dismissed_wp_pointers',
- 'show_welcome_panel',
- 'session_tokens',
- 'nickname',
- 'description',
- 'billing_first_name',
- 'billing_last_name',
- 'billing_company',
- 'billing_phone',
- 'billing_email',
- 'shipping_first_name',
- 'shipping_last_name',
- 'shipping_company',
- 'wptests_capabilities',
- 'wptests_user_level',
- 'syntax_highlighting',
- '_order_count',
- '_money_spent',
- );
- /**
- * Internal meta type used to store user data.
- *
- * @var string
- */
- protected $meta_type = 'user';
- /**
- * Callback to remove unwanted meta data.
- *
- * @param object $meta Meta object.
- * @return bool
- */
- protected function exclude_internal_meta_keys( $meta ) {
- global $wpdb;
- $table_prefix = $wpdb->prefix ? $wpdb->prefix : 'wp_';
- return ! in_array( $meta->meta_key, $this->internal_meta_keys, true )
- && 0 !== strpos( $meta->meta_key, '_woocommerce_persistent_cart' )
- && 0 !== strpos( $meta->meta_key, 'closedpostboxes_' )
- && 0 !== strpos( $meta->meta_key, 'metaboxhidden_' )
- && 0 !== strpos( $meta->meta_key, 'manageedit-' )
- && ! strstr( $meta->meta_key, $table_prefix )
- && 0 !== stripos( $meta->meta_key, 'wp_' );
- }
- /**
- * Method to create a new customer in the database.
- *
- * @since 3.0.0
- *
- * @param WC_Customer $customer Customer object.
- *
- * @throws WC_Data_Exception If unable to create new customer.
- */
- public function create( &$customer ) {
- $id = wc_create_new_customer( $customer->get_email(), $customer->get_username(), $customer->get_password() );
- if ( is_wp_error( $id ) ) {
- throw new WC_Data_Exception( $id->get_error_code(), $id->get_error_message() );
- }
- $customer->set_id( $id );
- $this->update_user_meta( $customer );
- // Prevent wp_update_user calls in the same request and customer trigger the 'Notice of Password Changed' email.
- $customer->set_password( '' );
- wp_update_user(
- apply_filters(
- 'woocommerce_update_customer_args', array(
- 'ID' => $customer->get_id(),
- 'role' => $customer->get_role(),
- 'display_name' => $customer->get_display_name(),
- ), $customer
- )
- );
- $wp_user = new WP_User( $customer->get_id() );
- $customer->set_date_created( $wp_user->user_registered );
- $customer->set_date_modified( get_user_meta( $customer->get_id(), 'last_update', true ) );
- $customer->save_meta_data();
- $customer->apply_changes();
- do_action( 'woocommerce_new_customer', $customer->get_id() );
- }
- /**
- * Method to read a customer object.
- *
- * @since 3.0.0
- * @param WC_Customer $customer Customer object.
- * @throws Exception If invalid customer.
- */
- public function read( &$customer ) {
- $user_object = $customer->get_id() ? get_user_by( 'id', $customer->get_id() ) : false;
- // User object is required.
- if ( ! $user_object || empty( $user_object->ID ) ) {
- throw new Exception( __( 'Invalid customer.', 'woocommerce' ) );
- }
- // Only users on this site should be read.
- if ( is_multisite() && ! is_user_member_of_blog( $customer->get_id() ) ) {
- throw new Exception( __( 'Invalid customer.', 'woocommerce' ) );
- }
- $customer_id = $customer->get_id();
- // Load meta but exclude deprecated props.
- $user_meta = array_diff_key(
- array_map( 'wc_flatten_meta_callback', get_user_meta( $customer_id ) ),
- array_flip( array( 'country', 'state', 'postcode', 'city', 'address', 'address_2', 'default', 'location' ) )
- );
- $customer->set_props( $user_meta );
- $customer->set_props(
- array(
- 'is_paying_customer' => get_user_meta( $customer_id, 'paying_customer', true ),
- 'email' => $user_object->user_email,
- 'username' => $user_object->user_login,
- 'display_name' => $user_object->display_name,
- 'date_created' => $user_object->user_registered, // Mysql string in local format.
- 'date_modified' => get_user_meta( $customer_id, 'last_update', true ),
- 'role' => ! empty( $user_object->roles[0] ) ? $user_object->roles[0] : 'customer',
- )
- );
- $customer->read_meta_data();
- $customer->set_object_read( true );
- do_action( 'woocommerce_customer_loaded', $customer );
- }
- /**
- * Updates a customer in the database.
- *
- * @since 3.0.0
- * @param WC_Customer $customer Customer object.
- */
- public function update( &$customer ) {
- wp_update_user(
- apply_filters(
- 'woocommerce_update_customer_args', array(
- 'ID' => $customer->get_id(),
- 'user_email' => $customer->get_email(),
- 'display_name' => $customer->get_display_name(),
- ), $customer
- )
- );
- // Only update password if a new one was set with set_password.
- if ( $customer->get_password() ) {
- wp_update_user(
- array(
- 'ID' => $customer->get_id(),
- 'user_pass' => $customer->get_password(),
- )
- );
- $customer->set_password( '' );
- }
- $this->update_user_meta( $customer );
- $customer->set_date_modified( get_user_meta( $customer->get_id(), 'last_update', true ) );
- $customer->save_meta_data();
- $customer->apply_changes();
- do_action( 'woocommerce_update_customer', $customer->get_id() );
- }
- /**
- * Deletes a customer from the database.
- *
- * @since 3.0.0
- * @param WC_Customer $customer Customer object.
- * @param array $args Array of args to pass to the delete method.
- */
- public function delete( &$customer, $args = array() ) {
- if ( ! $customer->get_id() ) {
- return;
- }
- $args = wp_parse_args(
- $args, array(
- 'reassign' => 0,
- )
- );
- $id = $customer->get_id();
- wp_delete_user( $id, $args['reassign'] );
- do_action( 'woocommerce_delete_customer', $id );
- }
- /**
- * Helper method that updates all the meta for a customer. Used for update & create.
- *
- * @since 3.0.0
- * @param WC_Customer $customer Customer object.
- */
- private function update_user_meta( $customer ) {
- $updated_props = array();
- $changed_props = $customer->get_changes();
- $meta_key_to_props = array(
- 'paying_customer' => 'is_paying_customer',
- 'first_name' => 'first_name',
- 'last_name' => 'last_name',
- );
- foreach ( $meta_key_to_props as $meta_key => $prop ) {
- if ( ! array_key_exists( $prop, $changed_props ) ) {
- continue;
- }
- if ( update_user_meta( $customer->get_id(), $meta_key, $customer->{"get_$prop"}( 'edit' ) ) ) {
- $updated_props[] = $prop;
- }
- }
- $billing_address_props = array(
- 'billing_first_name' => 'billing_first_name',
- 'billing_last_name' => 'billing_last_name',
- 'billing_company' => 'billing_company',
- 'billing_address_1' => 'billing_address_1',
- 'billing_address_2' => 'billing_address_2',
- 'billing_city' => 'billing_city',
- 'billing_state' => 'billing_state',
- 'billing_postcode' => 'billing_postcode',
- 'billing_country' => 'billing_country',
- 'billing_email' => 'billing_email',
- 'billing_phone' => 'billing_phone',
- );
- foreach ( $billing_address_props as $meta_key => $prop ) {
- $prop_key = substr( $prop, 8 );
- if ( ! isset( $changed_props['billing'] ) || ! array_key_exists( $prop_key, $changed_props['billing'] ) ) {
- continue;
- }
- if ( update_user_meta( $customer->get_id(), $meta_key, $customer->{"get_$prop"}( 'edit' ) ) ) {
- $updated_props[] = $prop;
- }
- }
- $shipping_address_props = array(
- 'shipping_first_name' => 'shipping_first_name',
- 'shipping_last_name' => 'shipping_last_name',
- 'shipping_company' => 'shipping_company',
- 'shipping_address_1' => 'shipping_address_1',
- 'shipping_address_2' => 'shipping_address_2',
- 'shipping_city' => 'shipping_city',
- 'shipping_state' => 'shipping_state',
- 'shipping_postcode' => 'shipping_postcode',
- 'shipping_country' => 'shipping_country',
- );
- foreach ( $shipping_address_props as $meta_key => $prop ) {
- $prop_key = substr( $prop, 9 );
- if ( ! isset( $changed_props['shipping'] ) || ! array_key_exists( $prop_key, $changed_props['shipping'] ) ) {
- continue;
- }
- if ( update_user_meta( $customer->get_id(), $meta_key, $customer->{"get_$prop"}( 'edit' ) ) ) {
- $updated_props[] = $prop;
- }
- }
- do_action( 'woocommerce_customer_object_updated_props', $customer, $updated_props );
- }
- /**
- * Gets the customers last order.
- *
- * @since 3.0.0
- * @param WC_Customer $customer Customer object.
- * @return WC_Order|false
- */
- public function get_last_order( &$customer ) {
- global $wpdb;
- $last_order = $wpdb->get_var(
- // phpcs:disable WordPress.WP.PreparedSQL.NotPrepared
- "SELECT posts.ID
- FROM $wpdb->posts AS posts
- LEFT JOIN {$wpdb->postmeta} AS meta on posts.ID = meta.post_id
- WHERE meta.meta_key = '_customer_user'
- AND meta.meta_value = '" . esc_sql( $customer->get_id() ) . "'
- AND posts.post_type = 'shop_order'
- AND posts.post_status IN ( '" . implode( "','", array_map( 'esc_sql', array_keys( wc_get_order_statuses() ) ) ) . "' )
- ORDER BY posts.ID DESC"
- // phpcs:enable
- );
- if ( ! $last_order ) {
- return false;
- }
- return wc_get_order( absint( $last_order ) );
- }
- /**
- * Return the number of orders this customer has.
- *
- * @since 3.0.0
- * @param WC_Customer $customer Customer object.
- * @return integer
- */
- public function get_order_count( &$customer ) {
- $count = get_user_meta( $customer->get_id(), '_order_count', true );
- if ( '' === $count ) {
- global $wpdb;
- $count = $wpdb->get_var(
- // phpcs:disable WordPress.WP.PreparedSQL.NotPrepared
- "SELECT COUNT(*)
- FROM $wpdb->posts as posts
- LEFT JOIN {$wpdb->postmeta} AS meta ON posts.ID = meta.post_id
- WHERE meta.meta_key = '_customer_user'
- AND posts.post_type = 'shop_order'
- AND posts.post_status IN ( '" . implode( "','", array_map( 'esc_sql', array_keys( wc_get_order_statuses() ) ) ) . "' )
- AND meta_value = '" . esc_sql( $customer->get_id() ) . "'"
- // phpcs:enable
- );
- update_user_meta( $customer->get_id(), '_order_count', $count );
- }
- return absint( $count );
- }
- /**
- * Return how much money this customer has spent.
- *
- * @since 3.0.0
- * @param WC_Customer $customer Customer object.
- * @return float
- */
- public function get_total_spent( &$customer ) {
- $spent = apply_filters(
- 'woocommerce_customer_get_total_spent',
- get_user_meta( $customer->get_id(), '_money_spent', true ),
- $customer
- );
- if ( '' === $spent ) {
- global $wpdb;
- $statuses = array_map( 'esc_sql', wc_get_is_paid_statuses() );
- $spent = $wpdb->get_var(
- // phpcs:disable WordPress.WP.PreparedSQL.NotPrepared
- apply_filters(
- 'woocommerce_customer_get_total_spent_query',
- "SELECT SUM(meta2.meta_value)
- FROM $wpdb->posts as posts
- LEFT JOIN {$wpdb->postmeta} AS meta ON posts.ID = meta.post_id
- LEFT JOIN {$wpdb->postmeta} AS meta2 ON posts.ID = meta2.post_id
- WHERE meta.meta_key = '_customer_user'
- AND meta.meta_value = '" . esc_sql( $customer->get_id() ) . "'
- AND posts.post_type = 'shop_order'
- AND posts.post_status IN ( 'wc-" . implode( "','wc-", $statuses ) . "' )
- AND meta2.meta_key = '_order_total'",
- $customer
- )
- // phpcs:enable
- );
- if ( ! $spent ) {
- $spent = 0;
- }
- update_user_meta( $customer->get_id(), '_money_spent', $spent );
- }
- return wc_format_decimal( $spent, 2 );
- }
- /**
- * Search customers and return customer IDs.
- *
- * @param string $term Search term.
- * @param int|string $limit Limit search results.
- * @since 3.0.7
- *
- * @return array
- */
- public function search_customers( $term, $limit = '' ) {
- $results = apply_filters( 'woocommerce_customer_pre_search_customers', false, $term, $limit );
- if ( is_array( $results ) ) {
- return $results;
- }
- $query = new WP_User_Query(
- apply_filters(
- 'woocommerce_customer_search_customers', array(
- 'search' => '*' . esc_attr( $term ) . '*',
- 'search_columns' => array( 'user_login', 'user_url', 'user_email', 'user_nicename', 'display_name' ),
- 'fields' => 'ID',
- 'number' => $limit,
- ), $term, $limit, 'main_query'
- )
- );
- $query2 = new WP_User_Query(
- apply_filters(
- 'woocommerce_customer_search_customers', array(
- 'fields' => 'ID',
- 'number' => $limit,
- 'meta_query' => array(
- 'relation' => 'OR',
- array(
- 'key' => 'first_name',
- 'value' => $term,
- 'compare' => 'LIKE',
- ),
- array(
- 'key' => 'last_name',
- 'value' => $term,
- 'compare' => 'LIKE',
- ),
- ),
- ), $term, $limit, 'meta_query'
- )
- );
- $results = wp_parse_id_list( array_merge( (array) $query->get_results(), (array) $query2->get_results() ) );
- if ( $limit && count( $results ) > $limit ) {
- $results = array_slice( $results, 0, $limit );
- }
- return $results;
- }
- }
|