123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210 |
- /*
- Search text engine.
- Glob-style pattern matching
- Copyright (C) 2009-2025
- Free Software Foundation, Inc.
- Written by:
- Slava Zanko <slavazanko@gmail.com>, 2009.
- This file is part of the Midnight Commander.
- The Midnight Commander 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.
- The Midnight Commander 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/>.
- */
- #include <config.h>
- #include "lib/global.h"
- #include "lib/strutil.h"
- #include "lib/search.h"
- #include "internal.h"
- /*** global variables ****************************************************************************/
- /*** file scope macro definitions ****************************************************************/
- /*** file scope type declarations ****************************************************************/
- /*** forward declarations (file scope functions) *************************************************/
- /*** file scope variables ************************************************************************/
- /* --------------------------------------------------------------------------------------------- */
- /*** file scope functions ************************************************************************/
- /* --------------------------------------------------------------------------------------------- */
- static GString *
- mc_search__glob_translate_to_regex (const GString *astr)
- {
- GString *buff;
- gsize loop;
- gboolean inside_group = FALSE;
- buff = g_string_sized_new (32);
- for (loop = 0; loop < astr->len; loop++)
- {
- const char *str = astr->str;
- gboolean not_escaped;
- not_escaped = !str_is_char_escaped (str, str + loop);
- switch (str[loop])
- {
- case '*':
- if (not_escaped)
- {
- g_string_append (buff, inside_group ? ".*" : "(.*)");
- continue;
- }
- break;
- case '?':
- if (not_escaped)
- {
- g_string_append (buff, inside_group ? "." : "(.)");
- continue;
- }
- break;
- case ',':
- if (not_escaped)
- {
- g_string_append_c (buff, inside_group ? '|' : ',');
- continue;
- }
- break;
- case '{':
- if (not_escaped)
- {
- g_string_append_c (buff, '(');
- inside_group = TRUE;
- continue;
- }
- break;
- case '}':
- if (not_escaped)
- {
- g_string_append_c (buff, ')');
- inside_group = FALSE;
- continue;
- }
- break;
- case '+':
- case '.':
- case '$':
- case '(':
- case ')':
- case '^':
- g_string_append_c (buff, '\\');
- break;
- default:
- break;
- }
- g_string_append_c (buff, str[loop]);
- }
- return buff;
- }
- /* --------------------------------------------------------------------------------------------- */
- static GString *
- mc_search__translate_replace_glob_to_regex (const char *str)
- {
- GString *buff;
- char cnt = '0';
- gboolean escaped_mode = FALSE;
- buff = g_string_sized_new (32);
- while (*str != '\0')
- {
- char c = *str++;
- switch (c)
- {
- case '\\':
- if (!escaped_mode)
- {
- escaped_mode = TRUE;
- g_string_append_c (buff, '\\');
- continue;
- }
- break;
- case '*':
- case '?':
- if (!escaped_mode)
- {
- g_string_append_c (buff, '\\');
- c = ++cnt;
- }
- break;
- case '&':
- if (!escaped_mode)
- g_string_append_c (buff, '\\');
- break;
- default:
- break;
- }
- g_string_append_c (buff, c);
- escaped_mode = FALSE;
- }
- return buff;
- }
- /*** public functions ****************************************************************************/
- void
- mc_search__cond_struct_new_init_glob (const char *charset, mc_search_t *lc_mc_search,
- mc_search_cond_t *mc_search_cond)
- {
- GString *tmp;
- tmp = mc_search__glob_translate_to_regex (mc_search_cond->str);
- g_string_free (mc_search_cond->str, TRUE);
- if (lc_mc_search->is_entire_line)
- {
- g_string_prepend_c (tmp, '^');
- g_string_append_c (tmp, '$');
- }
- mc_search_cond->str = tmp;
- mc_search__cond_struct_new_init_regex (charset, lc_mc_search, mc_search_cond);
- }
- /* --------------------------------------------------------------------------------------------- */
- gboolean
- mc_search__run_glob (mc_search_t *lc_mc_search, const void *user_data,
- off_t start_search, off_t end_search, gsize *found_len)
- {
- return mc_search__run_regex (lc_mc_search, user_data, start_search, end_search, found_len);
- }
- /* --------------------------------------------------------------------------------------------- */
- GString *
- mc_search_glob_prepare_replace_str (mc_search_t *lc_mc_search, GString *replace_str)
- {
- GString *repl, *res;
- repl = mc_search__translate_replace_glob_to_regex (replace_str->str);
- res = mc_search_regex_prepare_replace_str (lc_mc_search, repl);
- g_string_free (repl, TRUE);
- return res;
- }
- /* --------------------------------------------------------------------------------------------- */
|