mc/src/find.c

1188 lines
30 KiB
C
Raw Normal View History

1998-02-27 07:54:42 +03:00
/* Find file command for the Midnight Commander
Copyright (C) 1995, 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005,
2006, 2007 Free Software Foundation, Inc.
1998-02-27 07:54:42 +03:00
Written 1995 by Miguel de Icaza
Complete rewrote.
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 2 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, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */
1998-02-27 07:54:42 +03:00
/** \file find.c
* \brief Source: Find file command
*/
1998-02-27 07:54:42 +03:00
#include <config.h>
2005-02-08 12:04:03 +03:00
#include <ctype.h>
1998-02-27 07:54:42 +03:00
#include <stdio.h>
2005-02-08 12:04:03 +03:00
#include <stdlib.h>
#include <string.h>
2009-02-06 02:30:45 +03:00
#include <fcntl.h>
1998-02-27 07:54:42 +03:00
#include <sys/stat.h>
1998-02-27 07:54:42 +03:00
#include "global.h"
#include "tty.h"
1998-02-27 07:54:42 +03:00
#include "win.h"
#include "color.h"
#include "setup.h"
1998-12-03 00:27:27 +03:00
#include "find.h"
#include "strutil.h"
#include "dialog.h"
1998-02-27 07:54:42 +03:00
#include "widget.h"
#include "dir.h"
#include "panel.h" /* current_panel */
#include "main.h" /* do_cd, try_to_select */
#include "wtools.h"
#include "cmd.h" /* view_file_at_line */
#include "boxes.h"
#include "key.h"
#include "../src/search/search.h"
1998-02-27 07:54:42 +03:00
/* Size of the find parameters window */
#define FIND_Y 16
static int FIND_X = 60;
1998-02-27 07:54:42 +03:00
/* Size of the find window */
2003-09-11 01:33:12 +04:00
#define FIND2_Y (LINES - 4)
1998-02-27 07:54:42 +03:00
static int FIND2_X = 64;
2003-09-11 01:33:12 +04:00
#define FIND2_X_USE (FIND2_X - 20)
1998-02-27 07:54:42 +03:00
/* A couple of extra messages we need */
enum {
B_STOP = B_USER + 1,
B_AGAIN,
B_PANELIZE,
B_TREE,
B_VIEW
};
typedef enum {
FIND_CONT,
FIND_SUSPEND,
FIND_ABORT
} FindProgressStatus;
/* List of directories to be ignored, separated by ':' */
1998-02-27 07:54:42 +03:00
char *find_ignore_dirs = 0;
static WInput *in_start; /* Start path */
static WInput *in_name; /* Pattern to search */
static WInput *in_with; /* Text inside filename */
static WCheck *case_sense; /* "case sensitive" checkbox */
static WCheck *find_regex_cbox; /* [x] find regular expression */
1998-02-27 07:54:42 +03:00
static int running = 0; /* nice flag */
static char *find_pattern; /* Pattern to search */
static char *content_pattern; /* pattern to search inside files; if
content_regexp_flag is true, it contains the
regex pattern, else the search string. */
1998-02-27 07:54:42 +03:00
static int count; /* Number of files displayed */
static int matches; /* Number of matches */
static int is_start; /* Status of the start/stop toggle button */
static char *old_dir;
/* Where did we stop */
static int resuming;
static int last_line;
static int last_pos;
static Dlg_head *find_dlg; /* The dialog */
static WButton *stop_button; /* pointer to the stop button */
static WLabel *status_label; /* Finished, Searching etc. */
static WListbox *find_list; /* Listbox with the file list */
1998-02-27 07:54:42 +03:00
/* This keeps track of the directory stack */
typedef struct dir_stack {
char *name;
struct dir_stack *prev;
} dir_stack ;
static dir_stack *dir_stack_base = 0;
1998-02-27 07:54:42 +03:00
static struct {
2004-08-17 03:48:46 +04:00
const char* text;
int len; /* length including space and brackets */
int x;
} fbuts [] = {
{ N_("&Suspend"), 11, 29 },
{ N_("Con&tinue"), 12, 29 },
{ N_("&Chdir"), 11, 3 },
{ N_("&Again"), 9, 17 },
{ N_("&Quit"), 8, 43 },
{ N_("Pane&lize"), 12, 3 },
{ N_("&View - F3"), 13, 20 },
{ N_("&Edit - F4"), 13, 38 }
};
static inline char * add_to_list (const char *text, void *data) {
return listbox_add_item (find_list, 0, 0, text, data);
}
static inline void stop_idle (void *data) {
set_idle_proc (data, 0);
}
static inline void status_update (const char *text) {
label_set_text (status_label, text);
}
static void get_list_info (char **file, char **dir) {
listbox_get_current (find_list, file, dir);
}
static mc_search_t *search_file_handle = NULL;
static gboolean skip_hidden_flag = FALSE;
static gboolean file_regexp_flag = FALSE;
static gboolean file_all_charsets_flag = FALSE;
static gboolean file_case_sentitive = TRUE;
static mc_search_t *search_content_handle = NULL;
static gboolean content_regexp_flag = FALSE;
static gboolean content_all_charsets_flag = FALSE;
static gboolean content_case_sensitive = TRUE;
static int find_recursively = 1;
/*
* Callback for the parameter dialog.
* Validate regex, prevent closing the dialog if it's invalid.
*/
static cb_ret_t
find_parm_callback (struct Dlg_head *h, dlg_msg_t msg, int parm)
{
switch (msg) {
case DLG_VALIDATE:
if ((h->ret_value != B_ENTER) || !in_with->buffer[0]
|| !(find_regex_cbox->state & C_BOOL))
return MSG_HANDLED;
/*
if (regcomp (r, in_with->buffer, flags)) {
message (D_ERROR, MSG_ERROR, _(" Malformed regular expression "));
dlg_select_widget (in_with);
h->running = 1; // Don't stop the dialog
}
*/
return MSG_HANDLED;
default:
return default_dlg_callback (h, msg, parm);
}
}
1998-02-27 07:54:42 +03:00
/*
* find_parameters: gets information from the user
*
* If the return value is true, then the following holds:
*
* START_DIR and PATTERN are pointers to char * and upon return they
* contain the information provided by the user.
*
* CONTENT holds a strdup of the contents specified by the user if he
* asked for them or 0 if not (note, this is different from the
* behavior for the other two parameters.
*
*/
static int
find_parameters (char **start_dir, char **pattern, char **content)
{
int return_value;
char *temp_dir;
static const char *case_label = N_("case &Sensitive");
static const char *recurs_label = N_("&Find recursively");
static const char *skip_hidden_label = N_("S&kip hidden");
static const char *all_charsets_label = N_("All charsets");
static const char *regexp_label = N_("&Regular expression");
static const char *file_regexp_label = N_("Regular expression");
WCheck *recursively_cbox;
WCheck *file_regexp_cbox;
WCheck *skip_hidden_cbox;
#ifdef HAVE_CHARSET
WCheck *file_all_charsets_cbox;
WCheck *content_all_charsets_cbox;
#endif
1998-02-27 07:54:42 +03:00
static char *in_contents = NULL;
static char *in_start_dir = NULL;
static char *in_start_name = NULL;
static const char *labs[] =
{ N_("Start at:"), N_("Filename:"), N_("Content: ") };
static const char *buts[] = { N_("&OK"), N_("&Tree"), N_("&Cancel") };
static int ilen = 30, istart = 14;
static int b0 = 3, b1 = 16, b2 = 36;
#ifdef ENABLE_NLS
static int i18n_flag = 0;
if (!i18n_flag) {
register int i = sizeof (labs) / sizeof (labs[0]);
int l1, maxlen = 0;
while (i--) {
l1 = str_term_width1 (labs[i] = _(labs[i]));
if (l1 > maxlen)
maxlen = l1;
}
i = maxlen + ilen + 7;
if (i > FIND_X)
FIND_X = i;
for (i = sizeof (buts) / sizeof (buts[0]), l1 = 0; i--;) {
l1 += str_term_width1 (buts[i] = _(buts[i]));
}
l1 += 21;
if (l1 > FIND_X)
FIND_X = l1;
ilen = FIND_X - 7 - maxlen; /* for the case of very long buttons :) */
istart = FIND_X - 3 - ilen;
b1 = b0 + str_term_width1 (buts[0]) + 7;
b2 = FIND_X - (str_term_width1 (buts[2]) + 6);
i18n_flag = 1;
case_label = _(case_label);
recurs_label = _(recurs_label);
}
#endif /* ENABLE_NLS */
find_par_start:
1998-02-27 07:54:42 +03:00
if (!in_start_dir)
in_start_dir = g_strdup (".");
1998-02-27 07:54:42 +03:00
if (!in_start_name)
in_start_name = g_strdup (easy_patterns ? "*" : ".");
1998-02-27 07:54:42 +03:00
if (!in_contents)
in_contents = g_strdup ("");
find_dlg =
create_dlg (0, 0, FIND_Y, FIND_X, dialog_colors,
find_parm_callback, "[Find File]", _("Find File"),
DLG_CENTER | DLG_REVERSE);
add_widget (find_dlg,
button_new (FIND_Y - 3, b2, B_CANCEL, NORMAL_BUTTON, buts[2], 0));
add_widget (find_dlg,
button_new (FIND_Y - 3, b1, B_TREE, NORMAL_BUTTON, buts[1], 0));
add_widget (find_dlg,
button_new (FIND_Y - 3, b0, B_ENTER, DEFPUSH_BUTTON, buts[0], 0));
file_regexp_cbox = check_new (6, 3, file_regexp_flag, file_regexp_label);
recursively_cbox = check_new (7, 3, find_recursively, recurs_label);
skip_hidden_cbox = check_new (6, 33, skip_hidden_flag, skip_hidden_label);
#ifdef HAVE_CHARSET
file_all_charsets_cbox = check_new (7, 33, file_all_charsets_flag, all_charsets_label);
#endif
find_regex_cbox = check_new (11, 3, content_regexp_flag, regexp_label);
case_sense = check_new (10, 3, content_case_sensitive, case_label);
#ifdef HAVE_CHARSET
content_all_charsets_cbox = check_new (10, 33, content_all_charsets_flag, all_charsets_label);
#endif
in_with = input_new (9, istart, INPUT_COLOR, ilen, in_contents, "content", INPUT_COMPLETE_DEFAULT);
in_name = input_new (5, istart, INPUT_COLOR, ilen, in_start_name, "name", INPUT_COMPLETE_DEFAULT);
in_start = input_new (3, istart, INPUT_COLOR, ilen, in_start_dir, "start", INPUT_COMPLETE_DEFAULT);
#ifdef HAVE_CHARSET
add_widget (find_dlg, content_all_charsets_cbox);
#endif
add_widget (find_dlg, find_regex_cbox);
add_widget (find_dlg, case_sense);
add_widget (find_dlg, in_with);
#ifdef HAVE_CHARSET
add_widget (find_dlg, file_all_charsets_cbox);
#endif
add_widget (find_dlg, skip_hidden_cbox);
add_widget (find_dlg, recursively_cbox);
add_widget (find_dlg, file_regexp_cbox);
add_widget (find_dlg, in_name);
add_widget (find_dlg, in_start);
1998-02-27 07:54:42 +03:00
add_widget (find_dlg, label_new (9, 3, labs[2]));
add_widget (find_dlg, label_new (5, 3, labs[1]));
add_widget (find_dlg, label_new (3, 3, labs[0]));
1998-02-27 07:54:42 +03:00
dlg_select_widget (in_name);
1998-02-27 07:54:42 +03:00
run_dlg (find_dlg);
switch (find_dlg->ret_value) {
case B_CANCEL:
1998-02-27 07:54:42 +03:00
return_value = 0;
break;
case B_TREE:
temp_dir = g_strdup (in_start->buffer);
#ifdef HAVE_CHARSET
file_all_charsets_flag = file_all_charsets_cbox->state & C_BOOL;
content_all_charsets_flag = content_all_charsets_cbox->state & C_BOOL;
#endif
content_case_sensitive = case_sense->state & C_BOOL;
content_regexp_flag = find_regex_cbox->state & C_BOOL;
file_regexp_flag = file_regexp_cbox->state & C_BOOL;
find_recursively = recursively_cbox->state & C_BOOL;
skip_hidden_flag = skip_hidden_cbox->state & C_BOOL;
1998-02-27 07:54:42 +03:00
destroy_dlg (find_dlg);
g_free (in_start_dir);
if (strcmp (temp_dir, ".") == 0) {
g_free (temp_dir);
temp_dir = g_strdup (current_panel->cwd);
1998-02-27 07:54:42 +03:00
}
in_start_dir = tree_box (temp_dir);
1998-02-27 07:54:42 +03:00
if (in_start_dir)
g_free (temp_dir);
1998-02-27 07:54:42 +03:00
else
in_start_dir = temp_dir;
/* Warning: Dreadful goto */
goto find_par_start;
break;
default:
g_free (in_contents);
if (in_with->buffer[0]) {
*content = g_strdup (in_with->buffer);
in_contents = g_strdup (*content);
} else {
1998-02-27 07:54:42 +03:00
*content = in_contents = NULL;
}
#ifdef HAVE_CHARSET
file_all_charsets_flag = file_all_charsets_cbox->state & C_BOOL;
content_all_charsets_flag = content_all_charsets_cbox->state & C_BOOL;
#endif
content_case_sensitive = case_sense->state & C_BOOL;
content_regexp_flag = find_regex_cbox->state & C_BOOL;
find_recursively = recursively_cbox->state & C_BOOL;
file_regexp_flag = file_regexp_cbox->state & C_BOOL;
skip_hidden_flag = skip_hidden_cbox->state & C_BOOL;
return_value = 1;
*start_dir = g_strdup (in_start->buffer);
*pattern = g_strdup (in_name->buffer);
1998-02-27 07:54:42 +03:00
g_free (in_start_dir);
in_start_dir = g_strdup (*start_dir);
g_free (in_start_name);
in_start_name = g_strdup (*pattern);
1998-02-27 07:54:42 +03:00
}
destroy_dlg (find_dlg);
1998-02-27 07:54:42 +03:00
return return_value;
}
static void
push_directory (const char *dir)
1998-02-27 07:54:42 +03:00
{
dir_stack *new;
new = g_new (dir_stack, 1);
new->name = concat_dir_and_file (dir, "");
1998-02-27 07:54:42 +03:00
new->prev = dir_stack_base;
dir_stack_base = new;
}
static char*
pop_directory (void)
{
char *name;
dir_stack *next;
if (dir_stack_base){
name = dir_stack_base->name;
next = dir_stack_base->prev;
g_free (dir_stack_base);
1998-02-27 07:54:42 +03:00
dir_stack_base = next;
return name;
} else
return 0;
}
static void
insert_file (const char *dir, const char *file)
1998-02-27 07:54:42 +03:00
{
char *tmp_name;
static char *dirname;
while (dir [0] == PATH_SEP && dir [1] == PATH_SEP)
1998-02-27 07:54:42 +03:00
dir++;
if (old_dir){
if (strcmp (old_dir, dir)){
g_free (old_dir);
old_dir = g_strdup (dir);
dirname = add_to_list (dir, NULL);
1998-02-27 07:54:42 +03:00
}
} else {
old_dir = g_strdup (dir);
dirname = add_to_list (dir, NULL);
1998-02-27 07:54:42 +03:00
}
tmp_name = g_strconcat (" ", file, (char *) NULL);
add_to_list (tmp_name, dirname);
g_free (tmp_name);
1998-02-27 07:54:42 +03:00
}
static void
find_add_match (Dlg_head *h, const char *dir, const char *file)
1998-02-27 07:54:42 +03:00
{
int p = ++matches & 7;
(void) h;
1998-02-27 07:54:42 +03:00
insert_file (dir, file);
1998-02-27 07:54:42 +03:00
/* Scroll nicely */
if (!p)
listbox_select_last (find_list, 1);
else
listbox_select_last (find_list, 0);
/* Updates the current listing */
send_message (&find_list->widget, WIDGET_DRAW, 0);
1998-02-27 07:54:42 +03:00
if (p == 7)
mc_refresh ();
}
/*
* get_line_at:
*
* Returns malloced null-terminated line from file file_fd.
* Input is buffered in buf_size long buffer.
* Current pos in buf is stored in pos.
* n_read - number of read chars.
* has_newline - is there newline ?
*/
static char *
get_line_at (int file_fd, char *buf, int *pos, int *n_read, int buf_size,
int *has_newline)
{
char *buffer = 0;
int buffer_size = 0;
char ch = 0;
int i = 0;
for (;;) {
if (*pos >= *n_read) {
*pos = 0;
if ((*n_read = mc_read (file_fd, buf, buf_size)) <= 0)
break;
}
ch = buf[(*pos)++];
if (ch == 0) {
/* skip possible leading zero(s) */
if (i == 0)
continue;
else
break;
}
if (i >= buffer_size - 1) {
buffer = g_realloc (buffer, buffer_size += 80);
}
/* Strip newline */
if (ch == '\n')
break;
buffer[i++] = ch;
}
*has_newline = ch ? 1 : 0;
if (buffer) {
buffer[i] = 0;
}
return buffer;
}
static FindProgressStatus
check_find_events(Dlg_head *h)
{
Gpm_Event event;
int c;
c = get_event (&event, h->mouse_status == MOU_REPEAT, 0);
if (c != EV_NONE) {
dlg_process_event (h, c, &event);
if (h->ret_value == B_ENTER
|| h->ret_value == B_CANCEL
|| h->ret_value == B_AGAIN
|| h->ret_value == B_PANELIZE) {
/* dialog terminated */
return FIND_ABORT;
}
if (!(h->flags & DLG_WANT_IDLE)) {
/* searching suspended */
return FIND_SUSPEND;
}
}
return FIND_CONT;
}
1998-02-27 07:54:42 +03:00
/*
* search_content:
*
* Search the global (FIXME) regexp compiled content_pattern string in the
1998-02-27 07:54:42 +03:00
* DIRECTORY/FILE. It will add the found entries to the find listbox.
*
* returns 0 if do_search should look for another file
* 1 if do_search should exit and proceed to the event handler
1998-02-27 07:54:42 +03:00
*/
static int
search_content (Dlg_head *h, const char *directory, const char *filename)
1998-02-27 07:54:42 +03:00
{
struct stat s;
char buffer [BUF_4K];
char *fname;
int file_fd;
int ret_val = 0;
1998-02-27 07:54:42 +03:00
fname = concat_dir_and_file (directory, filename);
1998-02-27 07:54:42 +03:00
if (mc_stat (fname, &s) != 0 || !S_ISREG (s.st_mode)){
g_free (fname);
return 0;
1998-02-27 07:54:42 +03:00
}
1998-02-27 07:54:42 +03:00
file_fd = mc_open (fname, O_RDONLY);
g_free (fname);
1998-02-27 07:54:42 +03:00
if (file_fd == -1)
return 0;
1998-02-27 07:54:42 +03:00
g_snprintf (buffer, sizeof (buffer), _("Grepping in %s"), str_trunc (filename, FIND2_X_USE));
1998-02-27 07:54:42 +03:00
status_update (buffer);
1998-02-27 07:54:42 +03:00
mc_refresh ();
1998-02-27 07:54:42 +03:00
enable_interrupt_key ();
got_interrupt ();
{
int line = 1;
int pos = 0;
int n_read = 0;
int has_newline;
char *p;
gboolean found = FALSE;
gsize founded_len;
if (resuming) {
/* We've been previously suspended, start from the previous position */
resuming = 0;
line = last_line;
pos = last_pos;
}
while ((p = get_line_at (file_fd, buffer, &pos, &n_read, sizeof (buffer), &has_newline)) && (ret_val == 0)){
if (found == FALSE){ /* Search in binary line once */
if (mc_search_run (search_content_handle, (const void *) p, 0, strlen(p), &founded_len))
{
char *fnd_info = g_strdup_printf ("%d:%s", line, filename);
find_add_match (h, directory, fnd_info);
found = TRUE;
}
}
if (has_newline){
line++;
found = FALSE;
}
g_free (p);
if ((line & 0xff) == 0) {
FindProgressStatus res;
res = check_find_events(h);
switch (res) {
case FIND_ABORT:
stop_idle (h);
ret_val = 1;
break;
case FIND_SUSPEND:
resuming = 1;
last_line = line;
last_pos = pos;
ret_val = 1;
break;
default:
break;
}
}
}
1998-02-27 07:54:42 +03:00
}
disable_interrupt_key ();
mc_close (file_fd);
return ret_val;
1998-02-27 07:54:42 +03:00
}
static int
1998-02-27 07:54:42 +03:00
do_search (struct Dlg_head *h)
{
static struct dirent *dp = 0;
static DIR *dirp = 0;
static char *directory;
1998-02-27 07:54:42 +03:00
struct stat tmp_stat;
static int pos;
static int subdirs_left = 0;
gsize bytes_found;
1998-02-27 07:54:42 +03:00
if (!h) { /* someone forces me to close dirp */
if (dirp) {
mc_closedir (dirp);
dirp = 0;
}
g_free (directory);
directory = NULL;
1998-02-27 07:54:42 +03:00
dp = 0;
return 1;
1998-02-27 07:54:42 +03:00
}
search_content_handle = mc_search_new(content_pattern, -1);
if (search_content_handle)
{
search_content_handle->search_type = (content_regexp_flag) ? MC_SEARCH_T_REGEX : MC_SEARCH_T_NORMAL;
search_content_handle->is_case_sentitive = content_case_sensitive;
search_content_handle->is_all_charsets = content_all_charsets_flag;
}
search_file_handle = mc_search_new(find_pattern, -1);
search_file_handle->search_type = (file_regexp_flag) ? MC_SEARCH_T_REGEX : MC_SEARCH_T_GLOB;
search_file_handle->is_case_sentitive = file_case_sentitive;
search_file_handle->is_all_charsets = file_all_charsets_flag;
search_file_handle->is_entire_line = !file_regexp_flag;
1998-02-27 07:54:42 +03:00
do_search_begin:
while (!dp){
if (dirp){
mc_closedir (dirp);
dirp = 0;
}
while (!dirp){
char *tmp;
attrset (REVERSE_COLOR);
while (1) {
tmp = pop_directory ();
if (!tmp){
running = 0;
status_update (_("Finished"));
stop_idle (h);
mc_search_free(search_file_handle);
search_file_handle = NULL;
mc_search_free(search_content_handle);
search_content_handle = NULL;
return 0;
1998-02-27 07:54:42 +03:00
}
if (find_ignore_dirs){
int found;
char *temp_dir = g_strconcat (":", tmp, ":", (char *) NULL);
found = strstr (find_ignore_dirs, temp_dir) != 0;
g_free (temp_dir);
if (found)
g_free (tmp);
1998-02-27 07:54:42 +03:00
else
break;
} else
break;
}
g_free (directory);
directory = tmp;
1998-02-27 07:54:42 +03:00
if (verbose){
char buffer [BUF_SMALL];
1998-02-27 07:54:42 +03:00
g_snprintf (buffer, sizeof (buffer), _("Searching %s"),
str_trunc (directory, FIND2_X_USE));
status_update (buffer);
1998-02-27 07:54:42 +03:00
}
/* mc_stat should not be called after mc_opendir
because vfs_s_opendir modifies the st_nlink
*/
if (!mc_stat (directory, &tmp_stat))
subdirs_left = tmp_stat.st_nlink - 2;
else
subdirs_left = 0;
1998-02-27 07:54:42 +03:00
/* Commented out as unnecessary
if (subdirs_left < 0)
subdirs_left = MAXINT;
*/
dirp = mc_opendir (directory);
} /* while (!dirp) */
1998-02-27 07:54:42 +03:00
dp = mc_readdir (dirp);
/* skip invalid filenames */
while (dp != NULL && !str_is_valid_string (dp->d_name))
dp = mc_readdir (dirp);
} /* while (!dp) */
1998-02-27 07:54:42 +03:00
if (strcmp (dp->d_name, ".") == 0 ||
strcmp (dp->d_name, "..") == 0){
dp = mc_readdir (dirp);
/* skip invalid filenames */
while (dp != NULL && !str_is_valid_string (dp->d_name))
dp = mc_readdir (dirp);
mc_search_free(search_file_handle);
search_file_handle = NULL;
mc_search_free(search_content_handle);
search_content_handle = NULL;
return 1;
1998-02-27 07:54:42 +03:00
}
if (!(skip_hidden_flag && dp->d_name[0] == '.')) {
if (subdirs_left && find_recursively && directory) { /* Can directory be NULL ? */
char *tmp_name = concat_dir_and_file (directory, dp->d_name);
if (!mc_lstat (tmp_name, &tmp_stat)
&& S_ISDIR (tmp_stat.st_mode)) {
push_directory (tmp_name);
subdirs_left--;
}
g_free (tmp_name);
}
if (mc_search_run (search_file_handle,dp->d_name, 0, strlen (dp->d_name), &bytes_found)) {
if (content_pattern) {
if (search_content (h, directory, dp->d_name)) {
mc_search_free(search_file_handle);
search_file_handle = NULL;
mc_search_free(search_content_handle);
search_content_handle = NULL;
return 1;
}
} else
find_add_match (h, directory, dp->d_name);
}
1998-02-27 07:54:42 +03:00
}
dp = mc_readdir (dirp);
/* skip invalid filenames */
while (dp != NULL && !str_is_valid_string (dp->d_name))
dp = mc_readdir (dirp);
1998-02-27 07:54:42 +03:00
/* Displays the nice dot */
count++;
if (!(count & 31)){
/* For nice updating */
const char *rotating_dash = "|/-\\";
1998-02-27 07:54:42 +03:00
if (verbose){
pos = (pos + 1) % 4;
attrset (DLG_NORMALC (h));
1998-02-27 07:54:42 +03:00
dlg_move (h, FIND2_Y-6, FIND2_X - 4);
addch (rotating_dash [pos]);
mc_refresh ();
}
} else
goto do_search_begin;
mc_search_free(search_file_handle);
search_file_handle = NULL;
mc_search_free(search_content_handle);
search_content_handle = NULL;
return 1;
1998-02-27 07:54:42 +03:00
}
static void
init_find_vars (void)
1998-02-27 07:54:42 +03:00
{
char *dir;
g_free (old_dir);
old_dir = 0;
count = 0;
matches = 0;
1998-02-27 07:54:42 +03:00
/* Remove all the items in the stack */
while ((dir = pop_directory ()) != NULL)
g_free (dir);
}
1998-02-27 07:54:42 +03:00
static char *
make_fullname (const char *dirname, const char *filename)
{
if (strcmp(dirname, ".") == 0 || strcmp(dirname, "."PATH_SEP_STR) == 0)
return g_strdup (filename);
if (strncmp(dirname, "."PATH_SEP_STR, 2) == 0)
return concat_dir_and_file (dirname + 2, filename);
return concat_dir_and_file (dirname, filename);
}
static void
find_do_view_edit (int unparsed_view, int edit, char *dir, char *file)
{
char *fullname;
const char *filename;
int line;
1998-02-27 07:54:42 +03:00
if (content_pattern){
filename = strchr (file + 4, ':') + 1;
line = atoi (file + 4);
1998-02-27 07:54:42 +03:00
} else {
filename = file + 4;
1998-02-27 07:54:42 +03:00
line = 0;
}
fullname = make_fullname (dir, filename);
1998-02-27 07:54:42 +03:00
if (edit)
do_edit_at_line (fullname, line);
else
view_file_at_line (fullname, unparsed_view, use_internal_view, line);
g_free (fullname);
}
static int
view_edit_currently_selected_file (int unparsed_view, int edit)
{
WLEntry *entry = find_list->current;
char *dir;
if (!entry)
return MSG_NOT_HANDLED;
dir = entry->data;
if (!entry->text || !dir)
return MSG_NOT_HANDLED;
find_do_view_edit (unparsed_view, edit, dir, entry->text);
1998-02-27 07:54:42 +03:00
return MSG_HANDLED;
}
static cb_ret_t
find_callback (struct Dlg_head *h, dlg_msg_t msg, int parm)
1998-02-27 07:54:42 +03:00
{
switch (msg) {
1998-02-27 07:54:42 +03:00
case DLG_KEY:
if (parm == KEY_F (3) || parm == KEY_F (13)) {
int unparsed_view = (parm == KEY_F (13));
1998-02-27 07:54:42 +03:00
return view_edit_currently_selected_file (unparsed_view, 0);
}
if (parm == KEY_F (4)) {
1998-02-27 07:54:42 +03:00
return view_edit_currently_selected_file (0, 1);
}
return MSG_NOT_HANDLED;
case DLG_IDLE:
do_search (h);
return MSG_HANDLED;
default:
return default_dlg_callback (h, msg, parm);
}
1998-02-27 07:54:42 +03:00
}
/* Handles the Stop/Start button in the find window */
static int
start_stop (int button)
1998-02-27 07:54:42 +03:00
{
(void) button;
1998-02-27 07:54:42 +03:00
running = is_start;
set_idle_proc (find_dlg, running);
is_start = !is_start;
status_update (is_start ? _("Stopped") : _("Searching"));
button_set_text (stop_button, fbuts [is_start].text);
1998-02-27 07:54:42 +03:00
return 0;
}
/* Handle view command, when invoked as a button */
static int
find_do_view_file (int button)
1998-02-27 07:54:42 +03:00
{
(void) button;
1998-02-27 07:54:42 +03:00
view_edit_currently_selected_file (0, 0);
return 0;
}
/* Handle edit command, when invoked as a button */
static int
find_do_edit_file (int button)
1998-02-27 07:54:42 +03:00
{
(void) button;
1998-02-27 07:54:42 +03:00
view_edit_currently_selected_file (0, 1);
return 0;
}
static void
setup_gui (void)
1998-02-27 07:54:42 +03:00
{
#ifdef ENABLE_NLS
static int i18n_flag = 0;
if (!i18n_flag) {
register int i = sizeof (fbuts) / sizeof (fbuts[0]);
while (i--)
fbuts[i].len = str_term_width1 (fbuts[i].text = _(fbuts[i].text)) + 3;
fbuts[2].len += 2; /* DEFPUSH_BUTTON */
i18n_flag = 1;
}
#endif /* ENABLE_NLS */
/*
* Dynamically place buttons centered within current window size
*/
{
int l0 = max (fbuts[0].len, fbuts[1].len);
int l1 = fbuts[2].len + fbuts[3].len + l0 + fbuts[4].len;
int l2 = fbuts[5].len + fbuts[6].len + fbuts[7].len;
int r1, r2;
FIND2_X = COLS - 16;
/* Check, if both button rows fit within FIND2_X */
if (l1 + 9 > FIND2_X)
FIND2_X = l1 + 9;
if (l2 + 8 > FIND2_X)
FIND2_X = l2 + 8;
/* compute amount of space between buttons for each row */
r1 = (FIND2_X - 4 - l1) % 5;
l1 = (FIND2_X - 4 - l1) / 5;
r2 = (FIND2_X - 4 - l2) % 4;
l2 = (FIND2_X - 4 - l2) / 4;
/* ...and finally, place buttons */
fbuts[2].x = 2 + r1 / 2 + l1;
fbuts[3].x = fbuts[2].x + fbuts[2].len + l1;
fbuts[0].x = fbuts[3].x + fbuts[3].len + l1;
fbuts[4].x = fbuts[0].x + l0 + l1;
fbuts[5].x = 2 + r2 / 2 + l2;
fbuts[6].x = fbuts[5].x + fbuts[5].len + l2;
fbuts[7].x = fbuts[6].x + fbuts[6].len + l2;
}
find_dlg =
create_dlg (0, 0, FIND2_Y, FIND2_X, dialog_colors, find_callback,
"[Find File]", _("Find File"), DLG_CENTER | DLG_REVERSE);
add_widget (find_dlg,
button_new (FIND2_Y - 3, fbuts[7].x, B_VIEW, NORMAL_BUTTON,
fbuts[7].text, find_do_edit_file));
add_widget (find_dlg,
button_new (FIND2_Y - 3, fbuts[6].x, B_VIEW, NORMAL_BUTTON,
fbuts[6].text, find_do_view_file));
add_widget (find_dlg,
button_new (FIND2_Y - 3, fbuts[5].x, B_PANELIZE,
NORMAL_BUTTON, fbuts[5].text, 0));
add_widget (find_dlg,
button_new (FIND2_Y - 4, fbuts[4].x, B_CANCEL,
NORMAL_BUTTON, fbuts[4].text, 0));
stop_button =
button_new (FIND2_Y - 4, fbuts[0].x, B_STOP, NORMAL_BUTTON,
fbuts[0].text, start_stop);
add_widget (find_dlg, stop_button);
add_widget (find_dlg,
button_new (FIND2_Y - 4, fbuts[3].x, B_AGAIN,
NORMAL_BUTTON, fbuts[3].text, 0));
add_widget (find_dlg,
button_new (FIND2_Y - 4, fbuts[2].x, B_ENTER,
DEFPUSH_BUTTON, fbuts[2].text, 0));
status_label = label_new (FIND2_Y - 6, 4, _("Searching"));
add_widget (find_dlg, status_label);
1998-02-27 07:54:42 +03:00
find_list =
listbox_new (2, 2, FIND2_Y - 9, FIND2_X - 4, NULL);
add_widget (find_dlg, find_list);
}
1998-02-27 07:54:42 +03:00
static int
run_process (void)
{
resuming = 0;
set_idle_proc (find_dlg, 1);
run_dlg (find_dlg);
return find_dlg->ret_value;
}
static void
kill_gui (void)
{
set_idle_proc (find_dlg, 0);
destroy_dlg (find_dlg);
}
static int
find_file (char *start_dir, char *pattern, char *content, char **dirname,
char **filename)
{
int return_value = 0;
char *dir;
char *dir_tmp, *file_tmp;
setup_gui ();
1998-02-27 07:54:42 +03:00
/* FIXME: Need to cleanup this, this ought to be passed non-globaly */
find_pattern = pattern;
content_pattern = (content != NULL && str_is_valid_string (content))
? str_create_search_needle (content, content_case_sensitive)
: NULL;
1998-02-27 07:54:42 +03:00
init_find_vars ();
push_directory (start_dir);
return_value = run_process ();
1998-02-27 07:54:42 +03:00
/* Remove all the items in the stack */
while ((dir = pop_directory ()) != NULL)
g_free (dir);
get_list_info (&file_tmp, &dir_tmp);
1998-02-27 07:54:42 +03:00
if (dir_tmp)
*dirname = g_strdup (dir_tmp);
1998-02-27 07:54:42 +03:00
if (file_tmp)
*filename = g_strdup (file_tmp);
if (return_value == B_PANELIZE && *filename) {
int status, link_to_dir, stale_link;
1998-02-27 07:54:42 +03:00
int next_free = 0;
int i;
struct stat st;
1998-02-27 07:54:42 +03:00
WLEntry *entry = find_list->list;
dir_list *list = &current_panel->dir;
char *name;
1998-02-27 07:54:42 +03:00
for (i = 0; entry && i < find_list->count;
entry = entry->next, i++) {
const char *filename;
1998-02-27 07:54:42 +03:00
if (!entry->text || !entry->data)
continue;
1998-02-27 07:54:42 +03:00
if (content_pattern)
filename = strchr (entry->text + 4, ':') + 1;
1998-02-27 07:54:42 +03:00
else
filename = entry->text + 4;
1998-02-27 07:54:42 +03:00
name = make_fullname (entry->data, filename);
status =
handle_path (list, name, &st, next_free, &link_to_dir,
&stale_link);
1998-02-27 07:54:42 +03:00
if (status == 0) {
g_free (name);
1998-02-27 07:54:42 +03:00
continue;
}
if (status == -1) {
g_free (name);
1998-02-27 07:54:42 +03:00
break;
}
/* don't add files more than once to the panel */
if (content_pattern && next_free > 0) {
if (strcmp (list->list[next_free - 1].fname, name) == 0) {
g_free (name);
1998-02-27 07:54:42 +03:00
continue;
}
}
if (!next_free) /* first turn i.e clean old list */
panel_clean_dir (current_panel);
list->list[next_free].fnamelen = strlen (name);
list->list[next_free].fname = name;
list->list[next_free].f.marked = 0;
list->list[next_free].f.link_to_dir = link_to_dir;
list->list[next_free].f.stale_link = stale_link;
list->list[next_free].f.dir_size_computed = 0;
list->list[next_free].st = st;
list->list[next_free].sort_key = NULL;
list->list[next_free].second_sort_key = NULL;
1998-02-27 07:54:42 +03:00
next_free++;
if (!(next_free & 15))
rotate_dash ();
1998-02-27 07:54:42 +03:00
}
if (next_free) {
current_panel->count = next_free;
current_panel->is_panelized = 1;
/* Done by panel_clean_dir a few lines above
current_panel->dirs_marked = 0;
current_panel->marked = 0;
current_panel->total = 0;
current_panel->top_file = 0;
current_panel->selected = 0; */
if (start_dir[0] == PATH_SEP) {
strcpy (current_panel->cwd, PATH_SEP_STR);
1998-02-27 07:54:42 +03:00
chdir (PATH_SEP_STR);
}
}
}
if (content_pattern != NULL) str_release_search_needle (content_pattern, content_case_sensitive);
kill_gui ();
do_search (0); /* force do_search to release resources */
g_free (old_dir);
old_dir = 0;
1998-02-27 07:54:42 +03:00
return return_value;
}
void
do_find (void)
{
char *start_dir = NULL, *pattern = NULL, *content = NULL;
1998-02-27 07:54:42 +03:00
char *filename, *dirname;
int v, dir_and_file_set;
while (find_parameters (&start_dir, &pattern, &content)){
1998-02-27 07:54:42 +03:00
dirname = filename = NULL;
is_start = 0;
v = find_file (start_dir, pattern, content, &dirname, &filename);
g_free (start_dir);
g_free (pattern);
1998-02-27 07:54:42 +03:00
if (v == B_ENTER){
if (dirname || filename){
if (dirname){
do_cd (dirname, cd_exact);
if (filename)
try_to_select (current_panel, filename + (content ?
1998-02-27 07:54:42 +03:00
(strchr (filename + 4, ':') - filename + 1) : 4) );
} else if (filename)
do_cd (filename, cd_exact);
select_item (current_panel);
1998-02-27 07:54:42 +03:00
}
g_free (dirname);
g_free (filename);
1998-02-27 07:54:42 +03:00
break;
}
g_free (content);
1998-02-27 07:54:42 +03:00
dir_and_file_set = dirname && filename;
g_free (dirname);
g_free (filename);
1998-02-27 07:54:42 +03:00
if (v == B_CANCEL)
break;
if (v == B_PANELIZE){
if (dir_and_file_set){
try_to_select (current_panel, NULL);
panel_re_sort (current_panel);
try_to_select (current_panel, NULL);
1998-02-27 07:54:42 +03:00
}
break;
}
}
}