mc/src/view.c
2009-04-20 05:51:32 +00:00

4138 lines
111 KiB
C

/*
Internal file viewer for the Midnight Commander
Copyright (C) 1994, 1995, 1996 The Free Software Foundation
Written by: 1994, 1995, 1998 Miguel de Icaza
1994, 1995 Janne Kukonlehto
1995 Jakub Jelinek
1996 Joseph M. Hinkle
1997 Norbert Warmuth
1998 Pavel Machek
2004 Roland Illig <roland.illig@gmx.de>
2005 Roland Illig <roland.illig@gmx.de>
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.
*/
#ifdef HAVE_CONFIG_H
# include <config.h>
#endif
#include <assert.h>
#include <ctype.h>
#include <errno.h>
#include <limits.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include "global.h"
#include "tty.h"
#include "cmd.h" /* For view_other_cmd */
#include "dialog.h" /* Needed by widget.h */
#include "widget.h" /* Needed for buttonbar_new */
#include "color.h"
#include "mouse.h"
#include "help.h"
#include "key.h" /* For mi_getch() */
#include "layout.h"
#include "setup.h"
#include "wtools.h" /* For query_set_sel() */
#include "dir.h"
#include "panel.h" /* Needed for current_panel and other_panel */
#include "win.h"
#include "execute.h"
#include "main.h" /* slow_terminal */
#include "view.h"
#include "history.h"
#include "charsets.h"
#include "selcodepage.h"
#include "strutil.h"
/* Block size for reading files in parts */
#define VIEW_PAGE_SIZE ((size_t) 8192)
typedef unsigned char byte;
/* Offset in bytes into a file */
typedef unsigned long offset_type;
#define INVALID_OFFSET ((offset_type) -1)
#define OFFSETTYPE_MAX (~((offset_type) 0))
#define OFFSETTYPE_PRIX "lX"
#define OFFSETTYPE_PRId "lu"
/* A width or height on the screen */
typedef unsigned int screen_dimen;
/* A node for building a change list on change_list */
struct hexedit_change_node {
struct hexedit_change_node *next;
offset_type offset;
byte value;
};
/* data sources of the view */
enum view_ds {
DS_NONE, /* No data available */
DS_STDIO_PIPE, /* Data comes from a pipe using popen/pclose */
DS_VFS_PIPE, /* Data comes from a piped-in VFS file */
DS_FILE, /* Data comes from a VFS file */
DS_STRING /* Data comes from a string in memory */
};
struct area {
screen_dimen top, left;
screen_dimen height, width;
};
#define VLF_DISCARD 1
#define VLF_INIT 2
#define VLF_COMPLETE 3
/* basic structure for caching text, it correspond to one line in wrapped text.
* That makes easier to move in wrap mode.
* cache_lines are stored in two linked lists, one for normal mode
* and one for nroff mode
* cache_line is valid of end is not INVALID_OFFSET
* last line has set width to (screen_dimen) (-1)*/
/* never access next and previous in cache_line directly, use appropriately function
* instead */
struct cache_line {
/* number of line in text, so two cache_line have same number
* if they are on same text line (text line is wider than screen) */
long number;
/* previous cache_line in list*/
struct cache_line *prev;
/* next cache_line in list*/
struct cache_line *next;
/* offset when cache_line start in text,
* cache_line.start = cache_line->prev.end */
offset_type start;
/* offset when cache_line ends
* cache_line.end = cache_line->next.start */
offset_type end;
/* how many column take on screen */
screen_dimen width;
/* correct ident, if prevoius line ends by tabulator */
screen_dimen left;
};
struct WView {
Widget widget;
char *filename; /* Name of the file */
char *command; /* Command used to pipe data in */
enum view_ds datasource; /* Where the displayed data comes from */
/* stdio pipe data source */
FILE *ds_stdio_pipe; /* Output of a shell command */
/* vfs pipe data source */
int ds_vfs_pipe; /* Non-seekable vfs file descriptor */
/* vfs file data source */
int ds_file_fd; /* File with random access */
off_t ds_file_filesize; /* Size of the file */
off_t ds_file_offset; /* Offset of the currently loaded data */
byte *ds_file_data; /* Currently loaded data */
size_t ds_file_datalen; /* Number of valid bytes in file_data */
size_t ds_file_datasize; /* Number of allocated bytes in file_data */
/* string data source */
byte *ds_string_data; /* The characters of the string */
size_t ds_string_len; /* The length of the string */
/* Growing buffers information */
gboolean growbuf_in_use; /* Use the growing buffers? */
byte **growbuf_blockptr; /* Pointer to the block pointers */
size_t growbuf_blocks; /* The number of blocks in *block_ptr */
size_t growbuf_lastindex; /* Number of bytes in the last page of the
growing buffer */
gboolean growbuf_finished; /* TRUE when all data has been read. */
/* Editor modes */
gboolean hex_mode; /* Hexview or Hexedit */
gboolean hexedit_mode; /* Hexedit */
gboolean hexview_in_text; /* Is the hexview cursor in the text area? */
gboolean text_nroff_mode; /* Nroff-style highlighting */
gboolean text_wrap_mode; /* Wrap text lines to fit them on the screen */
gboolean magic_mode; /* Preprocess the file using external programs */
/* Additional editor state */
gboolean hexedit_lownibble; /* Are we editing the last significant nibble? */
/* Display information */
screen_dimen dpy_frame_size;/* Size of the frame surrounding the real viewer */
offset_type dpy_start; /* Offset of the displayed data */
offset_type dpy_end; /* Offset after the displayed data */
offset_type dpy_text_column;/* Number of skipped columns in non-wrap
* text mode */
offset_type hex_cursor; /* Hexview cursor position in file */
screen_dimen cursor_col; /* Cursor column */
screen_dimen cursor_row; /* Cursor row */
struct hexedit_change_node *change_list; /* Linked list of changes */
struct area status_area; /* Where the status line is displayed */
struct area ruler_area; /* Where the ruler is displayed */
struct area data_area; /* Where the data is displayed */
int dirty; /* Number of skipped updates */
gboolean dpy_bbar_dirty; /* Does the button bar need to be updated? */
/* Mode variables */
int bytes_per_line; /* Number of bytes per line in hex mode */
/* Search variables */
offset_type search_start; /* First character to start searching from */
offset_type search_end; /* Length of found string or 0 if none was found */
char *search_exp; /* The search expression */
int direction; /* 1= forward; -1 backward */
void (*last_search)(WView *);
/* Pointer to the last search command */
gboolean want_to_quit; /* Prepare for cleanup ... */
/* Markers */
int marker; /* mark to use */
offset_type marks [10]; /* 10 marks: 0..9 */
int move_dir; /* return value from widget:
* 0 do nothing
* -1 view previous file
* 1 view next file
*/
offset_type update_steps; /* The number of bytes between percent
* increments */
offset_type update_activate;/* Last point where we updated the status */
/* never access cache_line in view directly, use appropriately function
* instead */
/* first cache_line for normal mode */
struct cache_line *lines;
/* last cache_line for normal mode, is set when text is read to the end */
struct cache_line *lines_end;
/* first cache_line for nroff mode */
struct cache_line *nroff_lines;
/* last cache_line for nroff mode, is set when text is read to the end */
struct cache_line *nroff_lines_end;
/* cache_line, that is first showed on display (something like cursor)
* used for both normal adn nroff mode */
struct cache_line *first_showed_line;
/* converter for translation of text */
GIConv converter;
};
/* {{{ Global Variables }}} */
/* Maxlimit for skipping updates */
int max_dirt_limit = 10;
/* If set, show a ruler */
static enum ruler_type {
RULER_NONE,
RULER_TOP,
RULER_BOTTOM
} ruler = RULER_NONE;
/* Scrolling is done in pages or line increments */
int mouse_move_pages_viewer = 1;
/* wrap mode default */
int global_wrap_mode = 1;
int default_hex_mode = 0;
int default_magic_flag = 1;
int default_nroff_flag = 1;
int altered_hex_mode = 0;
int altered_magic_flag = 0;
int altered_nroff_flag = 0;
static const char hex_char[] = "0123456789ABCDEF";
int mcview_remember_file_position = FALSE;
/* {{{ Function Prototypes }}} */
/* Our widget callback */
static cb_ret_t view_callback (Widget *, widget_msg_t, int);
static int regexp_view_search (WView * view, char *pattern, char *string,
int match_type, size_t *match_start, size_t *match_end);
static void view_labels (WView * view);
static void view_init_growbuf (WView *);
static void view_place_cursor (WView *view);
static void display (WView *);
static void view_done (WView *);
/* {{{ Helper Functions }}} */
/* difference or zero */
static inline screen_dimen
dimen_doz (screen_dimen a, screen_dimen b)
{
return (a >= b) ? a - b : 0;
}
static inline screen_dimen
dimen_min (screen_dimen a, screen_dimen b)
{
return (a < b) ? a : b;
}
static inline offset_type
offset_doz (offset_type a, offset_type b)
{
return (a >= b) ? a - b : 0;
}
static inline offset_type
offset_rounddown (offset_type a, offset_type b)
{
assert (b != 0);
return a - a % b;
}
/* {{{ Simple Primitive Functions for WView }}} */
static inline gboolean
view_is_in_panel (WView *view)
{
return (view->dpy_frame_size != 0);
}
static inline int get_byte (WView *view, offset_type offset);
/* read on character from text, character is translated into terminal encoding
* writes result in ch, return how many bytes was read or -1 if end of text */
static int
view_get_char (WView *view, offset_type from, char *ch, int size)
{
static char buffer [MB_LEN_MAX + 1];
static int c;
static size_t result;
result = 0;
while (result < sizeof (buffer) - 1) {
c = get_byte (view, from + result);
if (c == -1) break;
buffer[result] = (unsigned char) c;
result++;
buffer[result] = '\0';
switch (str_translate_char (view->converter, buffer, result, ch, size)) {
case 0:
return (int) result;
case 1:
break;
case 2:
ch[0] = '?';
ch[1] = '\0';
return 1;
}
}
return -1;
}
/* this structure and view_read_* functions make reading text simpler
* view need remeber 4 following charsets: actual, next and two previous */
struct read_info {
char ch[4][MB_LEN_MAX + 1];
char *cnxt;
char *cact;
char *chi1;
char *chi2;
offset_type next;
offset_type actual;
int result;
};
/* set read_info into initial state and read first character to cnxt
* return how many bytes was read or -1 if end of text */
static void
view_read_start (WView *view, struct read_info *info, offset_type from)
{
info->ch[0][0] = '\0';
info->ch[1][0] = '\0';
info->ch[2][0] = '\0';
info->ch[3][0] = '\0';
info->cnxt = info->ch[0];
info->cact = info->ch[1];
info->chi1 = info->ch[2];
info->chi2 = info->ch[3];
info->next = from;
info->result = view_get_char (view, info->next, info->cnxt, MB_LEN_MAX + 1);
}
/* move characters in read_info one forward (chi2 = last previous is forgotten)
* read additional charsets into cnxt
* return how many bytes was read or -1 if end of text */
static void
view_read_continue (WView *view, struct read_info *info)
{
char *tmp;
tmp = info->chi2;
info->chi2 = info->chi1;
info->chi1 = info->cact;
info->cact = info->cnxt;
info->cnxt = tmp;
info->actual = info->next;
info->next+= info->result;
info->result = view_get_char (view, info->next, info->cnxt, MB_LEN_MAX + 1);
}
#define VRT_NW_NO 0
#define VRT_NW_YES 1
#define VRT_NW_CONTINUE 2
/* test if cact of read_info is newline
* return VRT_NW_NO, VRT_NW_YES
* or VRT_NW_CONTINUE follow after cact ("\r\n", ...) */
static int
view_read_test_new_line (WView *view, struct read_info *info)
{
#define cmp(t1,t2) (strcmp((t1),(t2)) == 0)
if (cmp (info->cact, "\n")) return VRT_NW_YES;
if (cmp (info->cact, "\r")) {
if (info->result != -1 && (cmp (info->cnxt, "\r") || cmp (info->cnxt, "\n")))
return VRT_NW_CONTINUE;
return VRT_NW_YES;
}
return VRT_NW_NO;
}
/* test if cact of read_info is tabulator
* return VRT_NW_NO or VRT_NW_YES */
static int
view_read_test_tabulator (WView *view, struct read_info *info)
{
#define cmp(t1,t2) (strcmp((t1),(t2)) == 0)
return cmp (info->cact, "\t");
}
/* test if read_info.cact is '\b' and charsets in read_info are correct
* nroff sequence, return VRT_NW_NO or VRT_NW_YES */
static int
view_read_test_nroff_back (WView *view, struct read_info *info)
{
#define cmp(t1,t2) (strcmp((t1),(t2)) == 0)
return view->text_nroff_mode && cmp (info->cact, "\b") &&
(info->result != -1) && str_isprint (info->cnxt) &&
!cmp (info->chi1, "") && str_isprint (info->chi1) &&
(cmp (info->chi1, info->cnxt) || cmp (info->chi1, "_")
|| (cmp (info->chi1, "+") && cmp (info->cnxt, "o")));
}
/* rutine for view_load_cache_line, line is ended and following cache_line is
* set up for loading
* used when end of line has been read in text */
static struct cache_line *
view_lc_create_next_line (struct cache_line *line, offset_type start)
{
struct cache_line *result = NULL;
line->end = start;
if (line->next == NULL) {
result = g_new0 (struct cache_line, 1);
line->next = result;
result->prev = line;
} else result = line->next;
result->start = start;
result->width = 0;
result->left = 0;
result->number = line->number + 1;
result->end = INVALID_OFFSET;
return result;
}
/* rutine for view_load_cache_line, line is ended and following cache_line is
* set up for loading
* used when line has maximum width */
static struct cache_line *
view_lc_create_wrap_line (struct cache_line *line, offset_type start,
screen_dimen left)
{
struct cache_line *result = NULL;
line->end = start;
if (line->next == NULL) {
result = g_new0 (struct cache_line, 1);
line->next = result;
result->prev = line;
} else result = line->next;
result->start = start;
result->width = 0;
result->left = left;
result->number = line->number;
result->end = INVALID_OFFSET;
return result;
}
/* read charsets fro text and set up correct width, end and start of next line
* view->data_area.width must be greater than 0 */
static struct cache_line *
view_load_cache_line (WView *view, struct cache_line *line)
{
const screen_dimen width = view->data_area.width;
struct read_info info;
offset_type nroff_start = 0;
int nroff_seq = 0;
int w;
line->width = 0;
view_read_start (view, &info, line->start);
while (info.result != -1) {
view_read_continue (view, &info);
switch (view_read_test_new_line (view, &info)) {
case VRT_NW_YES:
line = view_lc_create_next_line (line, info.next);
return line;
case VRT_NW_CONTINUE:
continue;
}
if (view_read_test_tabulator (view, &info)) {
line->width+= 8 - (line->left + line->width) % 8;
if ((width != 0) && (line->left + line->width >= width)) {
w = line->left + line->width - width;
/* if width of screen is very small, tabulator is cut to line
* left of next line is 0 */
w = (w < width) ? w : 0;
line->width = width - line->left;
line = view_lc_create_wrap_line (line, info.next, w);
return line;
}
}
if (view_read_test_nroff_back (view, &info)) {
w = str_term_width1 (info.chi1);
line->width-= w;
nroff_seq = 1;
continue;
}
/* assure, that nroff sequence never start in previous cache_line */
if (nroff_seq > 0)
nroff_seq--;
else
nroff_start = info.actual;
w = str_isprint (info.cact) ? str_term_width1 (info.cact) : 1;
if (line->left + line->width + w > width) {
line = view_lc_create_wrap_line (line, nroff_start, 0);
return line;
} else {
while (info.result != -1 && str_iscombiningmark (info.cnxt)) {
view_read_continue (view, &info);
}
}
line->width+= w;
}
/* text read to the end, seting lines_end*/
if (view->text_nroff_mode)
view->nroff_lines_end = line;
else
view->lines_end = line;
line = view_lc_create_next_line (line, info.next);
line->width = (screen_dimen) (-1);
line->end = info.next;
return line;
}
static void
view_lc_set_param (offset_type *param, offset_type value)
{
if ((*param) > value) (*param) = value;
}
/* offset to column or column to offset, value that will be maped must be
* set to INVALID_OFFSET, it is very analogous to view_load_cache_line
* and it is possible to integrate them in one function */
static void
view_map_offset_and_column (WView *view, struct cache_line *line,
offset_type *column, offset_type *offset)
{
const screen_dimen width = view->data_area.width;
struct read_info info;
offset_type nroff_start = 0;
int nroff_seq = 0;
int w;
screen_dimen col;
col = 0;
view_read_start (view, &info, line->start);
while (info.result != -1) {
view_read_continue (view, &info);
if (*column == INVALID_OFFSET) {
if (*offset < info.next) {
(*column) = col + line->left;
return;
}
}
switch (view_read_test_new_line (view, &info)) {
case VRT_NW_YES:
view_lc_set_param (offset, info.actual);
view_lc_set_param (column, line->left + col);
return;
case VRT_NW_CONTINUE:
continue;
}
if (view_read_test_tabulator (view, &info)) {
col+= 8 - (line->left + col) % 8;
if ((width != 0) && (line->left + col >= width)) {
w = line->left + col - width;
w = (w < width) ? w : 0;
col = width - line->left;
view_lc_set_param (offset, info.actual);
view_lc_set_param (column, line->left + col);
return;
}
}
if (view_read_test_nroff_back (view, &info)) {
w = str_term_width1 (info.chi1);
col-= w;
nroff_seq = 1;
continue;
}
if (nroff_seq > 0)
nroff_seq--;
else
nroff_start = info.actual;
w = str_isprint (info.cact) ? str_term_width1 (info.cact) : 1;
if (line->left + col + w > width) {
view_lc_set_param (offset, nroff_start);
view_lc_set_param (column, line->left + col);
return;
} else {
while (info.result != -1 && str_iscombiningmark (info.cnxt)) {
view_read_continue (view, &info);
}
}
col+= w;
if (*offset == INVALID_OFFSET) {
if (*column < col + line->left) {
(*offset) = nroff_start;
return;
}
}
}
view_lc_set_param (offset, info.actual);
view_lc_set_param (column, line->left + col);
}
/* macro, that iterate cache_line until stop holds,
* nf is function to iterate cache_line
* l is line to iterate and t is temporary line only */
#define view_move_to_stop(l,t,stop,nf) \
while (((t = nf (view, l)) != NULL) && (stop)) l = t;
/* make all cache_lines invalidet */
static void
view_reset_cache_lines (WView *view)
{
if (view->lines != NULL) {
view->lines->end = INVALID_OFFSET;
}
view->lines_end = NULL;
if (view->nroff_lines != NULL) {
view->nroff_lines->end = INVALID_OFFSET;
}
view->nroff_lines_end = NULL;
view->first_showed_line = NULL;
}
#define MAX_UNLOADED_CACHE_LINE 100
/* free some cache_lines, if count of cache_lines is bigger
* than MAX_UNLOADED_CACHE_LINE */
static void
view_reduce_cache_lines (WView *view)
{
struct cache_line *line;
struct cache_line *next;
int li;
li = 0;
line = view->lines;
while (line != NULL && li < MAX_UNLOADED_CACHE_LINE) {
line = line->next;
li++;
}
if (line != NULL) line->prev->next = NULL;
while (line != NULL) {
next = line->next;
g_free (line);
line = next;
}
view->lines_end = NULL;
line = view->nroff_lines;
while (line != NULL && li < MAX_UNLOADED_CACHE_LINE) {
line = line->next;
li++;
}
if (line != NULL) line->prev->next = NULL;
while (line != NULL) {
next = line->next;
g_free (line);
line = next;
}
view->nroff_lines_end = NULL;
view->first_showed_line = NULL;
}
/* return first cache_line for actual mode (normal / nroff) */
static struct cache_line *
view_get_first_line (WView *view)
{
struct cache_line **first_line;
first_line = (view->text_nroff_mode) ?
&(view->nroff_lines) : &(view->lines);
if (*first_line == NULL) {
(*first_line) = g_new0 (struct cache_line, 1);
(*first_line)->end = INVALID_OFFSET;
}
if ((*first_line)->end == INVALID_OFFSET)
view_load_cache_line (view, *first_line);
return (*first_line);
}
/* return following chahe_line or NULL */
static struct cache_line*
view_get_next_line (WView *view, struct cache_line *line)
{
struct cache_line *result;
if (line->next != NULL) {
result = line->next;
if (result->end == INVALID_OFFSET)
view_load_cache_line (view, result);
return (result->width != (screen_dimen) (-1)) ? result : NULL;
}
return NULL;
}
/* return last cache_line, it could take same time, because whole read text must
* be read (only once) */
static struct cache_line *
view_get_last_line (WView *view)
{
struct cache_line *result;
struct cache_line *next;
result = (view->text_nroff_mode) ?
view->nroff_lines_end : view->nroff_lines_end;
if (result != NULL) return result;
if (view->first_showed_line == NULL)
view->first_showed_line = view_get_first_line (view);
result = view->first_showed_line;
next = view_get_next_line (view, result);
while (next != NULL) {
result = next;
next = view_get_next_line (view, result);
}
return result;
}
/* return previous cache_line or NULL */
static struct cache_line *
view_get_previous_line (WView *view, struct cache_line *line)
{
return line->prev;
}
/* return first displayed cache_line */
static struct cache_line *
view_get_first_showed_line (WView *view)
{
struct cache_line *result;
if (view->first_showed_line == NULL)
view->first_showed_line = view_get_first_line (view);
result = view->first_showed_line;
return result;
}
/* return first cache_line with same number as line */
static struct cache_line *
view_get_start_of_whole_line (WView *view, struct cache_line *line)
{
struct cache_line *t;
if (line != NULL) {
view_move_to_stop (line, t, t->number == line->number, view_get_previous_line)
return line;
}
return NULL;
}
/* return last cache_line with same number as line */
static struct cache_line *
view_get_end_of_whole_line (WView *view, struct cache_line *line)
{
struct cache_line *t;
if (line != NULL) {
view_move_to_stop (line, t, t->number == line->number, view_get_next_line)
return line;
}
return NULL;
}
/* return last cache_line, that has number lesser than line
* or NULL */
static struct cache_line *
view_get_previous_whole_line (WView *view, struct cache_line *line)
{
line = view_get_start_of_whole_line (view, line);
return view_get_previous_line (view, line);
}
/* return first cache_line, that has number greater than line
* or NULL */
static struct cache_line *
view_get_next_whole_line (WView *view, struct cache_line *line)
{
line = view_get_end_of_whole_line (view, line);
return view_get_next_line (view, line);
}
/* return sum of widths of all cache_lines that has same number as line */
static screen_dimen
view_width_of_whole_line (WView *view, struct cache_line *line)
{
struct cache_line *next;
screen_dimen result = 0;
line = view_get_start_of_whole_line (view, line);
next = view_get_next_line (view, line);
while ((next != NULL) && (next->number == line->number)) {
result+= line->left + line->width;
line = next;
next = view_get_next_line (view, line);
}
result+= line->left + line->width;
return result;
}
/* return sum of widths of cache_lines before line, that has same number as line */
static screen_dimen
view_width_of_whole_line_before (WView *view, struct cache_line *line)
{
struct cache_line *next;
screen_dimen result = 0;
next = view_get_start_of_whole_line (view, line);
while (next != line) {
result+= next->left + next->width;
next = view_get_next_line (view, next);
}
return result;
}
/* map column to offset and cache_line */
static offset_type
view_column_to_offset (WView *view, struct cache_line **line, offset_type column)
{
struct cache_line *next;
offset_type result;
*line = view_get_start_of_whole_line (view, *line);
while (column >= (*line)->left + (*line)->width) {
column-= (*line)->left + (*line)->width;
result = (*line)->end;
next = view_get_next_line (view, *line);
if ((next == NULL) || (next->number != (*line)->number)) break;
(*line) = next;
}
// if (column < (*line)->left + (*line)->width) {
result = INVALID_OFFSET,
view_map_offset_and_column (view, *line, &column, &result);
// }
return result;
}
/* map offset to cache_line */
static struct cache_line *
view_offset_to_line (WView *view, offset_type from)
{
struct cache_line *result;
struct cache_line *t;
result = view_get_first_line (view);
view_move_to_stop (result, t, result->end <= from, view_get_next_line)
return result;
}
/* map offset to cache_line, searching starts from line */
static struct cache_line *
view_offset_to_line_from (WView *view, offset_type from, struct cache_line *line)
{
struct cache_line *result;
struct cache_line *t;
result = line;
view_move_to_stop (result, t, result->start > from, view_get_previous_line)
view_move_to_stop (result, t, result->end <= from, view_get_next_line)
return result;
}
/* mam offset to column */
static screen_dimen
view_offset_to_column (WView *view, struct cache_line *line, offset_type from)
{
offset_type result = INVALID_OFFSET;
view_map_offset_and_column (view, line, &result, &from);
result+= view_width_of_whole_line_before (view, line);
return result;
}
static void
view_compute_areas (WView *view)
{
struct area view_area;
struct cache_line *next;
screen_dimen height, rest, y;
screen_dimen old_width;
old_width = view->data_area.width;
/* The viewer is surrounded by a frame of size view->dpy_frame_size.
* Inside that frame, there are: The status line (at the top),
* the data area and an optional ruler, which is shown above or
* below the data area. */
view_area.top = view->dpy_frame_size;
view_area.left = view->dpy_frame_size;
view_area.height = dimen_doz(view->widget.lines, 2 * view->dpy_frame_size);
view_area.width = dimen_doz(view->widget.cols, 2 * view->dpy_frame_size);
/* Most coordinates of the areas equal those of the whole viewer */
view->status_area = view_area;
view->ruler_area = view_area;
view->data_area = view_area;
/* Compute the heights of the areas */
rest = view_area.height;
height = dimen_min(rest, 1);
view->status_area.height = height;
rest -= height;
height = dimen_min(rest, (ruler == RULER_NONE || view->hex_mode) ? 0 : 2);
view->ruler_area.height = height;
rest -= height;
view->data_area.height = rest;
/* Compute the position of the areas */
y = view_area.top;
view->status_area.top = y;
y += view->status_area.height;
if (ruler == RULER_TOP) {
view->ruler_area.top = y;
y += view->ruler_area.height;
}
view->data_area.top = y;
y += view->data_area.height;
if (ruler == RULER_BOTTOM) {
view->ruler_area.top = y;
y += view->ruler_area.height;
}
if (old_width != view->data_area.width) {
view_reset_cache_lines (view);
view->first_showed_line = view_get_first_line (view);
next = view_get_next_line (view, view->first_showed_line);
while ((next != NULL) && (view->first_showed_line->end <= view->dpy_start)) {
view->first_showed_line = next;
next = view_get_next_line (view, view->first_showed_line);
}
view->dpy_start = view->first_showed_line->start;
}
}
static void
view_hexedit_free_change_list (WView *view)
{
struct hexedit_change_node *curr, *next;
for (curr = view->change_list; curr != NULL; curr = next) {
next = curr->next;
g_free (curr);
}
view->change_list = NULL;
view->dirty++;
}
/* {{{ Growing buffer }}} */
static void
view_init_growbuf (WView *view)
{
view->growbuf_in_use = TRUE;
view->growbuf_blockptr = NULL;
view->growbuf_blocks = 0;
view->growbuf_lastindex = VIEW_PAGE_SIZE;
view->growbuf_finished = FALSE;
}
static void
view_growbuf_free (WView *view)
{
size_t i;
assert (view->growbuf_in_use);
for (i = 0; i < view->growbuf_blocks; i++)
g_free (view->growbuf_blockptr[i]);
g_free (view->growbuf_blockptr);
view->growbuf_blockptr = NULL;
view->growbuf_in_use = FALSE;
}
static offset_type
view_growbuf_filesize (WView *view)
{
assert(view->growbuf_in_use);
if (view->growbuf_blocks == 0)
return 0;
else
return ((offset_type) view->growbuf_blocks - 1) * VIEW_PAGE_SIZE
+ view->growbuf_lastindex;
}
/* Copies the output from the pipe to the growing buffer, until either
* the end-of-pipe is reached or the interval [0..ofs) of the growing
* buffer is completely filled. */
static void
view_growbuf_read_until (WView *view, offset_type ofs)
{
ssize_t nread;
byte *p;
size_t bytesfree;
gboolean short_read;
assert (view->growbuf_in_use);
if (view->growbuf_finished)
return;
short_read = FALSE;
while (view_growbuf_filesize (view) < ofs || short_read) {
if (view->growbuf_lastindex == VIEW_PAGE_SIZE) {
/* Append a new block to the growing buffer */
byte *newblock = g_try_malloc (VIEW_PAGE_SIZE);
byte **newblocks = g_try_malloc (sizeof (*newblocks) * (view->growbuf_blocks + 1));
if (!newblock || !newblocks) {
g_free (newblock);
g_free (newblocks);
return;
}
memcpy (newblocks, view->growbuf_blockptr, sizeof (*newblocks) * view->growbuf_blocks);
g_free (view->growbuf_blockptr);
view->growbuf_blockptr = newblocks;
view->growbuf_blockptr[view->growbuf_blocks++] = newblock;
view->growbuf_lastindex = 0;
}
p = view->growbuf_blockptr[view->growbuf_blocks - 1] + view->growbuf_lastindex;
bytesfree = VIEW_PAGE_SIZE - view->growbuf_lastindex;
if (view->datasource == DS_STDIO_PIPE) {
nread = fread (p, 1, bytesfree, view->ds_stdio_pipe);
if (nread == 0) {
view->growbuf_finished = TRUE;
(void) pclose (view->ds_stdio_pipe);
display (view);
close_error_pipe (D_NORMAL, NULL);
view->ds_stdio_pipe = NULL;
return;
}
} else {
assert (view->datasource == DS_VFS_PIPE);
do {
nread = mc_read (view->ds_vfs_pipe, p, bytesfree);
} while (nread == -1 && errno == EINTR);
if (nread == -1 || nread == 0) {
view->growbuf_finished = TRUE;
(void) mc_close (view->ds_vfs_pipe);
view->ds_vfs_pipe = -1;
return;
}
}
short_read = ((size_t)nread < bytesfree);
view->growbuf_lastindex += nread;
}
}
static int
get_byte_growing_buffer (WView *view, offset_type byte_index)
{
offset_type pageno = byte_index / VIEW_PAGE_SIZE;
offset_type pageindex = byte_index % VIEW_PAGE_SIZE;
assert (view->growbuf_in_use);
if ((size_t) pageno != pageno)
return -1;
view_growbuf_read_until (view, byte_index + 1);
if (view->growbuf_blocks == 0)
return -1;
if (pageno < view->growbuf_blocks - 1)
return view->growbuf_blockptr[pageno][pageindex];
if (pageno == view->growbuf_blocks - 1 && pageindex < view->growbuf_lastindex)
return view->growbuf_blockptr[pageno][pageindex];
return -1;
}
/* {{{ Data sources }}} */
/*
The data source provides the viewer with data from either a file, a
string or the output of a command. The get_byte() function can be
used to get the value of a byte at a specific offset. If the offset
is out of range, -1 is returned. The function get_byte_indexed(a,b)
returns the byte at the offset a+b, or -1 if a+b is out of range.
The view_set_byte() function has the effect that later calls to
get_byte() will return the specified byte for this offset. This
function is designed only for use by the hexedit component after
saving its changes. Inspect the source before you want to use it for
other purposes.
The view_get_filesize() function returns the current size of the
data source. If the growing buffer is used, this size may increase
later on. Use the view_may_still_grow() function when you want to
know if the size can change later.
*/
static offset_type
view_get_filesize (WView *view)
{
switch (view->datasource) {
case DS_NONE:
return 0;
case DS_STDIO_PIPE:
case DS_VFS_PIPE:
return view_growbuf_filesize (view);
case DS_FILE:
return view->ds_file_filesize;
case DS_STRING:
return view->ds_string_len;
default:
assert(!"Unknown datasource type");
return 0;
}
}
static inline gboolean
view_may_still_grow (WView *view)
{
return (view->growbuf_in_use && !view->growbuf_finished);
}
/* returns TRUE if the idx lies in the half-open interval
* [offset; offset + size), FALSE otherwise.
*/
static inline gboolean
already_loaded (offset_type offset, offset_type idx, size_t size)
{
return (offset <= idx && idx - offset < size);
}
static inline void
view_file_load_data (WView *view, offset_type byte_index)
{
offset_type blockoffset;
ssize_t res;
size_t bytes_read;
assert (view->datasource == DS_FILE);
if (already_loaded (view->ds_file_offset, byte_index, view->ds_file_datalen))
return;
if (byte_index >= view->ds_file_filesize)
return;
blockoffset = offset_rounddown (byte_index, view->ds_file_datasize);
if (mc_lseek (view->ds_file_fd, blockoffset, SEEK_SET) == -1)
goto error;
bytes_read = 0;
while (bytes_read < view->ds_file_datasize) {
res = mc_read (view->ds_file_fd, view->ds_file_data + bytes_read, view->ds_file_datasize - bytes_read);
if (res == -1)
goto error;
if (res == 0)
break;
bytes_read += (size_t) res;
}
view->ds_file_offset = blockoffset;
if (bytes_read > view->ds_file_filesize - view->ds_file_offset) {
/* the file has grown in the meantime -- stick to the old size */
view->ds_file_datalen = view->ds_file_filesize - view->ds_file_offset;
} else {
view->ds_file_datalen = bytes_read;
}
return;
error:
view->ds_file_datalen = 0;
}
static int
get_byte_none (WView *view, offset_type byte_index)
{
assert (view->datasource == DS_NONE);
(void) &view;
(void) byte_index;
return -1;
}
static inline int
get_byte_file (WView *view, offset_type byte_index)
{
assert (view->datasource == DS_FILE);
view_file_load_data (view, byte_index);
if (already_loaded(view->ds_file_offset, byte_index, view->ds_file_datalen))
return view->ds_file_data[byte_index - view->ds_file_offset];
return -1;
}
static int
get_byte_string (WView *view, offset_type byte_index)
{
assert (view->datasource == DS_STRING);
if (byte_index < view->ds_string_len)
return view->ds_string_data[byte_index];
return -1;
}
static inline int
get_byte (WView *view, offset_type offset)
{
switch (view->datasource) {
case DS_STDIO_PIPE:
case DS_VFS_PIPE:
return get_byte_growing_buffer (view, offset);
case DS_FILE:
return get_byte_file (view, offset);
case DS_STRING:
return get_byte_string (view, offset);
case DS_NONE:
return get_byte_none (view, offset);
}
assert(!"Unknown datasource type");
return -1;
}
static inline int
get_byte_indexed (WView *view, offset_type base, offset_type ofs)
{
if (base <= OFFSETTYPE_MAX - ofs)
return get_byte (view, base + ofs);
return -1;
}
static void
view_set_byte (WView *view, offset_type offset, byte b)
{
(void) &b;
assert (offset < view_get_filesize (view));
assert (view->datasource == DS_FILE);
view->ds_file_datalen = 0; /* just force reloading */
}
static void
view_set_datasource_none (WView *view)
{
view->datasource = DS_NONE;
}
static void
view_set_datasource_vfs_pipe (WView *view, int fd)
{
assert (fd != -1);
view->datasource = DS_VFS_PIPE;
view->ds_vfs_pipe = fd;
view_init_growbuf (view);
}
static void
view_set_datasource_stdio_pipe (WView *view, FILE *fp)
{
assert (fp != NULL);
view->datasource = DS_STDIO_PIPE;
view->ds_stdio_pipe = fp;
view_init_growbuf (view);
}
static void
view_set_datasource_string (WView *view, const char *s)
{
view->datasource = DS_STRING;
view->ds_string_data = (byte *) g_strdup (s);
view->ds_string_len = strlen (s);
}
static void
view_set_datasource_file (WView *view, int fd, const struct stat *st)
{
view->datasource = DS_FILE;
view->ds_file_fd = fd;
view->ds_file_filesize = st->st_size;
view->ds_file_offset = 0;
view->ds_file_data = g_malloc (4096);
view->ds_file_datalen = 0;
view->ds_file_datasize = 4096;
}
static void
view_close_datasource (WView *view)
{
switch (view->datasource) {
case DS_NONE:
break;
case DS_STDIO_PIPE:
if (view->ds_stdio_pipe != NULL) {
(void) pclose (view->ds_stdio_pipe);
display (view);
close_error_pipe (D_NORMAL, NULL);
view->ds_stdio_pipe = NULL;
}
view_growbuf_free (view);
break;
case DS_VFS_PIPE:
if (view->ds_vfs_pipe != -1) {
(void) mc_close (view->ds_vfs_pipe);
view->ds_vfs_pipe = -1;
}
view_growbuf_free (view);
break;
case DS_FILE:
(void) mc_close (view->ds_file_fd);
view->ds_file_fd = -1;
g_free (view->ds_file_data);
view->ds_file_data = NULL;
break;
case DS_STRING:
g_free (view->ds_string_data);
view->ds_string_data = NULL;
break;
default:
assert (!"Unknown datasource type");
}
view->datasource = DS_NONE;
}
/* {{{ Cursor Movement }}} */
/*
The following variables have to do with the current position and are
updated by the cursor movement functions.
In hex view and wrapped text view mode, dpy_start marks the offset of
the top-left corner on the screen, in non-wrapping text mode it is
the beginning of the current line. In hex mode, hex_cursor is the
offset of the cursor. In non-wrapping text mode, dpy_text_column is
the number of columns that are hidden on the left side on the screen.
In hex mode, dpy_start is updated by the view_fix_cursor_position()
function in order to keep the other functions simple. In
non-wrapping text mode dpy_start and dpy_text_column are normalized
such that dpy_text_column < view_get_datacolumns().
*/
/* prototypes for functions used by view_moveto_bottom() */
static void view_move_up (WView *, offset_type);
static void view_moveto_bol (WView *);
/* set view->first_showed_line and view->dpy_start
* use view->dpy_text_column in nowrap mode */
static void
view_set_first_showed (WView *view, struct cache_line *line)
{
if (view->text_wrap_mode) {
view->dpy_start = line->start;
view->first_showed_line = line;
} else {
view->dpy_start = view_column_to_offset (view, &line, view->dpy_text_column);
view->first_showed_line = line;
}
if (view->search_start == view->search_end) {
view->search_start = view->dpy_start;
view->search_end = view->dpy_start;
}
}
static void
view_scroll_to_cursor (WView *view)
{
if (view->hex_mode) {
const offset_type bytes = view->bytes_per_line;
const offset_type displaysize = view->data_area.height * bytes;
const offset_type cursor = view->hex_cursor;
offset_type topleft = view->dpy_start;
if (topleft + displaysize <= cursor)
topleft = offset_rounddown (cursor, bytes)
- (displaysize - bytes);
if (cursor < topleft)
topleft = offset_rounddown (cursor, bytes);
view->dpy_start = topleft;
} else if (view->text_wrap_mode) {
view->dpy_text_column = 0;
} else {
}
}
static void
view_movement_fixups (WView *view, gboolean reset_search)
{
view_scroll_to_cursor (view);
if (reset_search) {
view->search_start = view->dpy_start;
view->search_end = view->dpy_start;
}
view->dirty++;
}
static void
view_moveto_top (WView *view)
{
view->dpy_start = 0;
view->hex_cursor = 0;
view->first_showed_line = view_get_first_line (view);
view->dpy_text_column = 0;
view_movement_fixups (view, TRUE);
}
static void
view_moveto_bottom (WView *view)
{
offset_type datalines, lines_up, filesize, last_offset;
struct cache_line *line;
if (view->growbuf_in_use)
view_growbuf_read_until (view, OFFSETTYPE_MAX);
filesize = view_get_filesize (view);
last_offset = offset_doz(filesize, 1);
datalines = view->data_area.height;
lines_up = offset_doz(datalines, 1);
if (view->hex_mode) {
view->hex_cursor = filesize;
view_move_up (view, lines_up);
view->hex_cursor = last_offset;
} else {
line = view_get_last_line (view);
if (!view->text_wrap_mode)
line = view_get_start_of_whole_line (view, line);
view_set_first_showed (view, line);
view->dpy_text_column = 0;
view_move_up (view, lines_up);
}
view_movement_fixups (view, TRUE);
}
static void
view_moveto_bol (WView *view)
{
struct cache_line *line;
if (view->hex_mode) {
view->hex_cursor -= view->hex_cursor % view->bytes_per_line;
} else if (view->text_wrap_mode) {
/* do nothing */
} else {
line = view_get_first_showed_line (view);
line = view_get_start_of_whole_line (view, line);
view->dpy_text_column = 0;
view_set_first_showed (view, line);
}
view_movement_fixups (view, TRUE);
}
static void
view_moveto_eol (WView *view)
{
const screen_dimen width = view->data_area.width;
struct cache_line *line;
screen_dimen w;
if (view->hex_mode) {
offset_type filesize, bol;
bol = offset_rounddown (view->hex_cursor, view->bytes_per_line);
if (get_byte_indexed (view, bol, view->bytes_per_line - 1) != -1) {
view->hex_cursor = bol + view->bytes_per_line - 1;
} else {
filesize = view_get_filesize (view);
view->hex_cursor = offset_doz(filesize, 1);
}
} else if (view->text_wrap_mode) {
/* nothing to do */
} else {
line = view_get_first_showed_line (view);
line = view_get_start_of_whole_line (view, line);
w = view_width_of_whole_line (view, line);
if (w > width) {
view->dpy_text_column = w - width;
} else {
// if (w + width <= view->dpy_text_column) {
view->dpy_text_column = 0;
// }
}
view_set_first_showed (view, line);
}
view_movement_fixups (view, FALSE);
}
static void
view_moveto_offset (WView *view, offset_type offset)
{
struct cache_line *line;
if (view->hex_mode) {
view->hex_cursor = offset;
view->dpy_start = offset - offset % view->bytes_per_line;
} else {
line = view_offset_to_line (view, offset);
view->dpy_start = (view->text_wrap_mode) ? line->start : offset;
view->first_showed_line = line;
view->dpy_text_column = (view->text_wrap_mode) ?
0 : view_offset_to_column (view, line, offset);
}
view_movement_fixups (view, TRUE);
}
static void
view_moveto (WView *view, offset_type row, offset_type col)
{
struct cache_line *act;
struct cache_line *t;
act = view_get_first_line (view);
view_move_to_stop (act, t, act->number != row, view_get_next_line)
view->dpy_text_column = (view->text_wrap_mode) ? 0 : col;
view->dpy_start = view_column_to_offset (view, &act, col);
view->dpy_start = (view->text_wrap_mode) ? act->start : view->dpy_start;
view->first_showed_line = act;
if (view->hex_mode)
view_moveto_offset (view, view->dpy_start);
}
/* extendet view_move_to_stop, now has counter, too */
#define view_count_to_stop(l,t,i,stop,nf)\
while (((t = nf(view, l)) != NULL) && (stop)) {\
l = t;i++;}
static void
view_move_up (WView *view, offset_type lines)
{
struct cache_line *line, *t;
int li;
if (view->hex_mode) {
offset_type bytes = lines * view->bytes_per_line;
if (view->hex_cursor >= bytes) {
view->hex_cursor -= bytes;
if (view->hex_cursor < view->dpy_start)
view->dpy_start = offset_doz (view->dpy_start, bytes);
} else {
view->hex_cursor %= view->bytes_per_line;
}
} else if (view->text_wrap_mode) {
line = view_get_first_showed_line (view);
li = 0;
view_count_to_stop (line, t, li, (li < lines), view_get_previous_line)
view_set_first_showed (view, line);
} else {
line = view_get_first_showed_line (view);
li = 0;
view_count_to_stop (line, t, li, (li < lines), view_get_previous_whole_line)
view_set_first_showed (view, line);
}
view_movement_fixups (view, (lines != 1));
}
static void
view_move_down (WView *view, offset_type lines)
{
struct cache_line *line;
struct cache_line *t;
int li;
void
return_up (struct cache_line * (*ne) (WView *, struct cache_line *),
struct cache_line * (*pr) (WView *, struct cache_line *))
{
li = 0;
t = line;
while ((t != NULL) && (li < view->data_area.height)) {
li++;
t = ne (view, t);
}
li = view->data_area.height - li;
t = pr (view, line);
while ((t != NULL) && (li > 0)) {
line = t;
t = pr (view, line);
li--;
}
}
if (view->hex_mode) {
offset_type i, limit, last_byte;
last_byte = view_get_filesize (view);
if (last_byte >= (offset_type) view->bytes_per_line)
limit = last_byte - view->bytes_per_line;
else
limit = 0;
for (i = 0; i < lines && view->hex_cursor < limit; i++) {
view->hex_cursor += view->bytes_per_line;
if (lines != 1)
view->dpy_start += view->bytes_per_line;
}
} else if (view->text_wrap_mode) {
line = view_get_first_showed_line (view);
li = 0;
view_count_to_stop (line, t, li, li < lines, view_get_next_line)
// return_up (view_get_next_line, view_get_previous_line);
view_set_first_showed (view, line);
} else {
line = view_get_first_showed_line (view);
li = 0;
view_count_to_stop (line, t, li, li < lines, view_get_next_whole_line)
// return_up (view_get_next_whole_line, view_get_previous_whole_line);
view_set_first_showed (view, line);
}
view_movement_fixups (view, (lines != 1));
}
static void
view_move_left (WView *view, offset_type columns)
{
struct cache_line *line;
if (view->hex_mode) {
assert (columns == 1);
if (view->hexview_in_text || !view->hexedit_lownibble) {
if (view->hex_cursor > 0)
view->hex_cursor--;
}
if (!view->hexview_in_text)
view->hexedit_lownibble = !view->hexedit_lownibble;
} else if (view->text_wrap_mode) {
/* nothing to do */
} else {
if (view->dpy_text_column >= columns)
view->dpy_text_column-= columns;
else
view->dpy_text_column = 0;
line = view_get_first_showed_line (view);
view_set_first_showed (view, line);
}
view_movement_fixups (view, FALSE);
}
static void
view_move_right (WView *view, offset_type columns)
{
struct cache_line *line;
if (view->hex_mode) {
assert (columns == 1);
if (view->hexview_in_text || view->hexedit_lownibble) {
if (get_byte_indexed (view, view->hex_cursor, 1) != -1)
view->hex_cursor++;
}
if (!view->hexview_in_text)
view->hexedit_lownibble = !view->hexedit_lownibble;
} else if (view->text_wrap_mode) {
/* nothing to do */
} else {
view->dpy_text_column += columns;
line = view_get_first_showed_line (view);
view_set_first_showed (view, line);
}
view_movement_fixups (view, FALSE);
}
/* {{{ Toggling of viewer modes }}} */
static void
view_toggle_hex_mode (WView *view)
{
struct cache_line *line;
view->hex_mode = !view->hex_mode;
if (view->hex_mode) {
view->hex_cursor = view->dpy_start;
view->dpy_start =
offset_rounddown (view->dpy_start, view->bytes_per_line);
view->widget.options |= W_WANT_CURSOR;
} else {
line = view_offset_to_line (view, view->hex_cursor);
view->dpy_text_column = (view->text_wrap_mode) ? 0 :
view_offset_to_column (view, line, view->hex_cursor);
view_set_first_showed (view, line);
view->widget.options &= ~W_WANT_CURSOR;
}
altered_hex_mode = 1;
view->dpy_bbar_dirty = TRUE;
view->dirty++;
}
static void
view_toggle_hexedit_mode (WView *view)
{
view->hexedit_mode = !view->hexedit_mode;
view->dpy_bbar_dirty = TRUE;
view->dirty++;
}
static void
view_toggle_wrap_mode (WView *view)
{
struct cache_line *line;
view->text_wrap_mode = !view->text_wrap_mode;
if (view->text_wrap_mode) {
view->dpy_text_column = 0;
view->dpy_start = view_get_first_showed_line (view)->start;
} else {
line = view_get_first_showed_line (view);
view->dpy_text_column = view_width_of_whole_line_before (view, line);
}
view->dpy_bbar_dirty = TRUE;
view->dirty++;
}
static void
view_toggle_nroff_mode (WView *view)
{
struct cache_line *line;
struct cache_line *next;
view->text_nroff_mode = !view->text_nroff_mode;
altered_nroff_flag = 1;
view->dpy_bbar_dirty = TRUE;
view->dirty++;
line = view_get_first_line (view);
view_move_to_stop (line, next, line->end <= view->dpy_start, view_get_next_line)
view_set_first_showed (view, line);
}
static void
view_toggle_magic_mode (WView *view)
{
char *filename, *command;
altered_magic_flag = 1;
view->magic_mode = !view->magic_mode;
filename = g_strdup (view->filename);
command = g_strdup (view->command);
view_done (view);
view_load (view, command, filename, 0);
g_free (filename);
g_free (command);
view->dpy_bbar_dirty = TRUE;
view->dirty++;
}
/* {{{ Miscellaneous functions }}} */
static void
view_done (WView *view)
{
/* Save current file position */
if (mcview_remember_file_position && view->filename != NULL) {
struct cache_line *line;
char *canon_fname;
offset_type row, col;
canon_fname = vfs_canon (view->filename);
line = view_get_first_showed_line (view);
row = line->number + 1;
col = view_offset_to_column (view, line, view->dpy_start);
save_file_position (canon_fname, row, col);
g_free (canon_fname);
}
/* Write back the global viewer mode */
default_hex_mode = view->hex_mode;
default_nroff_flag = view->text_nroff_mode;
default_magic_flag = view->magic_mode;
global_wrap_mode = view->text_wrap_mode;
/* view->widget needs no destructor */
g_free (view->filename), view->filename = NULL;
g_free (view->command), view->command = NULL;
view_close_datasource (view);
/* the growing buffer is freed with the datasource */
view_hexedit_free_change_list (view);
/* FIXME: what about view->search_exp? */
/* Free memory used by the viewer */
view_reduce_cache_lines (view);
if (view->converter != str_cnv_from_term) str_close_conv (view->converter);
}
static void
view_show_error (WView *view, const char *msg)
{
view_close_datasource (view);
if (view_is_in_panel (view)) {
view_set_datasource_string (view, msg);
} else {
message (D_ERROR, MSG_ERROR, "%s", msg);
}
}
static gboolean
view_load_command_output (WView *view, const char *command)
{
FILE *fp;
view_close_datasource (view);
open_error_pipe ();
if ((fp = popen (command, "r")) == NULL) {
/* Avoid two messages. Message from stderr has priority. */
display (view);
if (!close_error_pipe (view_is_in_panel (view) ? -1 : D_ERROR, NULL))
view_show_error (view, _(" Cannot spawn child process "));
return FALSE;
}
/* First, check if filter produced any output */
view_set_datasource_stdio_pipe (view, fp);
if (get_byte (view, 0) == -1) {
view_close_datasource (view);
/* Avoid two messages. Message from stderr has priority. */
display (view);
if (!close_error_pipe (view_is_in_panel (view) ? -1 : D_ERROR, NULL))
view_show_error (view, _("Empty output from child filter"));
return FALSE;
}
return TRUE;
}
gboolean
view_load (WView *view, const char *command, const char *file,
int start_line)
{
int i, type;
int fd = -1;
char tmp[BUF_MEDIUM];
const char *enc;
char *canon_fname;
struct stat st;
gboolean retval = FALSE;
assert (view->bytes_per_line != 0);
view_done (view);
/* Set up the state */
view_set_datasource_none (view);
view->filename = g_strdup (file);
view->command = 0;
/* Clear the markers */
view->marker = 0;
for (i = 0; i < 10; i++)
view->marks[i] = 0;
if (!view_is_in_panel (view)) {
view->dpy_text_column = 0;
}
if (command && (view->magic_mode || file == NULL || file[0] == '\0')) {
retval = view_load_command_output (view, command);
} else if (file != NULL && file[0] != '\0') {
/* Open the file */
if ((fd = mc_open (file, O_RDONLY | O_NONBLOCK)) == -1) {
g_snprintf (tmp, sizeof (tmp), _(" Cannot open \"%s\"\n %s "),
file, unix_error_string (errno));
view_show_error (view, tmp);
g_free (view->filename);
view->filename = NULL;
goto finish;
}
/* Make sure we are working with a regular file */
if (mc_fstat (fd, &st) == -1) {
mc_close (fd);
g_snprintf (tmp, sizeof (tmp), _(" Cannot stat \"%s\"\n %s "),
file, unix_error_string (errno));
view_show_error (view, tmp);
g_free (view->filename);
view->filename = NULL;
goto finish;
}
if (!S_ISREG (st.st_mode)) {
mc_close (fd);
view_show_error (view, _(" Cannot view: not a regular file "));
g_free (view->filename);
view->filename = NULL;
goto finish;
}
if (st.st_size == 0 || mc_lseek (fd, 0, SEEK_SET) == -1) {
/* Must be one of those nice files that grow (/proc) */
view_set_datasource_vfs_pipe (view, fd);
} else {
type = get_compression_type (fd);
if (view->magic_mode && (type != COMPRESSION_NONE)) {
g_free (view->filename);
view->filename = g_strconcat (file, decompress_extension (type), (char *) NULL);
}
view_set_datasource_file (view, fd, &st);
}
retval = TRUE;
}
finish:
view->command = g_strdup (command);
view->dpy_start = 0;
view->search_start = 0;
view->search_end = 0;
view->dpy_text_column = 0;
view->last_search = 0; /* Start a new search */
view->converter = str_cnv_from_term;
/* try detect encoding from path */
if (view->filename != NULL) {
canon_fname = vfs_canon (view->filename);
enc = vfs_get_encoding (canon_fname);
if (enc != NULL) {
view->converter = str_crt_conv_from (enc);
if (view->converter == INVALID_CONV)
view->converter = str_cnv_from_term;
}
g_free (canon_fname);
}
view_compute_areas (view);
view_reset_cache_lines (view);
view->first_showed_line = view_get_first_line (view);
assert (view->bytes_per_line != 0);
if (mcview_remember_file_position && view->filename != NULL && start_line == 0) {
long line, col;
canon_fname = vfs_canon (view->filename);
load_file_position (file, &line, &col);
g_free (canon_fname);
view_moveto (view, offset_doz(line, 1), col);
} else if (start_line > 0) {
view_moveto (view, start_line - 1, 0);
}
view->hexedit_lownibble = FALSE;
view->hexview_in_text = FALSE;
view->change_list = NULL;
return retval;
}
/* {{{ Display management }}} */
static void
view_update_bytes_per_line (WView *view)
{
const screen_dimen cols = view->data_area.width;
int bytes;
if (cols < 8 + 17)
bytes = 4;
else
bytes = 4 * ((cols - 8) / ((cols < 80) ? 17 : 18));
assert(bytes != 0);
view->bytes_per_line = bytes;
view->dirty = max_dirt_limit + 1; /* To force refresh */
}
static void
view_percent (WView *view, offset_type p)
{
const screen_dimen top = view->status_area.top;
const screen_dimen right = view->status_area.left + view->status_area.width;
const screen_dimen height = view->status_area.height;
int percent;
offset_type filesize;
if (height < 1 || right < 4)
return;
if (view_may_still_grow (view))
return;
filesize = view_get_filesize (view);
if (filesize == 0 || view->dpy_end == filesize)
percent = 100;
else if (p > (INT_MAX / 100))
percent = p / (filesize / 100);
else
percent = p * 100 / filesize;
widget_move (view, top, right - 4);
tty_printf ("%3d%%", percent);
}
static void
view_display_status (WView *view)
{
const screen_dimen top = view->status_area.top;
const screen_dimen left = view->status_area.left;
const screen_dimen width = view->status_area.width;
const screen_dimen height = view->status_area.height;
const char *file_label, *file_name;
screen_dimen file_label_width;
int i;
char *tmp;
if (height < 1)
return;
tty_setcolor (SELECTED_COLOR);
widget_move (view, top, left);
hline (' ', width);
file_label = _("File: %s");
file_label_width = str_term_width1 (file_label) - 2;
file_name = view->filename ? view->filename
: view->command ? view->command
: "";
if (width < file_label_width + 6)
addstr (str_fit_to_term (file_name, width, J_LEFT_FIT));
else {
i = (width > 22 ? 22 : width) - file_label_width;
tmp = g_strdup_printf (file_label, str_fit_to_term (file_name, i, J_LEFT_FIT));
addstr (tmp);
g_free (tmp);
if (width > 46) {
widget_move (view, top, left + 24);
/* FIXME: the format strings need to be changed when offset_type changes */
if (view->hex_mode)
tty_printf (_("Offset 0x%08lx"), (unsigned long) view->hex_cursor);
else {
screen_dimen row, col;
struct cache_line *line;
line = view_get_first_showed_line (view);
row = line->number + 1;
col = (view->text_wrap_mode) ?
view_width_of_whole_line_before (view, line) :
view->dpy_text_column;
col++;
tty_printf (_("Line %lu Col %lu"),
(unsigned long) row, (unsigned long) col);
}
}
if (width > 62) {
offset_type filesize;
filesize = view_get_filesize (view);
widget_move (view, top, left + 43);
if (!view_may_still_grow (view)) {
tty_printf (_("%s bytes"), size_trunc (filesize));
} else {
tty_printf (_(">= %s bytes"), size_trunc (filesize));
}
}
if (width > 26) {
view_percent (view, view->hex_mode
? view->hex_cursor
: view->dpy_end);
}
}
tty_setcolor (SELECTED_COLOR);
}
static inline void
view_display_clean (WView *view)
{
tty_setcolor (NORMAL_COLOR);
widget_erase ((Widget *) view);
if (view->dpy_frame_size != 0) {
draw_double_box (view->widget.parent, view->widget.y,
view->widget.x, view->widget.lines,
view->widget.cols);
}
}
typedef enum {
MARK_NORMAL,
MARK_SELECTED,
MARK_CURSOR,
MARK_CHANGED
} mark_t;
static inline int
view_count_backspaces (WView *view, off_t offset)
{
int backspaces = 0;
while (offset >= 2 * backspaces
&& get_byte (view, offset - 2 * backspaces) == '\b')
backspaces++;
return backspaces;
}
static void
view_display_ruler (WView *view)
{
static const char ruler_chars[] = "|----*----";
const screen_dimen top = view->ruler_area.top;
const screen_dimen left = view->ruler_area.left;
const screen_dimen width = view->ruler_area.width;
const screen_dimen height = view->ruler_area.height;
const screen_dimen line_row = (ruler == RULER_TOP) ? 0 : 1;
const screen_dimen nums_row = (ruler == RULER_TOP) ? 1 : 0;
char r_buff[10];
offset_type cl;
screen_dimen c;
if (ruler == RULER_NONE || height < 1)
return;
tty_setcolor (MARKED_COLOR);
for (c = 0; c < width; c++) {
cl = view->dpy_text_column + c;
if (line_row < height) {
widget_move (view, top + line_row, left + c);
tty_print_char (ruler_chars[cl % 10]);
}
if ((cl != 0) && (cl % 10) == 0) {
g_snprintf (r_buff, sizeof (r_buff), "%"OFFSETTYPE_PRId, cl);
if (nums_row < height) {
widget_move (view, top + nums_row, left + c - 1);
tty_print_string (r_buff);
}
}
}
attrset (NORMAL_COLOR);
}
static void
view_display_hex (WView *view)
{
const screen_dimen top = view->data_area.top;
const screen_dimen left = view->data_area.left;
const screen_dimen height = view->data_area.height;
const screen_dimen width = view->data_area.width;
const int ngroups = view->bytes_per_line / 4;
const screen_dimen text_start =
8 + 13 * ngroups + ((width < 80) ? 0 : (ngroups - 1 + 1));
/* 8 characters are used for the file offset, and every hex group
* takes 13 characters. On ``big'' screens, the groups are separated
* by an extra vertical line, and there is an extra space before the
* text column.
*/
screen_dimen row, col;
offset_type from;
int c;
mark_t boldflag = MARK_NORMAL;
struct hexedit_change_node *curr = view->change_list;
size_t i;
char hex_buff[10]; /* A temporary buffer for sprintf and mvwaddstr */
int bytes; /* Number of bytes already printed on the line */
view_display_clean (view);
/* Find the first displayable changed byte */
from = view->dpy_start;
while (curr && (curr->offset < from)) {
curr = curr->next;
}
for (row = 0; get_byte (view, from) != -1 && row < height; row++) {
col = 0;
/* Print the hex offset */
g_snprintf (hex_buff, sizeof (hex_buff), "%08"OFFSETTYPE_PRIX" ", from);
widget_move (view, top + row, left);
tty_setcolor (MARKED_COLOR);
for (i = 0; col < width && hex_buff[i] != '\0'; i++) {
addch (hex_buff[i]);
/* tty_print_char(hex_buff[i]);*/
col += 1;
}
tty_setcolor (NORMAL_COLOR);
for (bytes = 0; bytes < view->bytes_per_line; bytes++, from++) {
if ((c = get_byte (view, from)) == -1)
break;
/* Save the cursor position for view_place_cursor() */
if (from == view->hex_cursor && !view->hexview_in_text) {
view->cursor_row = row;
view->cursor_col = col;
}
/* Determine the state of the current byte */
boldflag =
(from == view->hex_cursor) ? MARK_CURSOR
: (curr != NULL && from == curr->offset) ? MARK_CHANGED
: (view->search_start <= from &&
from < view->search_end) ? MARK_SELECTED
: MARK_NORMAL;
/* Determine the value of the current byte */
if (curr != NULL && from == curr->offset) {
c = curr->value;
curr = curr->next;
}
/* Select the color for the hex number */
tty_setcolor (
boldflag == MARK_NORMAL ? NORMAL_COLOR :
boldflag == MARK_SELECTED ? MARKED_COLOR :
boldflag == MARK_CHANGED ? VIEW_UNDERLINED_COLOR :
/* boldflag == MARK_CURSOR */
view->hexview_in_text ? MARKED_SELECTED_COLOR :
VIEW_UNDERLINED_COLOR);
/* Print the hex number */
widget_move (view, top + row, left + col);
if (col < width) {
tty_print_char (hex_char[c / 16]);
col += 1;
}
if (col < width) {
tty_print_char (hex_char[c % 16]);
col += 1;
}
/* Print the separator */
tty_setcolor (NORMAL_COLOR);
if (bytes != view->bytes_per_line - 1) {
if (col < width) {
tty_print_char (' ');
col += 1;
}
/* After every four bytes, print a group separator */
if (bytes % 4 == 3) {
if (view->data_area.width >= 80 && col < width) {
tty_print_one_vline ();
col += 1;
}
if (col < width) {
tty_print_char (' ');
col += 1;
}
}
}
/* Select the color for the character; this differs from the
* hex color when boldflag == MARK_CURSOR */
tty_setcolor (
boldflag == MARK_NORMAL ? NORMAL_COLOR :
boldflag == MARK_SELECTED ? MARKED_COLOR :
boldflag == MARK_CHANGED ? VIEW_UNDERLINED_COLOR :
/* boldflag == MARK_CURSOR */
view->hexview_in_text ? VIEW_UNDERLINED_COLOR :
MARKED_SELECTED_COLOR);
c = convert_to_display_c (c);
if (!g_ascii_isprint (c))
c = '.';
/* Print corresponding character on the text side */
if (text_start + bytes < width) {
widget_move (view, top + row, left + text_start + bytes);
tty_print_char (c);
}
/* Save the cursor position for view_place_cursor() */
if (from == view->hex_cursor && view->hexview_in_text) {
view->cursor_row = row;
view->cursor_col = text_start + bytes;
}
}
}
/* Be polite to the other functions */
tty_setcolor (NORMAL_COLOR);
view_place_cursor (view);
view->dpy_end = from;
}
static void
view_display_text (WView * view)
{
#define cmp(t1,t2) (strcmp((t1),(t2)) == 0)
const screen_dimen left = view->data_area.left;
const screen_dimen top = view->data_area.top;
const screen_dimen width = view->data_area.width;
const screen_dimen height = view->data_area.height;
struct read_info info;
offset_type row, col;
int w;
struct cache_line *line_act;
struct cache_line *line_nxt;
view_display_clean (view);
view_display_ruler (view);
tty_setcolor (NORMAL_COLOR);
widget_move (view, top, left);
line_act = view_get_first_showed_line (view);
row = 0;
/* set col correct value */
col = (view->text_wrap_mode) ? 0 : view_width_of_whole_line_before (view, line_act);
col+= line_act->left;
view_read_start (view, &info, line_act->start);
while ((info.result != -1) && (row < height)) {
/* real detection of new line */
if (info.next >= line_act->end) {
line_nxt = view_get_next_line (view, line_act);
if (line_nxt == NULL) break;
if (view->text_wrap_mode || (line_act->number != line_nxt->number)){
row++;
col = line_nxt->left;
}
line_act = line_nxt;
continue;
}
view_read_continue (view, &info);
if (view_read_test_nroff_back (view, &info)) {
w = str_term_width1 (info.chi1);
col-= w;
if (col >= view->dpy_text_column
&& col + w - view->dpy_text_column <= width) {
widget_move (view, top + row, left + (col - view->dpy_text_column));
int c;
for (c = 0; c < w; c++) addch (' ');
}
if (cmp (info.chi1, "_") && (!cmp (info.cnxt, "_") || !cmp (info.chi2, "\b")))
tty_setcolor (VIEW_UNDERLINED_COLOR);
else
tty_setcolor (MARKED_COLOR);
continue;
}
if (view_read_test_new_line (view, &info))
continue;
if (view_read_test_tabulator (view, &info)) {
col+= (8 - (col % 8));
continue;
}
if (view->search_start <= info.actual
&& info.actual < view->search_end) {
tty_setcolor (SELECTED_COLOR);
}
w = str_isprint (info.cact) ? str_term_width1 (info.cact) : 1;
if (col >= view->dpy_text_column
&& col + w - view->dpy_text_column <= width) {
widget_move (view, top + row, left + (col - view->dpy_text_column));
if (!str_iscombiningmark (info.cnxt)) {
if (str_isprint (info.cact)) {
addstr (str_term_form (info.cact));
} else {
addch ('.');
}
} else {
GString *comb = g_string_new ("");
if (str_isprint (info.cact)) {
g_string_append(comb,info.cact);
} else {
g_string_append(comb,".");
}
while (str_iscombiningmark (info.cnxt)) {
view_read_continue (view, &info);
g_string_append(comb,info.cact);
}
addstr (str_term_form (comb->str));
g_string_free (comb, TRUE);
}
} else {
while (str_iscombiningmark (info.cnxt)) {
view_read_continue (view, &info);
}
}
col+= w;
tty_setcolor (NORMAL_COLOR);
}
view->dpy_end = info.next;
}
/* Displays as much data from view->dpy_start as fits on the screen */
static void
display (WView *view)
{
view_compute_areas (view);
if (view->hex_mode) {
view_display_hex (view);
} else {
view_display_text (view);
}
view_display_status (view);
}
static void
view_place_cursor (WView *view)
{
const screen_dimen top = view->data_area.top;
const screen_dimen left = view->data_area.left;
screen_dimen col;
col = view->cursor_col;
if (!view->hexview_in_text && view->hexedit_lownibble)
col++;
widget_move (&view->widget, top + view->cursor_row, left + col);
}
static void
view_update (WView *view)
{
static int dirt_limit = 1;
if (view->dpy_bbar_dirty) {
view->dpy_bbar_dirty = FALSE;
view_labels (view);
buttonbar_redraw (view->widget.parent);
}
if (view->dirty > dirt_limit) {
/* Too many updates skipped -> force a update */
display (view);
view->dirty = 0;
/* Raise the update skipping limit */
dirt_limit++;
if (dirt_limit > max_dirt_limit)
dirt_limit = max_dirt_limit;
}
if (view->dirty) {
if (is_idle ()) {
/* We have time to update the screen properly */
display (view);
view->dirty = 0;
if (dirt_limit > 1)
dirt_limit--;
} else {
/* We are busy -> skipping full update,
only the status line is updated */
view_display_status (view);
}
/* Here we had a refresh, if fast scrolling does not work
restore the refresh, although this should not happen */
}
}
/* {{{ Hex editor }}} */
static void
enqueue_change (struct hexedit_change_node **head,
struct hexedit_change_node *node)
{
/* chnode always either points to the head of the list or
* to one of the ->next fields in the list. The value at
* this location will be overwritten with the new node. */
struct hexedit_change_node **chnode = head;
while (*chnode != NULL && (*chnode)->offset < node->offset)
chnode = &((*chnode)->next);
node->next = *chnode;
*chnode = node;
}
static cb_ret_t
view_handle_editkey (WView *view, int key)
{
struct hexedit_change_node *node;
byte byte_val;
/* Has there been a change at this position? */
node = view->change_list;
while (node && (node->offset != view->hex_cursor))
node = node->next;
if (!view->hexview_in_text) {
/* Hex editing */
unsigned int hexvalue = 0;
if (key >= '0' && key <= '9')
hexvalue = 0 + (key - '0');
else if (key >= 'A' && key <= 'F')
hexvalue = 10 + (key - 'A');
else if (key >= 'a' && key <= 'f')
hexvalue = 10 + (key - 'a');
else
return MSG_NOT_HANDLED;
if (node)
byte_val = node->value;
else
byte_val = get_byte (view, view->hex_cursor);
if (view->hexedit_lownibble) {
byte_val = (byte_val & 0xf0) | (hexvalue);
} else {
byte_val = (byte_val & 0x0f) | (hexvalue << 4);
}
} else {
/* Text editing */
if (key < 256 && (is_printable (key) || (key == '\n')))
byte_val = key;
else
return MSG_NOT_HANDLED;
}
if (!node) {
node = g_new (struct hexedit_change_node, 1);
node->offset = view->hex_cursor;
node->value = byte_val;
enqueue_change (&view->change_list, node);
} else {
node->value = byte_val;
}
view->dirty++;
view_update (view);
view_move_right (view, 1);
return MSG_HANDLED;
}
static gboolean
view_hexedit_save_changes (WView *view)
{
struct hexedit_change_node *curr, *next;
int fp, answer;
char *text, *error;
if (view->change_list == NULL)
return TRUE;
retry_save:
assert (view->filename != NULL);
fp = mc_open (view->filename, O_WRONLY);
if (fp == -1)
goto save_error;
for (curr = view->change_list; curr != NULL; curr = next) {
next = curr->next;
if (mc_lseek (fp, curr->offset, SEEK_SET) == -1
|| mc_write (fp, &(curr->value), 1) != 1)
goto save_error;
/* delete the saved item from the change list */
view->change_list = next;
view->dirty++;
view_set_byte (view, curr->offset, curr->value);
g_free (curr);
}
if (mc_close (fp) == -1) {
error = g_strdup (strerror (errno));
message (D_ERROR, _(" Save file "), _(
" Error while closing the file: \n %s \n"
" Data may have been written or not. "), error);
g_free (error);
}
view_update (view);
return TRUE;
save_error:
error = g_strdup (strerror (errno));
text = g_strdup_printf (_(" Cannot save file: \n %s "), error);
g_free (error);
(void) mc_close (fp);
answer = query_dialog (_(" Save file "), text, D_ERROR,
2, _("&Retry"), _("&Cancel"));
g_free (text);
if (answer == 0)
goto retry_save;
return FALSE;
}
/* {{{ Miscellaneous functions }}} */
static gboolean
view_ok_to_quit (WView *view)
{
int r;
if (view->change_list == NULL)
return TRUE;
r = query_dialog (_("Quit"),
_(" File was modified, Save with exit? "), D_NORMAL, 3,
_("&Cancel quit"), _("&Yes"), _("&No"));
switch (r) {
case 1:
return view_hexedit_save_changes (view);
case 2:
view_hexedit_free_change_list (view);
return TRUE;
default:
return FALSE;
}
}
static inline void
my_define (Dlg_head *h, int idx, const char *text, void (*fn) (WView *),
WView *view)
{
buttonbar_set_label_data (h, idx, text, (buttonbarfn) fn, view);
}
/* {{{ Searching }}} */
/* Case insensitive search of text in data */
static int
icase_search_p (WView *view, char *text, char *data, int nothing,
size_t *match_start, size_t *match_end)
{
const char *q;
(void) nothing;
q = (view->direction == 1)
? str_search_first (data, text, 0)
: str_search_last (data, text, 0);
if (q != NULL) {
(*match_start) = str_length_noncomb (data) - str_length_noncomb (q);
(*match_end) = (*match_start) + str_length_noncomb (text);
return 1;
}
return 0;
}
/* read one whole line into buffer, return where line start and end */
static int
view_get_line_at (WView *view, offset_type from, GString * buffer,
offset_type *buff_start, offset_type *buff_end)
{
#define cmp(t1,t2) (strcmp((t1),(t2)) == 0)
struct read_info info;
struct cache_line *line;
offset_type start;
offset_type end;
line = view_get_first_showed_line (view);
line = view_offset_to_line_from (view, from, line);
if (view->direction == 1) {
start = from;
end = view_get_end_of_whole_line (view, line)->end;
if (start >= end) return 0;
} else {
start = view_get_start_of_whole_line (view, line)->start;
end = from;
}
(*buff_start) = start;
(*buff_end) = end;
g_string_set_size(buffer,0);
view_read_start (view, &info, start);
while ((info.result != -1) && (info.next < end)) {
view_read_continue (view, &info);
/* if text contains '\0' */
if (cmp (info.cact, "")) {
if (info.actual < from) {
/* '\0' before start offset, continue */
g_string_set_size(buffer,0);
(*buff_start) = info.next;
continue;
} else {
/* '\0' after start offset, end */
(*buff_end) = info.next;
return 1;
}
}
if (view_read_test_new_line (view, &info))
continue;
if (view_read_test_nroff_back (view, &info)) {
g_string_truncate (buffer, buffer->len-1);
continue;
}
g_string_append(buffer,info.cact);
}
return 1;
}
/* map search result positions to offsets in text */
void
view_matchs_to_offsets (WView *view, offset_type start, offset_type end,
size_t match_start, size_t match_end,
offset_type *search_start, offset_type *search_end)
{
struct read_info info;
size_t c = 0;
(*search_start) = INVALID_OFFSET;
(*search_end) = INVALID_OFFSET;
view_read_start (view, &info, start);
while ((info.result != -1) && (info.next < end)) {
view_read_continue (view, &info);
if (view_read_test_nroff_back (view, &info)) {
c-= 1;
continue;
}
if ((c == match_start) && (*search_start == INVALID_OFFSET))
*search_start = info.actual;
if (c == match_end) (*search_end) = info.actual;
c+= !str_iscombiningmark (info.cact) || (c == 0);
}
if ((c == match_start) && (*search_start == INVALID_OFFSET)) *search_start = info.next;
if (c == match_end) (*search_end) = info.next;
}
/* we have set view->search_start and view->search_end and must set
* view->dpy_text_column, view->first_showed_line and view->dpy_start
* try to displaye maximum of match */
void
view_moveto_match (WView *view)
{
const screen_dimen height = view->data_area.height;
const screen_dimen height3 = height / 3;
const screen_dimen width = view->data_area.width;
struct cache_line *line;
struct cache_line *line_end, *line_start;
struct cache_line *t;
int start_off = -1;
int end_off = -1;
int off = 0;
line = view_get_first_showed_line (view);
if (view->text_wrap_mode) {
if (line->start > view->search_start) {
if (line->start <= view->search_start && line->end > view->search_start)
start_off = 0;
if (line->start <= view->search_end && line->end >= view->search_end)
end_off = 0;
t = view_get_previous_line (view, line);
while ((t != NULL) && ((start_off == -1) || (end_off == -1))) {
line = t;
t = view_get_previous_line (view, line);
off++;
if (line->start <= view->search_start && line->end > view->search_start)
start_off = off;
if (line->start <= view->search_end && line->end >= view->search_end)
end_off = off;
}
line = view_get_first_showed_line (view);
off = (start_off - end_off < height - height3) ? start_off + height3: end_off;
for (;off >= 0 && line->start > 0; off--)
line = view_get_previous_line (view, line);
} else {
/* start_off, end_off - how many cache_lines far are
* view->search_start, end from line */
if (line->start <= view->search_start && line->end > view->search_start)
start_off = 0;
if (line->start <= view->search_end && line->end >= view->search_end)
end_off = 0;
t = view_get_next_line (view, line);
while ((t != NULL) && ((start_off == -1) || (end_off == -1))) {
line = t;
t = view_get_next_line (view, line);
off++;
if (line->start <= view->search_start && line->end > view->search_start)
start_off = off;
if (line->start <= view->search_end && line->end >= view->search_end)
end_off = off;
}
line = view_get_first_showed_line (view);
// if view->search_end is farther then screen heigth */
if (end_off >= height) {
off = (end_off - start_off < height - height3) ? end_off - height + height3: start_off;
for (;off >= 0; off--)
line = view_get_next_line (view, line);
}
}
} else {
/* first part similar like in wrap mode,only wokrs with whole lines */
line = view_get_first_showed_line (view);
line = view_get_start_of_whole_line (view, line);
if (line->start > view->search_start) {
line_start = view_get_start_of_whole_line (view, line);
if (line_start->start <= view->search_start && line->end > view->search_start)
start_off = 0;
if (line_start->start <= view->search_end && line->end >= view->search_end)
end_off = 0;
t = view_get_previous_whole_line (view, line_start);
while ((t != NULL) && ((start_off == -1) || (end_off == -1))) {
line = t;
line_start = view_get_start_of_whole_line (view, line);
t = view_get_previous_whole_line (view, line_start);
off++;
if (line_start->start <= view->search_start && line->end > view->search_start)
start_off = off;
if (line_start->start <= view->search_end && line->end >= view->search_end)
end_off = off;
}
line = view_get_first_showed_line (view);
line = view_get_start_of_whole_line (view, line);
off = (start_off - end_off < height - height3) ? start_off + height3: end_off;
for (;off >= 0 && line->start > 0; off--) {
line = view_get_previous_whole_line (view, line);
line = view_get_start_of_whole_line (view, line);
}
} else {
line_end = view_get_end_of_whole_line (view, line);
if (line->start <= view->search_start && line_end->end > view->search_start)
start_off = 0;
if (line->start <= view->search_end && line_end->end >= view->search_end)
end_off = 0;
t = view_get_next_whole_line (view, line_end);
while ((t != NULL) && ((start_off == -1) || (end_off == -1))) {
line = t;
line_end = view_get_end_of_whole_line (view, line);
t = view_get_next_whole_line (view, line_end);
off++;
if (line->start <= view->search_start && line_end->end > view->search_start)
start_off = off;
if (line->start <= view->search_end && line_end->end >= view->search_end)
end_off = off;
}
line = view_get_first_showed_line (view);
line = view_get_start_of_whole_line (view, line);
if (end_off >= height) {
off = (end_off - start_off < height - height3) ? end_off - height + height3: start_off;
for (;off >= 0; off--)
line = view_get_next_whole_line (view, line);
}
}
/*now line point to begin of line, that we want show*/
t = view_offset_to_line_from (view, view->search_start, line);
start_off = view_offset_to_column (view, t, view->search_start);
t = view_offset_to_line_from (view, view->search_end, line);
end_off = view_offset_to_column (view, t, view->search_end);
if (end_off - start_off > width) end_off = start_off + width;
if (view->dpy_text_column > start_off) {
view->dpy_text_column = start_off;
} else {
if (view->dpy_text_column + width < end_off) {
view->dpy_text_column = end_off - width;
}
}
}
view_set_first_showed (view, line);
}
static void
search_update_steps (WView *view)
{
offset_type filesize = view_get_filesize (view);
if (filesize == 0)
view->update_steps = 40000;
else /* viewing a data stream, not a file */
view->update_steps = filesize / 100;
/* Do not update the percent display but every 20 ks */
if (view->update_steps < 20000)
view->update_steps = 20000;
}
static void
view_search (WView *view, char *text,
int (*search) (WView *, char *, char *, int, size_t *, size_t *))
{
GString *buffer;
offset_type search_start;
int search_status;
Dlg_head *d = 0;
offset_type line_start;
offset_type line_end;
size_t match_start;
size_t match_end;
if (verbose) {
d = create_message (D_NORMAL, _("Search"), _("Searching %s"), text);
mc_refresh ();
}
buffer = g_string_new ("");
search_start = (view->direction != 1) ? view->search_start :
view->search_end;
/* Compute the percent steps */
search_update_steps (view);
view->update_activate = 0;
enable_interrupt_key ();
search_status = -1;
while (1) {
if (search_start >= view->update_activate) {
view->update_activate += view->update_steps;
if (verbose) {
view_percent (view, search_start);
mc_refresh ();
}
if (got_interrupt ())
break;
}
if (!view_get_line_at (view, search_start, buffer, &line_start, &line_end))
break;
search_status = (*search) (view, text, buffer->str, match_normal,
&match_start, &match_end);
if (search_status < 0) {
break;
}
if (search_status == 0) {
if (view->direction == 1)
search_start = line_end;
else {
if (line_start > 0) search_start = line_start - 1;
else break;
}
continue;
}
/* We found the string */
view_matchs_to_offsets (view, line_start, line_end,
match_start, match_end,
&(view->search_start), &(view->search_end));
view_moveto_match (view);
break;
}
disable_interrupt_key ();
if (verbose) {
dlg_run_done (d);
destroy_dlg (d);
}
if (search_status <= 0) {
message (D_NORMAL, _("Search"), _(" Search string not found "));
view->search_end = view->search_start;
}
g_string_free (buffer, TRUE);
}
/* Search buffer (its size is len) in the complete buffer
* returns the position where the block was found or INVALID_OFFSET
* if not found */
static offset_type
block_search (WView *view, const char *buffer, int len)
{
int direction = view->direction;
const char *d = buffer;
char b;
offset_type e;
enable_interrupt_key ();
if (direction == 1)
e = view->search_start + ((view->search_start != view->search_end) ? 1 : 0);
else
e = view->search_start
- ((view->search_end != view->search_start && view->search_start >= 1) ? 1 : 0);
search_update_steps (view);
view->update_activate = 0;
if (direction == -1) {
for (d += len - 1;; e--) {
if (e <= view->update_activate) {
view->update_activate -= view->update_steps;
if (verbose) {
view_percent (view, e);
mc_refresh ();
}
if (got_interrupt ())
break;
}
b = get_byte (view, e);
if (*d == b) {
if (d == buffer) {
disable_interrupt_key ();
return e;
}
d--;
} else {
e += buffer + len - 1 - d;
d = buffer + len - 1;
}
if (e == 0)
break;
}
} else {
while (get_byte (view, e) != -1) {
if (e >= view->update_activate) {
view->update_activate += view->update_steps;
if (verbose) {
view_percent (view, e);
mc_refresh ();
}
if (got_interrupt ())
break;
}
b = get_byte (view, e++);
if (*d == b) {
d++;
if (d - buffer == len) {
disable_interrupt_key ();
return e - len;
}
} else {
e -= d - buffer;
d = buffer;
}
}
}
disable_interrupt_key ();
return INVALID_OFFSET;
}
/*
* Search in the hex mode. Supported input:
* - numbers (oct, dec, hex). Each of them matches one byte.
* - strings in double quotes. Matches exactly without quotes.
*/
static void
hex_search (WView *view, const char *text)
{
char *buffer; /* Parsed search string */
char *cur; /* Current position in it */
int block_len; /* Length of the search string */
offset_type pos; /* Position of the string in the file */
int parse_error = 0;
if (!*text) {
view->search_end = view->search_start;
return;
}
/* buffer will never be longer that text */
buffer = g_new (char, strlen (text));
cur = buffer;
/* First convert the string to a stream of bytes */
while (*text) {
int val;
int ptr;
/* Skip leading spaces */
if (*text == ' ' || *text == '\t') {
text++;
continue;
}
/* %i matches octal, decimal, and hexadecimal numbers */
if (sscanf (text, "%i%n", &val, &ptr) > 0) {
/* Allow signed and unsigned char in the user input */
if (val < -128 || val > 255) {
parse_error = 1;
break;
}
*cur++ = (char) val;
text += ptr;
continue;
}
/* Try quoted string, strip quotes */
if (*text == '"') {
const char *next_quote;
text++;
next_quote = strchr (text, '"');
if (next_quote) {
memcpy (cur, text, next_quote - text);
cur += next_quote - text;
text = next_quote + 1;
continue;
}
/* fall through */
}
parse_error = 1;
break;
}
block_len = cur - buffer;
/* No valid bytes in the user input */
if (block_len <= 0 || parse_error) {
message (D_NORMAL, _("Search"), _("Invalid hex search expression"));
g_free (buffer);
view->search_end = view->search_start;
return;
}
/* Then start the search */
pos = block_search (view, buffer, block_len);
g_free (buffer);
if (pos == INVALID_OFFSET) {
message (D_NORMAL, _("Search"), _(" Search string not found "));
return;
}
view->search_start = pos;
view->search_end = pos + block_len;
/* Set the edit cursor to the search position, left nibble */
view->hex_cursor = view->search_start;
view->hexedit_lownibble = FALSE;
/* Adjust the file offset */
view->dpy_start = pos - pos % view->bytes_per_line;
}
static int
regexp_view_search (WView *view, char *pattern, char *string,
int match_type, size_t *match_start, size_t *match_end)
{
static regex_t r;
static char *old_pattern = NULL;
static int old_type;
regmatch_t pmatch[1];
int i, flags = REG_ICASE;
if (old_pattern == NULL || strcmp (old_pattern, pattern) != 0
|| old_type != match_type) {
if (old_pattern != NULL) {
regfree (&r);
g_free (old_pattern);
old_pattern = 0;
}
for (i = 0; pattern[i] != '\0'; i++) {
if (isupper ((unsigned char) pattern[i])) {
flags = 0;
break;
}
}
flags |= REG_EXTENDED;
if (regcomp (&r, pattern, flags)) {
message (D_ERROR, MSG_ERROR, _(" Invalid regular expression "));
return -1;
}
old_pattern = g_strdup (pattern);
old_type = match_type;
}
if (regexec (&r, string, 1, pmatch, 0) != 0)
return 0;
i = str_length (string);
(*match_start) = i - str_length (string + pmatch[0].rm_so);
(*match_end) = i - str_length (string + pmatch[0].rm_eo);
return 1;
}
static void
do_regexp_search (WView *view)
{
view_search (view, view->search_exp, regexp_view_search);
/* Had a refresh here */
view->dirty++;
view_update (view);
}
static void
do_normal_search (WView *view)
{
if (view->hex_mode)
hex_search (view, view->search_exp);
else {
char *needle = str_create_search_needle (view->search_exp, 0);
view_search (view, needle, icase_search_p);
str_release_search_needle (needle, 0);
}
/* Had a refresh here */
view->dirty++;
view_update (view);
}
/* {{{ User-definable commands }}} */
/*
The functions in this section can be bound to hotkeys. They are all
of the same type (taking a pointer to WView as parameter and
returning void). TODO: In the not-too-distant future, these commands
will become fully configurable, like they already are in the
internal editor. By convention, all the function names end in
"_cmd".
*/
static void
view_help_cmd (void)
{
interactive_display (NULL, "[Internal File Viewer]");
}
/* Toggle between hexview and hexedit mode */
static void
view_toggle_hexedit_mode_cmd (WView *view)
{
view_toggle_hexedit_mode (view);
view_update (view);
}
/* Toggle between wrapped and unwrapped view */
static void
view_toggle_wrap_mode_cmd (WView *view)
{
view_toggle_wrap_mode (view);
view_update (view);
}
/* Toggle between hex view and text view */
static void
view_toggle_hex_mode_cmd (WView *view)
{
view_toggle_hex_mode (view);
view_update (view);
}
static void
view_moveto_line_cmd (WView *view)
{
char *answer, *answer_end, prompt[BUF_SMALL];
struct cache_line *line;
offset_type row;
line = view_get_first_showed_line (view);
row = line->number + 1;
g_snprintf (prompt, sizeof (prompt),
_(" The current line number is %d.\n"
" Enter the new line number:"), (int) (line + 1));
answer = input_dialog (_(" Goto line "), prompt, MC_HISTORY_VIEW_GOTO_LINE, "");
if (answer != NULL && answer[0] != '\0') {
errno = 0;
row = strtoul (answer, &answer_end, 10);
if (*answer_end == '\0' && errno == 0 && row >= 1)
view_moveto (view, row - 1, 0);
}
g_free (answer);
view->dirty++;
view_update (view);
}
static void
view_moveto_addr_cmd (WView *view)
{
char *line, *error, prompt[BUF_SMALL];
offset_type addr;
g_snprintf (prompt, sizeof (prompt),
_(" The current address is 0x%lx.\n"
" Enter the new address:"), view->hex_cursor);
line = input_dialog (_(" Goto Address "), prompt, MC_HISTORY_VIEW_GOTO_ADDR, "");
if (line != NULL) {
if (*line != '\0') {
addr = strtoul (line, &error, 0);
if ((*error == '\0') && get_byte (view, addr) != -1) {
view_moveto_offset (view, addr);
} else {
message (D_ERROR, _("Warning"), _(" Invalid address "));
}
}
g_free (line);
}
view->dirty++;
view_update (view);
}
static void
view_hexedit_save_changes_cmd (WView *view)
{
(void) view_hexedit_save_changes (view);
}
/* {{{ Searching }}} */
static void
regexp_search (WView *view, int direction)
{
const char *defval;
char *regexp;
static char *last_regexp;
defval = (last_regexp != NULL ? last_regexp : "");
regexp = input_dialog (_("Search"), _(" Enter regexp:"), MC_HISTORY_VIEW_SEARCH_REGEX, defval);
if (regexp == NULL || regexp[0] == '\0') {
g_free (regexp);
return;
}
g_free (last_regexp);
view->search_exp = last_regexp = regexp;
view->direction = direction;
do_regexp_search (view);
view->last_search = do_regexp_search;
}
/* {{{ User-definable commands }}} */
static void
view_regexp_search_cmd (WView *view)
{
regexp_search (view, 1);
}
/* Both views */
static void
view_normal_search_cmd (WView *view)
{
char *defval, *exp = NULL;
static char *last_search_string;
enum {
SEARCH_DLG_HEIGHT = 8,
SEARCH_DLG_WIDTH = 58
};
static int replace_backwards;
int treplace_backwards = replace_backwards;
static QuickWidget quick_widgets[] = {
{quick_button, 6, 10, 5, SEARCH_DLG_HEIGHT, N_("&Cancel"), 0,
B_CANCEL,
0, 0, NULL},
{quick_button, 2, 10, 5, SEARCH_DLG_HEIGHT, N_("&OK"), 0, B_ENTER,
0, 0, NULL},
{quick_checkbox, 3, SEARCH_DLG_WIDTH, 4, SEARCH_DLG_HEIGHT,
N_("&Backwards"), 0, 0,
0, 0, NULL},
{quick_input, 3, SEARCH_DLG_WIDTH, 3, SEARCH_DLG_HEIGHT, "", 52, 0,
0, 0, N_("Search")},
{quick_label, 2, SEARCH_DLG_WIDTH, 2, SEARCH_DLG_HEIGHT,
N_(" Enter search string:"), 0, 0,
0, 0, 0},
NULL_QuickWidget
};
static QuickDialog Quick_input = {
SEARCH_DLG_WIDTH, SEARCH_DLG_HEIGHT, -1, 0, N_("Search"),
"[Input Line Keys]", quick_widgets, 0
};
defval = g_strdup (last_search_string != NULL ? last_search_string : "");
convert_to_display (defval);
quick_widgets[2].result = &treplace_backwards;
quick_widgets[3].str_result = &exp;
quick_widgets[3].text = defval;
if (quick_dialog (&Quick_input) == B_CANCEL)
goto cleanup;
replace_backwards = treplace_backwards;
if (exp == NULL || exp[0] == '\0')
goto cleanup;
convert_from_input (exp);
g_free (last_search_string);
view->search_exp = last_search_string = exp;
exp = NULL;
view->direction = replace_backwards ? -1 : 1;
do_normal_search (view);
view->last_search = do_normal_search;
cleanup:
g_free (exp);
g_free (defval);
}
static void
view_toggle_magic_mode_cmd (WView *view)
{
view_toggle_magic_mode (view);
view_update (view);
}
static void
view_toggle_nroff_mode_cmd (WView *view)
{
view_toggle_nroff_mode (view);
view_update (view);
}
static void
view_quit_cmd (WView *view)
{
if (view_ok_to_quit (view))
dlg_stop (view->widget.parent);
}
/* {{{ Miscellaneous functions }}} */
/* Define labels and handlers for functional keys */
static void
view_labels (WView *view)
{
const char *text;
Dlg_head *h = view->widget.parent;
buttonbar_set_label (h, 1, Q_("ButtonBar|Help"), view_help_cmd);
my_define (h, 10, Q_("ButtonBar|Quit"), view_quit_cmd, view);
text = view->hex_mode ? "ButtonBar|Ascii" : "ButtonBar|Hex";
my_define (h, 4, Q_(text), view_toggle_hex_mode_cmd, view);
text = view->hex_mode ?"ButtonBar|Goto": "ButtonBar|Line";
my_define (h, 5, Q_(text),
view->hex_mode ? view_moveto_addr_cmd : view_moveto_line_cmd, view);
if (view->hex_mode) {
if (view->hexedit_mode) {
my_define (h, 2, Q_("ButtonBar|View"),
view_toggle_hexedit_mode_cmd, view);
} else if (view->datasource == DS_FILE) {
my_define (h, 2, Q_("ButtonBar|Edit"),
view_toggle_hexedit_mode_cmd, view);
} else {
buttonbar_clear_label (h, 2);
}
my_define (h, 6, Q_("ButtonBar|Save"),
view_hexedit_save_changes_cmd, view);
} else {
text = view->text_wrap_mode ? "ButtonBar|UnWrap" : "ButtonBar|Wrap";
my_define (h, 2, Q_(text), view_toggle_wrap_mode_cmd, view);
my_define (h, 6, Q_("ButtonBar|RxSrch"),
view_regexp_search_cmd, view);
}
text = view->hex_mode ? "ButtonBar|HxSrch" : "ButtonBar|Search";
my_define (h, 7, Q_(text), view_normal_search_cmd, view);
text = view->magic_mode ? "ButtonBar|Raw" : "ButtonBar|Parse";
my_define (h, 8, Q_(text), view_toggle_magic_mode_cmd, view);
/* don't override the key to access the main menu */
if (!view_is_in_panel (view)) {
text = view->text_nroff_mode ? "ButtonBar|Unform" : "ButtonBar|Format";
my_define (h, 9, Q_(text), view_toggle_nroff_mode_cmd, view);
my_define (h, 3, Q_("ButtonBar|Quit"), view_quit_cmd, view);
}
}
/* {{{ Event handling }}} */
/* Check for left and right arrows, possibly with modifiers */
static cb_ret_t
check_left_right_keys (WView *view, int c)
{
if (c == KEY_LEFT) {
view_move_left (view, 1);
return MSG_HANDLED;
}
if (c == KEY_RIGHT) {
view_move_right (view, 1);
return MSG_HANDLED;
}
/* Ctrl with arrows moves by 10 postions in the unwrap mode */
if (view->hex_mode || view->text_wrap_mode)
return MSG_NOT_HANDLED;
if (c == (KEY_M_CTRL | KEY_LEFT)) {
if (view->dpy_text_column >= 10)
view->dpy_text_column -= 10;
else
view->dpy_text_column = 0;
view->dirty++;
return MSG_HANDLED;
}
if (c == (KEY_M_CTRL | KEY_RIGHT)) {
if (view->dpy_text_column <= OFFSETTYPE_MAX - 10)
view->dpy_text_column += 10;
else
view->dpy_text_column = OFFSETTYPE_MAX;
view->dirty++;
return MSG_HANDLED;
}
return MSG_NOT_HANDLED;
}
/* {{{ User-definable commands }}} */
static void
view_continue_search_cmd (WView *view)
{
if (view->last_search) {
view->last_search (view);
} else {
/* if not... then ask for an expression */
view_normal_search_cmd (view);
}
}
static void
view_toggle_ruler_cmd (WView *view)
{
static const enum ruler_type next[3] = {
RULER_TOP,
RULER_BOTTOM,
RULER_NONE
};
assert ((size_t) ruler < 3);
ruler = next[(size_t) ruler];
view->dirty++;
}
/* {{{ Event handling }}} */
static void view_cmk_move_up (void *w, int n) {
view_move_up ((WView *) w, n);
}
static void view_cmk_move_down (void *w, int n) {
view_move_down ((WView *) w, n);
}
static void view_cmk_moveto_top (void *w, int n) {
(void) &n;
view_moveto_top ((WView *) w);
}
static void view_cmk_moveto_bottom (void *w, int n) {
(void) &n;
view_moveto_bottom ((WView *) w);
}
static void
view_select_encoding (WView *view)
{
char *enc = NULL;
GIConv conv;
struct cache_line *line;
#ifdef HAVE_CHARSET
do_select_codepage ();
enc = g_strdup( get_codepage_id ( source_codepage ) );
#endif
if ( enc ) {
conv = str_crt_conv_from (enc);
if (conv != INVALID_CONV) {
if (view->converter != str_cnv_from_term)
str_close_conv (view->converter);
view->converter = conv;
view_reset_cache_lines (view);
line = view_offset_to_line (view, view->dpy_start);
view_set_first_showed (view, line);
}
g_free(enc);
}
}
/* Both views */
static cb_ret_t
view_handle_key (WView *view, int c)
{
c = convert_from_input_c (c);
if (view->hex_mode) {
switch (c) {
case '\t':
view->hexview_in_text = !view->hexview_in_text;
view->dirty++;
return MSG_HANDLED;
case XCTRL ('a'):
view_moveto_bol (view);
view->dirty++;
return MSG_HANDLED;
case XCTRL ('b'):
view_move_left (view, 1);
return MSG_HANDLED;
case XCTRL ('e'):
view_moveto_eol (view);
return MSG_HANDLED;
case XCTRL ('f'):
view_move_right (view, 1);
return MSG_HANDLED;
}
if (view->hexedit_mode
&& view_handle_editkey (view, c) == MSG_HANDLED)
return MSG_HANDLED;
}
if (check_left_right_keys (view, c))
return MSG_HANDLED;
if (check_movement_keys (c, view->data_area.height + 1, view,
view_cmk_move_up, view_cmk_move_down,
view_cmk_moveto_top, view_cmk_moveto_bottom))
return MSG_HANDLED;
switch (c) {
case '?':
regexp_search (view, -1);
return MSG_HANDLED;
case '/':
regexp_search (view, 1);
return MSG_HANDLED;
/* Continue search */
case XCTRL ('r'):
case XCTRL ('s'):
case 'n':
case KEY_F (17):
view_continue_search_cmd (view);
return MSG_HANDLED;
/* toggle ruler */
case ALT ('r'):
view_toggle_ruler_cmd (view);
return MSG_HANDLED;
case 'h':
view_move_left (view, 1);
return MSG_HANDLED;
case 'j':
case '\n':
case 'e':
view_move_down (view, 1);
return MSG_HANDLED;
case 'd':
view_move_down (view, (view->data_area.height + 1) / 2);
return MSG_HANDLED;
case 'u':
view_move_up (view, (view->data_area.height + 1) / 2);
return MSG_HANDLED;
case 'k':
case 'y':
view_move_up (view, 1);
return MSG_HANDLED;
case 'l':
view_move_right (view, 1);
return MSG_HANDLED;
case ' ':
case 'f':
view_move_down (view, view->data_area.height);
return MSG_HANDLED;
case XCTRL ('o'):
view_other_cmd ();
return MSG_HANDLED;
/* Unlike Ctrl-O, run a new shell if the subshell is not running. */
case '!':
exec_shell ();
return MSG_HANDLED;
case 'b':
view_move_up (view, view->data_area.height);
return MSG_HANDLED;
case KEY_IC:
view_move_up (view, 2);
return MSG_HANDLED;
case KEY_DC:
view_move_down (view, 2);
return MSG_HANDLED;
case 'm':
view->marks[view->marker] = view->dpy_start;
return MSG_HANDLED;
case 'r':
view->dpy_start = view->marks[view->marker];
view->dirty++;
return MSG_HANDLED;
/* Use to indicate parent that we want to see the next/previous file */
/* Does not work in panel mode */
case XCTRL ('f'):
case XCTRL ('b'):
if (!view_is_in_panel (view))
view->move_dir = c == XCTRL ('f') ? 1 : -1;
/* FALLTHROUGH */
case 'q':
case XCTRL ('g'):
case ESC_CHAR:
if (view_ok_to_quit (view))
view->want_to_quit = TRUE;
return MSG_HANDLED;
case XCTRL ('t'):
view_select_encoding (view);
view->dirty++;
view_update (view);
return MSG_HANDLED;
#ifdef MC_ENABLE_DEBUGGING_CODE
case 't': /* mnemonic: "test" */
view_ccache_dump (view);
return MSG_HANDLED;
#endif
}
if (c >= '0' && c <= '9')
view->marker = c - '0';
/* Key not used */
return MSG_NOT_HANDLED;
}
/* Both views */
static int
view_event (WView *view, Gpm_Event *event, int *result)
{
screen_dimen y, x;
*result = MOU_NORMAL;
/* We are not interested in the release events */
if (!(event->type & (GPM_DOWN | GPM_DRAG)))
return 0;
/* Wheel events */
if ((event->buttons & GPM_B_UP) && (event->type & GPM_DOWN)) {
view_move_up (view, 2);
return 1;
}
if ((event->buttons & GPM_B_DOWN) && (event->type & GPM_DOWN)) {
view_move_down (view, 2);
return 1;
}
x = event->x;
y = event->y;
/* Scrolling left and right */
if (!view->text_wrap_mode) {
if (x < view->data_area.width * 1/4) {
view_move_left (view, 1);
goto processed;
} else if (x < view->data_area.width * 3/4) {
/* ignore the click */
} else {
view_move_right (view, 1);
goto processed;
}
}
/* Scrolling up and down */
if (y < view->data_area.top + view->data_area.height * 1/3) {
if (mouse_move_pages_viewer)
view_move_up (view, view->data_area.height / 2);
else
view_move_up (view, 1);
goto processed;
} else if (y < view->data_area.top + view->data_area.height * 2/3) {
/* ignore the click */
} else {
if (mouse_move_pages_viewer)
view_move_down (view, view->data_area.height / 2);
else
view_move_down (view, 1);
goto processed;
}
return 0;
processed:
*result = MOU_REPEAT;
return 1;
}
/* Real view only */
static int
real_view_event (Gpm_Event *event, void *x)
{
WView *view = (WView *) x;
int result;
if (view_event (view, event, &result))
view_update (view);
return result;
}
static void
view_adjust_size (Dlg_head *h)
{
WView *view;
WButtonBar *bar;
/* Look up the viewer and the buttonbar, we assume only two widgets here */
view = (WView *) find_widget_type (h, view_callback);
bar = find_buttonbar (h);
widget_set_size (&view->widget, 0, 0, LINES - 1, COLS);
widget_set_size ((Widget *) bar, LINES - 1, 0, 1, COLS);
view_compute_areas (view);
view_update_bytes_per_line (view);
}
/* Callback for the view dialog */
static cb_ret_t
view_dialog_callback (Dlg_head *h, dlg_msg_t msg, int parm)
{
switch (msg) {
case DLG_RESIZE:
view_adjust_size (h);
return MSG_HANDLED;
default:
return default_dlg_callback (h, msg, parm);
}
}
/* {{{ External interface }}} */
/* Real view only */
int
mc_internal_viewer (const char *command, const char *file,
int *move_dir_p, int start_line)
{
gboolean succeeded;
WView *wview;
WButtonBar *bar;
Dlg_head *view_dlg;
/* Create dialog and widgets, put them on the dialog */
view_dlg =
create_dlg (0, 0, LINES, COLS, NULL, view_dialog_callback,
"[Internal File Viewer]", NULL, DLG_WANT_TAB);
wview = view_new (0, 0, COLS, LINES - 1, 0);
bar = buttonbar_new (1);
add_widget (view_dlg, bar);
add_widget (view_dlg, wview);
succeeded = view_load (wview, command, file, start_line);
if (succeeded) {
run_dlg (view_dlg);
if (move_dir_p)
*move_dir_p = wview->move_dir;
} else {
if (move_dir_p)
*move_dir_p = 0;
}
destroy_dlg (view_dlg);
return succeeded;
}
/* {{{ Miscellaneous functions }}} */
static void
view_hook (void *v)
{
WView *view = (WView *) v;
WPanel *panel;
/* If the user is busy typing, wait until he finishes to update the
screen */
if (!is_idle ()) {
if (!hook_present (idle_hook, view_hook))
add_hook (&idle_hook, view_hook, v);
return;
}
delete_hook (&idle_hook, view_hook);
if (get_current_type () == view_listing)
panel = current_panel;
else if (get_other_type () == view_listing)
panel = other_panel;
else
return;
view_load (view, 0, panel->dir.list[panel->selected].fname, 0);
display (view);
}
/* {{{ Event handling }}} */
static cb_ret_t
view_callback (Widget *w, widget_msg_t msg, int parm)
{
WView *view = (WView *) w;
cb_ret_t i;
Dlg_head *h = view->widget.parent;
view_compute_areas (view);
view_update_bytes_per_line (view);
switch (msg) {
case WIDGET_INIT:
if (view_is_in_panel (view))
add_hook (&select_file_hook, view_hook, view);
else
view->dpy_bbar_dirty = TRUE;
return MSG_HANDLED;
case WIDGET_DRAW:
display (view);
return MSG_HANDLED;
case WIDGET_CURSOR:
if (view->hex_mode)
view_place_cursor (view);
return MSG_HANDLED;
case WIDGET_KEY:
i = view_handle_key ((WView *) view, parm);
if (view->want_to_quit && !view_is_in_panel (view))
dlg_stop (h);
else {
view_update (view);
}
return i;
case WIDGET_FOCUS:
view->dpy_bbar_dirty = TRUE;
view_update (view);
return MSG_HANDLED;
case WIDGET_DESTROY:
view_done (view);
if (view_is_in_panel (view))
delete_hook (&select_file_hook, view_hook);
return MSG_HANDLED;
default:
return default_proc (msg, parm);
}
}
/* {{{ External interface }}} */
WView *
view_new (int y, int x, int cols, int lines, int is_panel)
{
WView *view = g_new0 (WView, 1);
size_t i;
init_widget (&view->widget, y, x, lines, cols,
view_callback,
real_view_event);
view->filename = NULL;
view->command = NULL;
view_set_datasource_none (view);
view->growbuf_in_use = FALSE;
/* leave the other growbuf fields uninitialized */
view->hex_mode = FALSE;
view->hexedit_mode = FALSE;
view->hexview_in_text = FALSE;
view->text_nroff_mode = FALSE;
view->text_wrap_mode = FALSE;
view->magic_mode = FALSE;
view->hexedit_lownibble = FALSE;
view->dpy_frame_size = is_panel ? 1 : 0;
view->dpy_start = 0;
view->dpy_text_column = 0;
view->dpy_end = 0;
view->hex_cursor = 0;
view->cursor_col = 0;
view->cursor_row = 0;
view->change_list = NULL;
view->converter = str_cnv_from_term;
/* {status,ruler,data}_area are left uninitialized */
view->dirty = 0;
view->dpy_bbar_dirty = TRUE;
view->bytes_per_line = 1;
view->search_start = 0;
view->search_end = 0;
view->search_exp = NULL;
view->direction = 1; /* forward */
view->last_search = 0; /* it's a function */
view->want_to_quit = FALSE;
view->marker = 0;
for (i = 0; i < sizeof(view->marks) / sizeof(view->marks[0]); i++)
view->marks[i] = 0;
view->move_dir = 0;
view->update_steps = 0;
view->update_activate = 0;
if (default_hex_mode)
view_toggle_hex_mode (view);
if (default_nroff_flag)
view_toggle_nroff_mode (view);
if (global_wrap_mode)
view_toggle_wrap_mode (view);
if (default_magic_flag)
view_toggle_magic_mode (view);
return view;
}