2012-04-15 20:28:46 -07:00
|
|
|
// Copyright (C) 2011, 2012 Strahinja Val Markovic <val@markovic.io>
|
2012-04-15 16:57:10 -07:00
|
|
|
//
|
|
|
|
// This file is part of YouCompleteMe.
|
|
|
|
//
|
|
|
|
// YouCompleteMe is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// YouCompleteMe is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with YouCompleteMe. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
#include "Result.h"
|
2012-05-06 00:01:32 -07:00
|
|
|
#include "standard.h"
|
2012-04-15 16:57:10 -07:00
|
|
|
#include "Utils.h"
|
|
|
|
#include <boost/algorithm/string.hpp>
|
2012-06-24 17:58:20 -07:00
|
|
|
#include <algorithm>
|
2012-04-15 16:57:10 -07:00
|
|
|
|
|
|
|
using boost::algorithm::istarts_with;
|
|
|
|
|
2013-01-19 20:10:52 -08:00
|
|
|
namespace YouCompleteMe {
|
2012-04-15 16:57:10 -07:00
|
|
|
|
2013-01-19 20:10:52 -08:00
|
|
|
namespace {
|
2012-04-15 16:57:10 -07:00
|
|
|
|
2012-07-02 11:10:28 -07:00
|
|
|
int LongestCommonSubsequenceLength( const std::string &first,
|
2013-01-19 20:10:52 -08:00
|
|
|
const std::string &second ) {
|
2012-06-24 18:13:35 -07:00
|
|
|
const std::string &longer = first.size() > second.size() ? first : second;
|
|
|
|
const std::string &shorter = first.size() > second.size() ? second : first;
|
2012-06-24 17:58:20 -07:00
|
|
|
|
|
|
|
int longer_len = longer.size();
|
|
|
|
int shorter_len = shorter.size();
|
|
|
|
|
|
|
|
std::vector<int> previous( shorter_len + 1, 0 );
|
|
|
|
std::vector<int> current( shorter_len + 1, 0 );
|
|
|
|
|
2013-01-19 20:10:52 -08:00
|
|
|
for ( int i = 0; i < longer_len; ++i ) {
|
|
|
|
for ( int j = 0; j < shorter_len; ++j ) {
|
2012-06-24 18:13:35 -07:00
|
|
|
if ( toupper( longer[ i ] ) == toupper( shorter[ j ] ) )
|
2012-06-24 17:58:20 -07:00
|
|
|
current[ j + 1 ] = previous[ j ] + 1;
|
|
|
|
else
|
|
|
|
current[ j + 1 ] = std::max( current[ j ], previous[ j + 1 ] );
|
|
|
|
}
|
|
|
|
|
2013-01-19 20:10:52 -08:00
|
|
|
for ( int j = 0; j < shorter_len; ++j ) {
|
2012-06-24 17:58:20 -07:00
|
|
|
previous[ j + 1 ] = current[ j + 1 ];
|
|
|
|
}
|
2012-04-15 16:57:10 -07:00
|
|
|
}
|
|
|
|
|
2012-06-24 17:58:20 -07:00
|
|
|
return current[ shorter_len ];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int NumWordBoundaryCharMatches( const std::string &query,
|
2013-01-19 20:10:52 -08:00
|
|
|
const std::string &word_boundary_chars ) {
|
|
|
|
return LongestCommonSubsequenceLength( query, word_boundary_chars );
|
2012-04-15 16:57:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
} // unnamed namespace
|
|
|
|
|
2012-07-19 21:17:39 -07:00
|
|
|
Result::Result()
|
|
|
|
:
|
|
|
|
query_is_empty_( true ),
|
|
|
|
is_subsequence_( false ),
|
|
|
|
first_char_same_in_query_and_text_( false ),
|
|
|
|
ratio_of_word_boundary_chars_in_query_( 0 ),
|
|
|
|
word_boundary_char_utilization_( 0 ),
|
|
|
|
query_is_candidate_prefix_( false ),
|
|
|
|
text_is_lowercase_( false ),
|
|
|
|
char_match_index_sum_( 0 ),
|
2013-01-19 20:10:52 -08:00
|
|
|
text_( NULL ) {
|
2012-07-19 21:17:39 -07:00
|
|
|
}
|
|
|
|
|
2012-04-15 16:57:10 -07:00
|
|
|
|
|
|
|
Result::Result( bool is_subsequence )
|
|
|
|
:
|
2012-07-15 19:40:24 -07:00
|
|
|
query_is_empty_( true ),
|
2012-04-15 16:57:10 -07:00
|
|
|
is_subsequence_( is_subsequence ),
|
|
|
|
first_char_same_in_query_and_text_( false ),
|
|
|
|
ratio_of_word_boundary_chars_in_query_( 0 ),
|
|
|
|
word_boundary_char_utilization_( 0 ),
|
|
|
|
query_is_candidate_prefix_( false ),
|
2012-04-16 22:13:05 -07:00
|
|
|
text_is_lowercase_( false ),
|
2012-06-24 15:35:00 -07:00
|
|
|
char_match_index_sum_( 0 ),
|
2013-01-19 20:10:52 -08:00
|
|
|
text_( NULL ) {
|
2012-04-15 16:57:10 -07:00
|
|
|
}
|
|
|
|
|
2012-07-19 21:17:39 -07:00
|
|
|
|
2012-04-15 16:57:10 -07:00
|
|
|
Result::Result( bool is_subsequence,
|
|
|
|
const std::string *text,
|
2012-04-16 22:13:05 -07:00
|
|
|
bool text_is_lowercase,
|
2012-06-24 15:35:00 -07:00
|
|
|
int char_match_index_sum,
|
2012-04-15 16:57:10 -07:00
|
|
|
const std::string &word_boundary_chars,
|
|
|
|
const std::string &query )
|
|
|
|
:
|
2012-07-15 19:40:24 -07:00
|
|
|
query_is_empty_( true ),
|
2012-04-15 16:57:10 -07:00
|
|
|
is_subsequence_( is_subsequence ),
|
|
|
|
first_char_same_in_query_and_text_( false ),
|
|
|
|
ratio_of_word_boundary_chars_in_query_( 0 ),
|
|
|
|
word_boundary_char_utilization_( 0 ),
|
|
|
|
query_is_candidate_prefix_( false ),
|
2012-04-16 22:13:05 -07:00
|
|
|
text_is_lowercase_( text_is_lowercase ),
|
2012-06-24 15:35:00 -07:00
|
|
|
char_match_index_sum_( char_match_index_sum ),
|
2013-01-19 20:10:52 -08:00
|
|
|
text_( text ) {
|
2012-04-15 16:57:10 -07:00
|
|
|
if ( is_subsequence )
|
2012-04-16 22:13:05 -07:00
|
|
|
SetResultFeaturesFromQuery( word_boundary_chars, query );
|
2012-04-15 16:57:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-01-19 20:10:52 -08:00
|
|
|
bool Result::operator< ( const Result &other ) const {
|
2012-04-15 16:57:10 -07:00
|
|
|
// Yes, this is ugly but it also needs to be fast. Since this is called a
|
|
|
|
// bazillion times, we have to make sure only the required comparisons are
|
|
|
|
// made, and no more.
|
|
|
|
|
2013-01-19 20:10:52 -08:00
|
|
|
if ( !query_is_empty_ ) {
|
2012-07-15 19:40:24 -07:00
|
|
|
if ( first_char_same_in_query_and_text_ !=
|
2013-01-19 20:10:52 -08:00
|
|
|
other.first_char_same_in_query_and_text_ ) {
|
2012-07-15 19:40:24 -07:00
|
|
|
return first_char_same_in_query_and_text_;
|
|
|
|
}
|
2012-04-15 16:57:10 -07:00
|
|
|
|
2012-07-15 19:40:24 -07:00
|
|
|
bool equal_wb_ratios = AlmostEqual(
|
2013-01-19 20:10:52 -08:00
|
|
|
ratio_of_word_boundary_chars_in_query_,
|
|
|
|
other.ratio_of_word_boundary_chars_in_query_ );
|
2012-04-15 16:57:10 -07:00
|
|
|
|
2012-07-15 19:40:24 -07:00
|
|
|
bool equal_wb_utilization = AlmostEqual(
|
2013-01-19 20:10:52 -08:00
|
|
|
word_boundary_char_utilization_,
|
|
|
|
other.word_boundary_char_utilization_ );
|
2012-07-15 19:40:24 -07:00
|
|
|
|
|
|
|
if ( AlmostEqual( ratio_of_word_boundary_chars_in_query_, 1.0 ) ||
|
2013-01-19 20:10:52 -08:00
|
|
|
AlmostEqual( other.ratio_of_word_boundary_chars_in_query_, 1.0 ) ) {
|
|
|
|
if ( !equal_wb_ratios ) {
|
2012-07-15 19:40:24 -07:00
|
|
|
return ratio_of_word_boundary_chars_in_query_ >
|
2013-01-19 20:10:52 -08:00
|
|
|
other.ratio_of_word_boundary_chars_in_query_;
|
2012-07-15 19:40:24 -07:00
|
|
|
}
|
|
|
|
|
2013-01-19 20:10:52 -08:00
|
|
|
else {
|
2012-07-15 19:40:24 -07:00
|
|
|
if ( !equal_wb_utilization )
|
|
|
|
return word_boundary_char_utilization_ >
|
2013-01-19 20:10:52 -08:00
|
|
|
other.word_boundary_char_utilization_;
|
2012-07-15 19:40:24 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( query_is_candidate_prefix_ != other.query_is_candidate_prefix_ )
|
|
|
|
return query_is_candidate_prefix_;
|
2012-04-15 16:57:10 -07:00
|
|
|
|
2013-01-19 20:10:52 -08:00
|
|
|
if ( !equal_wb_ratios ) {
|
2012-04-15 16:57:10 -07:00
|
|
|
return ratio_of_word_boundary_chars_in_query_ >
|
2013-01-19 20:10:52 -08:00
|
|
|
other.ratio_of_word_boundary_chars_in_query_;
|
2012-04-15 16:57:10 -07:00
|
|
|
}
|
|
|
|
|
2013-01-19 20:10:52 -08:00
|
|
|
else {
|
2012-04-15 16:57:10 -07:00
|
|
|
if ( !equal_wb_utilization )
|
|
|
|
return word_boundary_char_utilization_ >
|
2013-01-19 20:10:52 -08:00
|
|
|
other.word_boundary_char_utilization_;
|
2012-04-15 16:57:10 -07:00
|
|
|
}
|
|
|
|
|
2012-07-15 19:40:24 -07:00
|
|
|
if ( char_match_index_sum_ != other.char_match_index_sum_ )
|
|
|
|
return char_match_index_sum_ < other.char_match_index_sum_;
|
2012-04-15 16:57:10 -07:00
|
|
|
|
2012-07-15 19:40:24 -07:00
|
|
|
if ( text_->length() != other.text_->length() )
|
|
|
|
return text_->length() < other.text_->length();
|
2012-04-15 16:57:10 -07:00
|
|
|
|
2012-07-15 19:40:24 -07:00
|
|
|
if ( text_is_lowercase_ != other.text_is_lowercase_ )
|
|
|
|
return text_is_lowercase_;
|
2012-04-15 16:57:10 -07:00
|
|
|
}
|
|
|
|
|
2012-07-15 19:40:24 -07:00
|
|
|
// Lexicographic comparison
|
2012-04-15 16:57:10 -07:00
|
|
|
return *text_ < *other.text_;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Result::SetResultFeaturesFromQuery(
|
2013-01-19 20:10:52 -08:00
|
|
|
const std::string &word_boundary_chars,
|
|
|
|
const std::string &query ) {
|
2012-07-15 19:40:24 -07:00
|
|
|
query_is_empty_ = query.empty();
|
|
|
|
|
2012-04-15 16:57:10 -07:00
|
|
|
if ( query.empty() || text_->empty() )
|
|
|
|
return;
|
|
|
|
|
2012-04-15 20:10:39 -07:00
|
|
|
first_char_same_in_query_and_text_ =
|
2013-01-19 20:10:52 -08:00
|
|
|
toupper( query[ 0 ] ) == toupper( ( *text_ )[ 0 ] );
|
2012-04-15 20:10:39 -07:00
|
|
|
int num_wb_matches = NumWordBoundaryCharMatches( query,
|
2012-04-15 16:57:10 -07:00
|
|
|
word_boundary_chars );
|
|
|
|
ratio_of_word_boundary_chars_in_query_ =
|
|
|
|
num_wb_matches / static_cast< double >( query.length() );
|
|
|
|
word_boundary_char_utilization_ =
|
|
|
|
num_wb_matches / static_cast< double >( word_boundary_chars.length() );
|
|
|
|
query_is_candidate_prefix_ = istarts_with( *text_, query );
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace YouCompleteMe
|