216 lines
5.5 KiB
C
216 lines
5.5 KiB
C
|
/* ide-search-reducer.c
|
||
|
*
|
||
|
* Copyright 2015-2019 Christian Hergert <chergert@redhat.com>
|
||
|
*
|
||
|
* This program 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.
|
||
|
*
|
||
|
* This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
|
||
|
*
|
||
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||
|
*/
|
||
|
|
||
|
#define G_LOG_DOMAIN "ide-search-reducer"
|
||
|
|
||
|
#include "config.h"
|
||
|
|
||
|
#include "ide-search-reducer.h"
|
||
|
#include "ide-search-result.h"
|
||
|
|
||
|
/**
|
||
|
* SECTION:ide-search-reducer
|
||
|
* @title: IdeSearchReducer
|
||
|
* @short_description: Reduces the number of search results created
|
||
|
*
|
||
|
* This is a helper structure for search engines to reduce the number
|
||
|
* of items they inflate when performing a search.
|
||
|
*
|
||
|
* Since: 3.32
|
||
|
*/
|
||
|
|
||
|
#define DEFAULT_MAX_ITEMS 1000
|
||
|
|
||
|
/**
|
||
|
* ide_search_reducer_init:
|
||
|
* @reducer: (out): The reducer to initialize
|
||
|
* @max_results: max result size for the reuslts
|
||
|
*
|
||
|
* Initializes a new #IdeSearchReducer to be used to reduce the number of
|
||
|
* search results that are created. This is generally just used to help
|
||
|
* keep search performance good.
|
||
|
*
|
||
|
* Since: 3.32
|
||
|
*/
|
||
|
void
|
||
|
ide_search_reducer_init (IdeSearchReducer *reducer,
|
||
|
gsize max_results)
|
||
|
{
|
||
|
g_return_if_fail (reducer != NULL);
|
||
|
|
||
|
reducer->sequence = g_sequence_new (g_object_unref);
|
||
|
reducer->max_results = max_results ? max_results : DEFAULT_MAX_ITEMS;
|
||
|
reducer->count = 0;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* ide_search_reducer_destroy:
|
||
|
* @reducer: a #IdeSearchReducer
|
||
|
*
|
||
|
* Frees the results.
|
||
|
*
|
||
|
* Since: 3.32
|
||
|
*/
|
||
|
void
|
||
|
ide_search_reducer_destroy (IdeSearchReducer *reducer)
|
||
|
{
|
||
|
g_return_if_fail (reducer != NULL);
|
||
|
|
||
|
if (reducer->sequence != NULL)
|
||
|
g_sequence_free (reducer->sequence);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* ide_search_reducer_free:
|
||
|
* @reducer: a #IdeSearchReducer
|
||
|
* @free_results: %TRUE if the results should be discarded
|
||
|
*
|
||
|
* Frees all items associated with the result set, unless @free_results is
|
||
|
* %FALSE and then the results are returned as an array.
|
||
|
*
|
||
|
* Returns: (nullable) (transfer container) (element-type IdeSearchResult):
|
||
|
* An array of #IdeSearchResult unless @free_results is %TRUE, then
|
||
|
* %NULL is returned.
|
||
|
*
|
||
|
* Since: 3.32
|
||
|
*/
|
||
|
GPtrArray *
|
||
|
ide_search_reducer_free (IdeSearchReducer *reducer,
|
||
|
gboolean free_results)
|
||
|
{
|
||
|
GPtrArray *ar;
|
||
|
GSequenceIter *iter;
|
||
|
GSequenceIter *end;
|
||
|
|
||
|
g_return_val_if_fail (reducer != NULL, NULL);
|
||
|
|
||
|
if (free_results)
|
||
|
{
|
||
|
ide_search_reducer_destroy (reducer);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
ar = g_ptr_array_new_with_free_func (g_object_unref);
|
||
|
|
||
|
end = g_sequence_get_end_iter (reducer->sequence);
|
||
|
|
||
|
for (iter = g_sequence_get_begin_iter (reducer->sequence);
|
||
|
iter != end;
|
||
|
iter = g_sequence_iter_next (iter))
|
||
|
{
|
||
|
IdeSearchResult *result = g_sequence_get (iter);
|
||
|
g_ptr_array_add (ar, g_object_ref (result));
|
||
|
}
|
||
|
|
||
|
g_sequence_free (reducer->sequence);
|
||
|
|
||
|
reducer->sequence = NULL;
|
||
|
reducer->max_results = 0;
|
||
|
reducer->count = 0;
|
||
|
|
||
|
return ar;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* ide_search_reducer_take:
|
||
|
* @reducer: an #IdeSearchReducer
|
||
|
* @result: (transfer full): an #IdeSearchResult
|
||
|
*
|
||
|
* Like ide_search_reducer_push() but takes ownership of @result by
|
||
|
* stealing the reference.
|
||
|
*
|
||
|
* Since: 3.32
|
||
|
*/
|
||
|
void
|
||
|
ide_search_reducer_take (IdeSearchReducer *reducer,
|
||
|
IdeSearchResult *result)
|
||
|
{
|
||
|
g_assert (reducer != NULL);
|
||
|
g_assert (IDE_IS_SEARCH_RESULT (result));
|
||
|
|
||
|
if (reducer->count == reducer->max_results)
|
||
|
/* Remove lowest score */
|
||
|
g_sequence_remove (g_sequence_get_begin_iter (reducer->sequence));
|
||
|
else
|
||
|
reducer->count++;
|
||
|
|
||
|
g_sequence_insert_sorted (reducer->sequence,
|
||
|
result,
|
||
|
(GCompareDataFunc)ide_search_result_compare,
|
||
|
NULL);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* ide_search_reducer_push:
|
||
|
* @reducer: an #IdeSearchReducer
|
||
|
* @result: an #IdeSearchResult
|
||
|
*
|
||
|
* Adds result to the set unless it scores too low.
|
||
|
*
|
||
|
* Since: 3.32
|
||
|
*/
|
||
|
void
|
||
|
ide_search_reducer_push (IdeSearchReducer *reducer,
|
||
|
IdeSearchResult *result)
|
||
|
{
|
||
|
g_assert (reducer != NULL);
|
||
|
g_assert (IDE_IS_SEARCH_RESULT (result));
|
||
|
|
||
|
ide_search_reducer_take (reducer, g_object_ref (result));
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* ide_search_reducer_accepts:
|
||
|
* @reducer: a #IdeSearchReducer
|
||
|
* @score: a score for the result
|
||
|
*
|
||
|
* This helper allows you to check if a result with @score would be allowed
|
||
|
* into the result set, or if the score is too low. This helps in situations
|
||
|
* where you want to avoid inflating an #IdeSearchResult unless necessary.
|
||
|
*
|
||
|
* Returns: %TRUE if there is space for a result with a score of @score.
|
||
|
*
|
||
|
* Since: 3.32
|
||
|
*/
|
||
|
gboolean
|
||
|
ide_search_reducer_accepts (IdeSearchReducer *reducer,
|
||
|
gfloat score)
|
||
|
{
|
||
|
GSequenceIter *iter;
|
||
|
|
||
|
g_return_val_if_fail (reducer, FALSE);
|
||
|
|
||
|
if (reducer->count < reducer->max_results)
|
||
|
return TRUE;
|
||
|
|
||
|
iter = g_sequence_get_begin_iter (reducer->sequence);
|
||
|
|
||
|
if (iter != NULL)
|
||
|
{
|
||
|
IdeSearchResult *result;
|
||
|
|
||
|
result = g_sequence_get (iter);
|
||
|
if (result)
|
||
|
return score > ide_search_result_get_score (result);
|
||
|
}
|
||
|
|
||
|
return FALSE;
|
||
|
}
|