setup(); } return $instance; } /** * Constructor */ public function __construct() {} /** * Setup actions and filters for all things settings */ public function setup() { $this->data['settings'] = get_option( 'aws_settings' ); add_action( 'wp_ajax_aws_action', array( $this, 'action_callback' ) ); add_action( 'wp_ajax_nopriv_aws_action', array( $this, 'action_callback' ) ); } /** * Get caching option name */ private function get_cache_name( $s ) { $cache_option_name = 'aws_search_term_' . $s; if ( has_filter('wpml_current_language') ) { $current_lang = apply_filters('wpml_current_language', NULL); if ( $current_lang ) { $cache_option_name = $cache_option_name . '_' . $current_lang; } } return $cache_option_name; } public function search( $keyword = '' ) { global $wpdb; $cache = AWS()->get_settings( 'cache' ); $s = $keyword ? esc_attr( $keyword ) : esc_attr( $_POST['keyword'] ); $s = stripslashes( $s ); $s = str_replace( array( "\r", "\n" ), '', $s ); $cache_option_name = $this->get_cache_name( $s ); if ( $cache === 'true' ) { // Check if value was already cached if ( $cached_value = get_option( $cache_option_name ) ) { $cached_value['cache'] = 'cached'; return $cached_value; } } $show_cats = AWS()->get_settings( 'show_cats' ); $show_tags = AWS()->get_settings( 'show_tags' ); $results_num = $keyword ? 100 : AWS()->get_settings( 'results_num' ); $search_in = AWS()->get_settings( 'search_in' ); $outofstock = AWS()->get_settings( 'outofstock' ); $search_in_arr = explode( ',', AWS()->get_settings( 'search_in' ) ); // Search in title if all options is disabled if ( ! $search_in ) { $search_in_arr = array( 'title' ); } $categories_array = array(); $tags_array = array(); $this->data['s'] = $s; $this->data['results_num'] = $results_num ? $results_num : 10; $this->data['search_terms'] = array(); $this->data['search_terms'] = array_unique( explode( ' ', $s ) ); $this->data['search_in'] = $search_in_arr; $this->data['outofstock'] = $outofstock; $posts_ids = $this->query_index_table(); $products_array = $this->get_products( $posts_ids ); if ( $show_cats === 'true' ) { $categories_array = $this->get_taxonomies( 'product_cat' ); } if ( $show_tags === 'true' ) { $tags_array = $this->get_taxonomies( 'product_tag' ); } $result_array = array( 'cats' => $categories_array, 'tags' => $tags_array, 'products' => $products_array ); if ( $cache === 'true' ) { update_option( $cache_option_name, $result_array ); } return $result_array; } /* * AJAX call action callback */ public function action_callback() { echo json_encode( $this->search() ); die; } /* * Query in index table */ private function query_index_table() { global $wpdb; $table_name = $wpdb->prefix . AWS_INDEX_TABLE_NAME; $search_in_arr = $this->data['search_in']; $results_num = $this->data['results_num']; $outofstock = $this->data['outofstock']; $reindex_version = get_option( 'aws_reindex_version' ); $query = array(); $query['search'] = ''; $query['source'] = ''; $query['relevance'] = ''; $query['stock'] = ''; $query['visibility'] = ''; $query['lang'] = ''; $search_array = array(); $source_array = array(); $relevance_array = array(); $new_relevance_array = array(); foreach ( $this->data['search_terms'] as $search_term ) { $search_term_len = strlen( $search_term ); $relevance_title = 200 + 20 * $search_term_len; $relevance_content = 35 + 4 * $search_term_len; $relevance_title_like = 40 + 2 * $search_term_len; $relevance_content_like = 35 + 1 * $search_term_len; $like = '%' . $wpdb->esc_like( $search_term ) . '%'; if ( $search_term_len > 1 ) { $search_array[] = $wpdb->prepare( '( term LIKE %s )', $like ); } else { $search_array[] = $wpdb->prepare( '( term = "%s" )', $search_term ); } foreach ( $search_in_arr as $search_in_term ) { switch ( $search_in_term ) { case 'title': $relevance_array['title'][] = $wpdb->prepare( "( case when ( term_source = 'title' AND term = '%s' ) then {$relevance_title} * count else 0 end )", $search_term ); $relevance_array['title'][] = $wpdb->prepare( "( case when ( term_source = 'title' AND term LIKE %s ) then {$relevance_title_like} * count else 0 end )", $like ); break; case 'content': $relevance_array['content'][] = $wpdb->prepare( "( case when ( term_source = 'content' AND term = '%s' ) then {$relevance_content} * count else 0 end )", $search_term ); $relevance_array['content'][] = $wpdb->prepare( "( case when ( term_source = 'content' AND term LIKE %s ) then {$relevance_content_like} * count else 0 end )", $like ); break; case 'excerpt': $relevance_array['content'][] = $wpdb->prepare( "( case when ( term_source = 'excerpt' AND term = '%s' ) then {$relevance_content} * count else 0 end )", $search_term ); $relevance_array['content'][] = $wpdb->prepare( "( case when ( term_source = 'excerpt' AND term LIKE %s ) then {$relevance_content_like} * count else 0 end )", $like ); break; case 'category': $relevance_array['category'][] = $wpdb->prepare( "( case when ( term_source = 'category' AND term = '%s' ) then 35 else 0 end )", $search_term ); $relevance_array['category'][] = $wpdb->prepare( "( case when ( term_source = 'category' AND term LIKE %s ) then 5 else 0 end )", $like ); break; case 'tag': $relevance_array['tag'][] = $wpdb->prepare( "( case when ( term_source = 'tag' AND term = '%s' ) then 35 else 0 end )", $search_term ); $relevance_array['tag'][] = $wpdb->prepare( "( case when ( term_source = 'tag' AND term LIKE %s ) then 5 else 0 end )", $like ); break; } } } // Sort 'relevance' queries in the array by search priority foreach ( $search_in_arr as $search_in_item ) { if ( isset( $relevance_array[$search_in_item] ) ) { $new_relevance_array[$search_in_item] = implode( ' + ', $relevance_array[$search_in_item] ); } } foreach ( $search_in_arr as $search_in_term ) { $source_array[] = "term_source = '{$search_in_term}'"; } $query['relevance'] .= sprintf( ' (SUM( %s )) ', implode( ' + ', $new_relevance_array ) ); $query['search'] .= sprintf( ' AND ( %s )', implode( ' OR ', $search_array ) ); $query['source'] .= sprintf( ' AND ( %s )', implode( ' OR ', $source_array ) ); if ( $reindex_version && version_compare( $reindex_version, '1.16', '>=' ) ) { if ( $outofstock !== 'true' ) { $query['stock'] .= " AND in_stock = 1"; } $query['visibility'] .= " AND NOT visibility LIKE '%hidden%'"; } if ( $reindex_version && version_compare( $reindex_version, '1.20', '>=' ) ) { if ( has_filter('wpml_current_language') ) { $current_lang = apply_filters( 'wpml_current_language', NULL ); if ( $current_lang ) { $query['lang'] .= $wpdb->prepare( " AND lang LIKE %s", $current_lang ); } } } $sql = "SELECT distinct ID, {$query['relevance']} as relevance FROM {$table_name} WHERE type = 'product' {$query['source']} {$query['search']} {$query['stock']} {$query['visibility']} {$query['lang']} GROUP BY ID ORDER BY relevance DESC LIMIT 0, {$results_num} "; $posts_ids = $this->get_posts_ids( $sql ); return $posts_ids; } /* * Get array of included to search result posts ids */ private function get_posts_ids( $sql ) { global $wpdb; $posts_ids = array(); $search_results = $wpdb->get_results( $sql ); if ( !empty( $search_results ) && !is_wp_error( $search_results ) && is_array( $search_results ) ) { foreach ( $search_results as $search_result ) { $post_id = intval( $search_result->ID ); if ( ! in_array( $post_id, $posts_ids ) ) { $posts_ids[] = $post_id; } } } unset( $search_results ); return $posts_ids; } /* * Get products info */ private function get_products( $posts_ids ) { $products_array = array(); if ( count( $posts_ids ) > 0 ) { $show_excerpt = AWS()->get_settings( 'show_excerpt' ); $excerpt_source = AWS()->get_settings( 'desc_source' ); $excerpt_length = AWS()->get_settings( 'excerpt_length' ); $mark_search_words = AWS()->get_settings( 'mark_words' ); $show_price = AWS()->get_settings( 'show_price' ); $show_sale = AWS()->get_settings( 'show_sale' ); $show_image = AWS()->get_settings( 'show_image' ); $show_sku = AWS()->get_settings( 'show_sku' ); foreach ( $posts_ids as $post_id ) { $product = wc_get_product( $post_id ); $post_data = get_post( $post_id ); $title = $product->get_title(); $title = AWS_Helpers::html2txt( $title ); $excerpt = ''; $price = ''; $on_sale = ''; $image = ''; $sku = ''; if ( $show_excerpt === 'true' ) { $excerpt = ( $excerpt_source === 'excerpt' && $post_data->post_excerpt ) ? $post_data->post_excerpt : $post_data->post_content; $excerpt = AWS_Helpers::html2txt( $excerpt ); $excerpt = str_replace('"', "'", $excerpt); $excerpt = strip_shortcodes( $excerpt ); } if ( $mark_search_words === 'true' ) { $marked_content = $this->mark_search_words( $title, $excerpt ); $title = $marked_content['title']; $excerpt = $marked_content['content']; } else { $excerpt = wp_trim_words( $excerpt, $excerpt_length, '...' ); } if ( $show_price === 'true' ) { $price = $product->get_price_html(); } if ( $show_sale === 'true' ) { $on_sale = $product->is_on_sale(); } if ( $show_image === 'true' ) { $image_id = $product->get_image_id(); $image_attributes = wp_get_attachment_image_src( $image_id ); $image = $image_attributes[0]; } if ( $show_sku === 'true' ) { $sku = $product->get_sku(); } // $categories = $product->get_categories( ',' ); // $tags = $product->get_tags( ',' ); $new_result = array( 'title' => $title, 'excerpt' => $excerpt, 'link' => get_permalink( $post_id ), 'image' => $image, 'price' => $price, 'on_sale' => $on_sale, 'sku' => $sku, 'post_data' => $post_data ); $products_array[] = $new_result; } } return $products_array; } /* * Mark search words */ private function mark_search_words( $title, $content ) { $pattern = array(); $exact_pattern = array(); $exact_words = array(); $words = array(); foreach( $this->data['search_terms'] as $search_in ) { $exact_words[] = '\b' . $search_in . '\b'; $exact_pattern[] = '(\b' . $search_in . '\b)+'; if ( strlen( $search_in ) > 1 ) { $pattern[] = '(' . $search_in . ')+'; $words[] = $search_in; } else { $pattern[] = '\b[' . $search_in . ']{1}\b'; $words[] = '\b' . $search_in . '\b'; } } usort( $exact_words, array( $this, 'sort_by_length' ) ); $exact_words = implode( '|', $exact_words ); usort( $words, array( $this, 'sort_by_length' ) ); $words = implode( '|', $words ); usort( $exact_pattern, array( $this, 'sort_by_length' ) ); $exact_pattern = implode( '|', $exact_pattern ); $exact_pattern = sprintf( '/%s/i', $exact_pattern ); usort( $pattern, array( $this, 'sort_by_length' ) ); $pattern = implode( '|', $pattern ); $pattern = sprintf( '/%s/i', $pattern ); preg_match( '/([^.?!]*?)(' . $exact_words . '){1}(.*?[.!?])/i', $content, $matches ); if ( ! isset( $matches[0] ) ) { preg_match( '/([^.?!]*?)(' . $words . '){1}(.*?[.!?])/i', $content, $matches ); } if ( ! isset( $matches[0] ) ) { preg_match( '/([^.?!]*?)(.*?)(.*?[.!?])/i', $content, $matches ); } if ( isset( $matches[0] ) ) { $content = $matches[0]; // Trim to long content if (str_word_count(strip_tags($content)) > 34) { if (str_word_count(strip_tags($matches[3])) > 34) { $matches[3] = wp_trim_words($matches[3], 30, '...'); } $content = '...' . $matches[2] . $matches[3]; } } else { $content = ''; } $title_has_exact = preg_match( '/(' . $exact_words . '){1}/i', $title ); $content_has_exact = preg_match( '/(' . $exact_words . '){1}/i', $content ); if ( $title_has_exact === 1 || $content_has_exact === 1 ) { $title = preg_replace($exact_pattern, '${0}', $title ); $content = preg_replace($exact_pattern, '${0}', $content ); } else { $title = preg_replace($pattern, '${0}', $title ); $content = preg_replace( $pattern, '${0}', $content ); } return array( 'title' => $title, 'content' => $content ); } /* * Sort array by its values length */ private function sort_by_length( $a, $b ) { return strlen( $b ) - strlen( $a ); } /* * Query product taxonomies */ private function get_taxonomies( $taxonomy ) { global $wpdb; $result_array = array(); $search_array = array(); $excludes = ''; $search_query = ''; foreach ( $this->data['search_terms'] as $search_term ) { $like = '%' . $wpdb->esc_like($search_term) . '%'; $search_array[] = $wpdb->prepare('( name LIKE %s )', $like); } $search_query .= sprintf( ' AND ( %s )', implode( ' OR ', $search_array ) ); $sql = " SELECT distinct($wpdb->terms.name), $wpdb->terms.term_id, $wpdb->term_taxonomy.taxonomy, $wpdb->term_taxonomy.count FROM $wpdb->terms , $wpdb->term_taxonomy WHERE 1 = 1 {$search_query} AND $wpdb->term_taxonomy.taxonomy = '{$taxonomy}' AND $wpdb->term_taxonomy.term_id = $wpdb->terms.term_id $excludes LIMIT 0, 10"; $search_results = $wpdb->get_results( $sql ); if ( ! empty( $search_results ) && !is_wp_error( $search_results ) ) { foreach ( $search_results as $result ) { if ( function_exists( 'wpml_object_id_filter' ) ) { $term = wpml_object_id_filter( $result->term_id, $result->taxonomy ); if ( $term != $result->term_id ) { continue; } } else { $term = get_term( $result->term_id, $result->taxonomy ); } if ( $term != null && !is_wp_error( $term ) ) { $term_link = get_term_link( $term ); } else { continue; } $new_result = array( 'name' => $result->name, 'count' => $result->count, 'link' => $term_link ); $result_array[] = $new_result; } } return $result_array; } } endif; AWS_Search::factory(); function aws_search( $keyword = '' ) { return AWS_Search::factory()->search( $keyword ); }