2009-04-30 17:57:46 +04:00
|
|
|
/*
|
|
|
|
Search text engine.
|
|
|
|
Glob-style pattern matching
|
|
|
|
|
2011-10-15 14:56:47 +04:00
|
|
|
Copyright (C) 2009, 2011
|
|
|
|
The Free Software Foundation, Inc.
|
2009-04-30 17:57:46 +04:00
|
|
|
|
|
|
|
Written by:
|
|
|
|
Slava Zanko <slavazanko@gmail.com>, 2009.
|
|
|
|
|
|
|
|
This file is part of the Midnight Commander.
|
|
|
|
|
2011-10-15 14:56:47 +04:00
|
|
|
The Midnight Commander is free software: you can redistribute it
|
2009-04-30 17:57:46 +04:00
|
|
|
and/or modify it under the terms of the GNU General Public License as
|
2011-10-15 14:56:47 +04:00
|
|
|
published by the Free Software Foundation, either version 3 of the License,
|
|
|
|
or (at your option) any later version.
|
2009-04-30 17:57:46 +04:00
|
|
|
|
2011-10-15 14:56:47 +04:00
|
|
|
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.
|
2009-04-30 17:57:46 +04:00
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
2011-10-15 14:56:47 +04:00
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
2009-04-30 17:57:46 +04:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
2010-01-20 18:11:52 +03:00
|
|
|
#include "lib/global.h"
|
2010-01-21 15:17:26 +03:00
|
|
|
#include "lib/strutil.h"
|
2010-01-21 16:06:15 +03:00
|
|
|
#include "lib/search.h"
|
2010-01-21 16:31:29 +03:00
|
|
|
#include "lib/strescape.h"
|
2010-01-21 15:17:26 +03:00
|
|
|
|
|
|
|
#include "internal.h"
|
|
|
|
|
2009-04-30 17:57:46 +04:00
|
|
|
/*** global variables ****************************************************************************/
|
|
|
|
|
|
|
|
/*** file scope macro definitions ****************************************************************/
|
|
|
|
|
|
|
|
/*** file scope type declarations ****************************************************************/
|
|
|
|
|
|
|
|
/*** file scope variables ************************************************************************/
|
|
|
|
|
|
|
|
/*** file scope functions ************************************************************************/
|
|
|
|
|
|
|
|
static GString *
|
2010-10-21 17:07:06 +04:00
|
|
|
mc_search__glob_translate_to_regex (const GString * astr)
|
2009-04-30 17:57:46 +04:00
|
|
|
{
|
2010-10-21 17:07:06 +04:00
|
|
|
const char *str = astr->str;
|
2010-10-21 17:18:18 +04:00
|
|
|
GString *buff;
|
2010-10-21 17:33:32 +04:00
|
|
|
gsize loop;
|
2009-06-08 14:54:35 +04:00
|
|
|
gboolean inside_group = FALSE;
|
2010-10-21 17:07:06 +04:00
|
|
|
|
2010-10-21 17:18:18 +04:00
|
|
|
buff = g_string_sized_new (32);
|
|
|
|
|
2010-10-21 17:33:32 +04:00
|
|
|
for (loop = 0; loop < astr->len; loop++)
|
2010-07-07 15:04:38 +04:00
|
|
|
switch (str[loop])
|
|
|
|
{
|
2009-04-30 17:57:46 +04:00
|
|
|
case '*':
|
2010-07-07 15:04:38 +04:00
|
|
|
if (!strutils_is_char_escaped (str, &(str[loop])))
|
2010-10-21 17:33:32 +04:00
|
|
|
g_string_append (buff, inside_group ? ".*" : "(.*)");
|
2009-04-30 17:57:46 +04:00
|
|
|
break;
|
|
|
|
case '?':
|
2010-07-07 15:04:38 +04:00
|
|
|
if (!strutils_is_char_escaped (str, &(str[loop])))
|
2010-10-21 17:33:32 +04:00
|
|
|
g_string_append (buff, inside_group ? "." : "(.)");
|
2009-04-30 17:57:46 +04:00
|
|
|
break;
|
2009-05-07 13:53:59 +04:00
|
|
|
case ',':
|
2010-07-07 15:04:38 +04:00
|
|
|
if (!strutils_is_char_escaped (str, &(str[loop])))
|
2010-10-21 17:33:32 +04:00
|
|
|
g_string_append_c (buff, '|');
|
2009-05-07 13:53:59 +04:00
|
|
|
break;
|
|
|
|
case '{':
|
2010-07-07 15:04:38 +04:00
|
|
|
if (!strutils_is_char_escaped (str, &(str[loop])))
|
|
|
|
{
|
2010-10-21 17:33:32 +04:00
|
|
|
g_string_append_c (buff, '(');
|
2009-06-08 14:54:35 +04:00
|
|
|
inside_group = TRUE;
|
2009-05-07 13:53:59 +04:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case '}':
|
2010-07-07 15:04:38 +04:00
|
|
|
if (!strutils_is_char_escaped (str, &(str[loop])))
|
|
|
|
{
|
2010-10-21 17:33:32 +04:00
|
|
|
g_string_append_c (buff, ')');
|
2009-06-08 14:54:35 +04:00
|
|
|
inside_group = FALSE;
|
2009-05-07 13:53:59 +04:00
|
|
|
}
|
|
|
|
break;
|
2009-04-30 17:57:46 +04:00
|
|
|
case '+':
|
|
|
|
case '.':
|
|
|
|
case '$':
|
2009-05-07 13:53:59 +04:00
|
|
|
case '(':
|
|
|
|
case ')':
|
2009-04-30 17:57:46 +04:00
|
|
|
case '^':
|
|
|
|
g_string_append_c (buff, '\\');
|
2010-10-21 17:33:32 +04:00
|
|
|
/* fall through */
|
|
|
|
default:
|
2009-04-30 17:57:46 +04:00
|
|
|
g_string_append_c (buff, str[loop]);
|
2010-10-21 17:33:32 +04:00
|
|
|
break;
|
2009-04-30 17:57:46 +04:00
|
|
|
}
|
2010-10-21 17:07:06 +04:00
|
|
|
|
2009-04-30 17:57:46 +04:00
|
|
|
return buff;
|
|
|
|
}
|
|
|
|
|
2009-08-07 15:47:26 +04:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
static GString *
|
2013-02-14 14:42:27 +04:00
|
|
|
mc_search__translate_replace_glob_to_regex (const char *str)
|
2009-08-07 15:47:26 +04:00
|
|
|
{
|
2010-10-21 17:18:18 +04:00
|
|
|
GString *buff;
|
2009-08-07 15:47:26 +04:00
|
|
|
int cnt = '0';
|
2010-07-07 15:01:15 +04:00
|
|
|
gboolean escaped_mode = FALSE;
|
2010-10-21 17:18:18 +04:00
|
|
|
|
|
|
|
buff = g_string_sized_new (32);
|
|
|
|
|
2010-07-07 15:04:38 +04:00
|
|
|
while (*str)
|
|
|
|
{
|
|
|
|
char c = *str++;
|
|
|
|
switch (c)
|
|
|
|
{
|
|
|
|
case '\\':
|
|
|
|
if (!escaped_mode)
|
|
|
|
{
|
|
|
|
escaped_mode = TRUE;
|
2011-10-07 13:14:17 +04:00
|
|
|
g_string_append_c (buff, '\\');
|
|
|
|
continue;
|
2010-07-07 15:04:38 +04:00
|
|
|
}
|
2011-10-07 13:14:17 +04:00
|
|
|
break;
|
2010-07-07 15:04:38 +04:00
|
|
|
case '*':
|
|
|
|
case '?':
|
|
|
|
if (!escaped_mode)
|
|
|
|
{
|
|
|
|
g_string_append_c (buff, '\\');
|
|
|
|
c = ++cnt;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case '&':
|
2011-10-07 13:14:17 +04:00
|
|
|
if (!escaped_mode)
|
|
|
|
g_string_append_c (buff, '\\');
|
2010-07-07 15:04:38 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
g_string_append_c (buff, c);
|
|
|
|
escaped_mode = FALSE;
|
2009-08-07 15:47:26 +04:00
|
|
|
}
|
|
|
|
return buff;
|
|
|
|
}
|
|
|
|
|
2009-04-30 17:57:46 +04:00
|
|
|
/*** public functions ****************************************************************************/
|
|
|
|
|
|
|
|
void
|
2009-10-30 04:12:04 +03:00
|
|
|
mc_search__cond_struct_new_init_glob (const char *charset, mc_search_t * lc_mc_search,
|
2009-04-30 17:57:46 +04:00
|
|
|
mc_search_cond_t * mc_search_cond)
|
|
|
|
{
|
2010-10-21 17:07:06 +04:00
|
|
|
GString *tmp;
|
2009-04-30 17:57:46 +04:00
|
|
|
|
2010-10-21 17:07:06 +04:00
|
|
|
tmp = mc_search__glob_translate_to_regex (mc_search_cond->str);
|
2009-04-30 17:57:46 +04:00
|
|
|
g_string_free (mc_search_cond->str, TRUE);
|
2009-06-10 12:57:45 +04:00
|
|
|
|
2010-07-07 15:04:38 +04:00
|
|
|
if (lc_mc_search->is_entire_line)
|
|
|
|
{
|
2009-06-10 12:57:45 +04:00
|
|
|
g_string_prepend_c (tmp, '^');
|
|
|
|
g_string_append_c (tmp, '$');
|
|
|
|
}
|
2009-04-30 17:57:46 +04:00
|
|
|
mc_search_cond->str = tmp;
|
|
|
|
|
2009-10-30 04:12:04 +03:00
|
|
|
mc_search__cond_struct_new_init_regex (charset, lc_mc_search, mc_search_cond);
|
2009-04-30 17:57:46 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
gboolean
|
2009-10-30 04:12:04 +03:00
|
|
|
mc_search__run_glob (mc_search_t * lc_mc_search, const void *user_data,
|
2009-04-30 17:57:46 +04:00
|
|
|
gsize start_search, gsize end_search, gsize * found_len)
|
|
|
|
{
|
2009-10-30 04:12:04 +03:00
|
|
|
return mc_search__run_regex (lc_mc_search, user_data, start_search, end_search, found_len);
|
2009-04-30 17:57:46 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
2009-08-07 15:47:26 +04:00
|
|
|
|
2009-04-30 17:57:46 +04:00
|
|
|
GString *
|
2009-10-30 04:12:04 +03:00
|
|
|
mc_search_glob_prepare_replace_str (mc_search_t * lc_mc_search, GString * replace_str)
|
2009-04-30 17:57:46 +04:00
|
|
|
{
|
2010-07-07 15:04:38 +04:00
|
|
|
GString *repl = mc_search__translate_replace_glob_to_regex (replace_str->str);
|
2009-10-30 04:12:04 +03:00
|
|
|
GString *res = mc_search_regex_prepare_replace_str (lc_mc_search, repl);
|
2009-08-07 15:47:26 +04:00
|
|
|
g_string_free (repl, TRUE);
|
|
|
|
return res;
|
2009-04-30 17:57:46 +04:00
|
|
|
}
|
2009-08-07 15:47:26 +04:00
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|