/* * autocomplete.c * * Copyright (C) 2012 - 2015 James Booth * * This file is part of Profanity. * * Profanity 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. * * Profanity 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 Profanity. If not, see . * * In addition, as a special exception, the copyright holders give permission to * link the code of portions of this program with the OpenSSL library under * certain conditions as described in each individual source file, and * distribute linked combinations including the two. * * You must obey the GNU General Public License in all respects for all of the * code used other than OpenSSL. If you modify file(s) with this exception, you * may extend this exception to your version of the file(s), but you are not * obligated to do so. If you do not wish to do so, delete this exception * statement from your version. If you delete this exception statement from all * source files in the program, then also delete it here. * */ #include #include #include #include "common.h" #include "tools/autocomplete.h" #include "tools/parser.h" struct autocomplete_t { GSList *items; GSList *last_found; gchar *search_str; }; static gchar* _search_from(Autocomplete ac, GSList *curr, gboolean quote); Autocomplete autocomplete_new(void) { Autocomplete new = malloc(sizeof(struct autocomplete_t)); new->items = NULL; new->last_found = NULL; new->search_str = NULL; return new; } void autocomplete_clear(Autocomplete ac) { if (ac) { g_slist_free_full(ac->items, free); ac->items = NULL; autocomplete_reset(ac); } } void autocomplete_reset(Autocomplete ac) { ac->last_found = NULL; FREE_SET_NULL(ac->search_str); } void autocomplete_free(Autocomplete ac) { if (ac) { autocomplete_clear(ac); free(ac); } } gint autocomplete_length(Autocomplete ac) { if (!ac) { return 0; } else if (!ac->items) { return 0; } else { return g_slist_length(ac->items); } } void autocomplete_add(Autocomplete ac, const char *item) { if (ac) { char *item_cpy; GSList *curr = g_slist_find_custom(ac->items, item, (GCompareFunc)strcmp); // if item already exists if (curr) { return; } item_cpy = strdup(item); ac->items = g_slist_insert_sorted(ac->items, item_cpy, (GCompareFunc)strcmp); } return; } void autocomplete_remove(Autocomplete ac, const char *const item) { if (ac) { GSList *curr = g_slist_find_custom(ac->items, item, (GCompareFunc)strcmp); if (!curr) { return; } // reset last found if it points to the item to be removed if (ac->last_found == curr) { ac->last_found = NULL; } free(curr->data); ac->items = g_slist_delete_link(ac->items, curr); } return; } GSList* autocomplete_create_list(Autocomplete ac) { GSList *copy = NULL; GSList *curr = ac->items; while(curr) { copy = g_slist_append(copy, strdup(curr->data)); curr = g_slist_next(curr); } return copy; } gboolean autocomplete_contains(Autocomplete ac, const char *value) { GSList *curr = ac->items; while(curr) { if (strcmp(curr->data, value) == 0) { return TRUE; } curr = g_slist_next(curr); } return FALSE; } gchar* autocomplete_complete(Autocomplete ac, const gchar *search_str, gboolean quote) { gchar *found = NULL; // no autocomplete to search if (!ac) { return NULL; } // no items to search if (!ac->items) { return NULL; } // first search attempt if (!ac->last_found) { if (ac->search_str) { FREE_SET_NULL(ac->search_str); } ac->search_str = strdup(search_str); found = _search_from(ac, ac->items, quote); return found; // subsequent search attempt } else { // search from here+1 to end found = _search_from(ac, g_slist_next(ac->last_found), quote); if (found) { return found; } // search from beginning found = _search_from(ac, ac->items, quote); if (found) { return found; } // we found nothing, reset search autocomplete_reset(ac); return NULL; } } char* autocomplete_param_with_func(const char *const input, char *command, autocomplete_func func) { GString *auto_msg = NULL; char *result = NULL; char command_cpy[strlen(command) + 2]; sprintf(command_cpy, "%s ", command); int len = strlen(command_cpy); if ((strncmp(input, command_cpy, len) == 0) && (strlen(input) > len)) { int i; int inp_len = strlen(input); char prefix[inp_len]; for(i = len; i < inp_len; i++) { prefix[i-len] = input[i]; } prefix[inp_len - len] = '\0'; char *found = func(prefix); if (found) { auto_msg = g_string_new(command_cpy); g_string_append(auto_msg, found); free(found); result = auto_msg->str; g_string_free(auto_msg, FALSE); } } return result; } char* autocomplete_param_with_ac(const char *const input, char *command, Autocomplete ac, gboolean quote) { GString *auto_msg = NULL; char *result = NULL; char *command_cpy = malloc(strlen(command) + 2); sprintf(command_cpy, "%s ", command); int len = strlen(command_cpy); int inp_len = strlen(input); if ((strncmp(input, command_cpy, len) == 0) && (strlen(input) > len)) { int i; char prefix[inp_len]; for(i = len; i < inp_len; i++) { prefix[i-len] = input[i]; } prefix[inp_len - len] = '\0'; char *found = autocomplete_complete(ac, prefix, quote); if (found) { auto_msg = g_string_new(command_cpy); g_string_append(auto_msg, found); free(found); result = auto_msg->str; g_string_free(auto_msg, FALSE); } } free(command_cpy); return result; } char* autocomplete_param_no_with_func(const char *const input, char *command, int arg_number, autocomplete_func func) { if (strncmp(input, command, strlen(command)) == 0 && (strlen(input) > strlen(command))) { GString *result_str = NULL; // count tokens properly int num_tokens = count_tokens(input); // if correct number of tokens, then candidate for autocompletion of last param if (num_tokens == arg_number) { gchar *start_str = get_start(input, arg_number); gchar *comp_str = g_strdup(&input[strlen(start_str)]); // autocomplete param if (comp_str) { char *found = func(comp_str); if (found) { result_str = g_string_new(""); g_string_append(result_str, start_str); g_string_append(result_str, found); char *result = result_str->str; g_string_free(result_str, FALSE); return result; } } } } return NULL; } static gchar* _search_from(Autocomplete ac, GSList *curr, gboolean quote) { while(curr) { // match found if (strncmp(curr->data, ac->search_str, strlen(ac->search_str)) == 0) { // set pointer to last found ac->last_found = curr; // if contains space, quote before returning if (quote && g_strrstr(curr->data, " ")) { GString *quoted = g_string_new("\""); g_string_append(quoted, curr->data); g_string_append(quoted, "\""); gchar *result = quoted->str; g_string_free(quoted, FALSE); return result; // otherwise just return the string } else { return strdup(curr->data); } } curr = g_slist_next(curr); } return NULL; }