ftp.nice.ch/pub/next/unix/network/news/nn.6.4.16.s.tar.gz#/nn/menu.c

This is menu.c in view mode; [Download] [Up]

/*
 *	(c) Copyright 1990, Kim Fabricius Storm.  All rights reserved.
 *
 * 	selection mode menu
 */

#include "config.h"
#include "articles.h"
#include "term.h"
#include "keymap.h"
#include "menu.h"
#include "regexp.h"

import char *news_lib_directory;

export int  echo_prefix_key = 1; /* echo prefix keys */
export int  preview_window = 0;	/* size of preview window */
export int  fmt_linenum    = 1; /* menu line format */
export int  fmt_rptsubj    = 0; /* repeat identical subjects if !0 */
export int  novice	   = 1; /* novice mode -- use extended prompts */
export int  long_menu	   = 0; /* don't put empty lines around menu lines */
export int  delay_redraw   = 0; /* prompt again if :-command clears screen */
export int  slow_mode	   = 0;	/* mark selected articles with *s */
export int  re_layout      = 0; /* Re: format presentation on menus */
export int  collapse_subject = 25; /* collapse long subjects at position */
export int  conf_group_entry = 0; /* ask whether group should be entered */
export int  conf_entry_limit = 0; /* ask only if more than .. unread */
export int  mark_read_skip = 4; /* effect of X command */
export int  mark_read_return = 0; /* effect of Z command */
export int  mark_next_group = 0; /* effect of N command */
export int  show_purpose_mode = 1; /* 0: never, 1: new, 2: always */
export int  read_ret_next_page = 0; /* Z returns to next page */

export int  consolidated_menu = 0; /* show only root articles */
export int  save_closed_mode = 13; /* ask how to save closed subj, dflt all */
export int  auto_select_closed = 1; /* select all in closed subject */
export int  menu_spacing = 0;	    /* number of screen lines per menu line */
export char *counter_delim_left  = "[";
export char *counter_delim_right = "] ";
export int  counter_padding = 5; /* counters are padded to align subjects */

export int  auto_preview_mode = 0; /* preview rather than select */
export int  preview_continuation = 12; /* what to do after preview */
export int  preview_mark_read = 1; /* previewed articles are A_READ */
export int  select_on_sender = 0; /* find command selects on sender */
export int  auto_select_subject = 0; /* auto select articles with same subj. */
export int  auto_read_limit = 0; /* ignore auto_read_mode if less articles */

export char delayed_msg[100] = "";	/* give to msg() after redraw */

export int  flush_typeahead = 0;

import int also_read_articles;
import int merged_menu;
import int case_fold_search;

extern group_completion();

static regexp *regular_expr = NULL;

static int firstl;	/* first menu line */

static article_number firsta;	/* first article on menu (0 based) */
static article_number nexta;	/* first article on next menu */
static int cura;	/* current article */
static int next_cura;	/* article to become cura if >= 0 */
static int numa;	/* no of articles on menu - 1 */
static attr_type last_attr;

#define INTERVAL1	('z' - 'a' + 1)
#define INTERVAL2	('9' - '0' + 1)

char ident[] = "abcdefghijklmnopqrstuvwxyz0123456789";

char attributes[30] = " .,+=#! **"; /* Corresponds to A_XXXX in data.h */

static int menu_length;		/* current no of line on menu */
static int menu_articles;	/* current no of articles on menu */

static struct menu_info {	/* info for each menu line */
    int mi_cura;	/* cura corresponding to this menu line */
    int mi_total;	/* total number of articles with this subject */
    int mi_unread;	/* no of unread articles with this subject */
    int mi_selected;	/* no of selected articles with this subject */
    int mi_art_id;	/* article id (for mark()) */
} menu_info[INTERVAL1+INTERVAL2];

static struct menu_info *art_id_to_mi[INTERVAL1+INTERVAL2];

#define IS_VISIBLE(ah)	(((ah)->flag & (A_CLOSED | A_ROOT_ART)) != A_CLOSED)

prt_replies(level)
{
    int re;

    if (level == 0) return 0;
    re = level & 0x80;
    level &= 0x7f;

    switch (re_layout) {
     case 1:
	if (!re) return 0;
	so_printf(">");
	return 1;
     case 2:
	switch (level) {
	 case 0:
	    return 0;
	 case 1:
	    so_printf(">");
	    return 1;
	 default:
	    so_printf("%d>", level);
	    return level < 10 ? 2 : 3;
	}
     case 3:
	so_printf("Re: ");
	return 4;
     case 4:
	if (level == 0 && re) level++;
	break;
    }

    if (level < 10) {
	so_printf("%-.*s", level, ">>>>>>>>>");
	return level;
    }

    so_printf(">>>%3d >>>>", level);
    return 11;
}

static article_number root_article(root)
register article_number root;
{
    register article_header *ah = articles[root];

    if (ah->flag & A_ROOT_ART)
	return root;

    if (ah->flag & A_CLOSED)	/* only root article is shown on menu */
	return firsta + menu_info[ah->menu_line].mi_cura;

    while (root > 0) {
	if (articles[root]->flag & A_ROOT_ART) break;
	root--;
    }
    return root;
}

static article_number next_root_article(root)
register article_number root;
{
    while (++root < n_articles)
	if (articles[root]->flag & A_ROOT_ART) break;
    return root;
}

static set_root_if_closed()
{
    if (articles[firsta + cura]->flag & A_CLOSED)
	cura = root_article(firsta + cura) - firsta;
}

/*
 *	count info for thread containing article #art (must be on menu!)
 *	returns article number for next root article
 */

static article_number thread_counters(art)
article_number art;
{
    register struct menu_info *mi;
    register article_number n;
    int total, unread, selected, invisible;
    
    if (!(articles[art]->flag & A_CLOSED)) return art + 1;
    
    total = unread = selected = 0;
    n = art = root_article(art);

    while (n < n_articles) {
	if (articles[n]->attr == 0) unread++;
	else if (articles[n]->attr & A_SELECT) selected++; 
	total++;
	if (++n == n_articles) break;
	if (articles[n]->flag & A_ROOT_ART) break;
    }
    unread += selected;

    mi = menu_info + articles[art]->menu_line;
    mi->mi_total = total;
    mi->mi_unread = unread;
    mi->mi_selected = selected;
    return n;
}

static cursor_at_id()
{
    gotoxy(0, firstl + articles[firsta + cura]->menu_line);
    fl; /* place cursor at current article id */
    save_xy();
}

static attr_type closed_attr(mi, cbuf)
register struct menu_info *mi;
char *cbuf;
{
    char sel[10], unr[10];
    attr_type cattr;
    
    if (mi->mi_unread == 0)
	cattr = A_READ;
    else if (mi->mi_total == mi->mi_selected)
	cattr = A_SELECT;
    else if (auto_select_closed == 1 && mi->mi_unread == mi->mi_selected)
	cattr = A_SELECT;
    else if (mi->mi_selected)
	cattr = A_KILL;	/* pseudo flag -> highlight cbuf */
    else
	cattr = 0;

    sel[0] = unr[0] = NUL;
    if (mi->mi_selected && mi->mi_selected < mi->mi_unread)
	sprintf(sel, "%d/", mi->mi_selected);
    if (mi->mi_unread && mi->mi_unread < mi->mi_total)
	sprintf(unr, "%d:", mi->mi_unread);
    sprintf(cbuf, "%s%s%d", sel, unr, mi->mi_total);

    return cattr;
}

static int subj_indent;

static mark()
{
    register article_header *ah;
    register struct menu_info *mi;
    int lno, lnum, lsubj, lname;
    int pad;
    char cbuf[80];
    attr_type cattr;

    ah = articles[firsta + cura];
    if (!IS_VISIBLE(ah)) return;
    
    last_attr = ah->attr;
    if (ah->disp_attr == A_NOT_DISPLAYED) {
	mi = &menu_info[ah->menu_line];
	lno = firstl + ah->menu_line;
	gotoxy(0, lno);
	putchar(ident[mi->mi_art_id]);
	cattr = closed_attr(mi, cbuf);
	goto print_line;
    }

    if (cura < 0 || cura > numa) return;

    lno = firstl + ah->menu_line;

    if (ah->flag & A_CLOSED) {
	struct menu_info old;
	char oldctr[80];
	attr_type oldattr;

	mi = &menu_info[ah->menu_line];
	old = *mi;
	thread_counters(firsta + cura);
	if (old.mi_total == mi->mi_total &&
	    old.mi_selected == mi->mi_selected &&
	    old.mi_unread == mi->mi_unread) return;

	cattr = closed_attr(mi, cbuf);

	if (!slow_mode) goto print_line;
	oldattr = closed_attr(&old, oldctr);
	if (strcmp(cbuf, oldctr)) goto print_line;
	last_attr = cattr;
    }

    if (last_attr == ah->disp_attr) return;

    /* A_AUTO_SELECT will not occur here! */

    if (!slow_mode)
	if (last_attr == A_SELECT) {
	    if ((ah->disp_attr & A_SELECT) == 0) goto print_line;
	} else {
	    if (ah->disp_attr & A_SELECT) goto print_line;
	}

    gotoxy(1, lno);
    putchar(attributes[ah->attr]);
    goto out;

 print_line:
    /* menu line formats:
                1  3    8 10     20 22        xx
		:  :    :  :      :  :        :
       -1	id name:8 subject
       0	id name           subject     +lines
       1	id name       lines  subject
       2	id  lines  subject
       3	id subject
       4	id   subject  (or as 1 if short subject)
     */

    if ((ah->flag & A_CLOSED) == 0) {
	cattr = ah->attr;
	cbuf[0] = NUL;
    }

    if (fmt_linenum > 4) fmt_linenum = 1;

    if (!slow_mode && (cattr & A_SELECT)) {
	if (so_gotoxy(1, lno, 1) == 0)
	    putchar(attributes[A_SELECT]);
    } else {
    	gotoxy(1, lno);
	putchar(cattr == A_KILL ? ' ' : attributes[cattr]);
    }

    if (ah->lines <    10) lnum = 1; else
    if (ah->lines <   100) lnum = 2; else
    if (ah->lines <  1000) lnum = 3; else
    if (ah->lines < 10000) lnum = 4; else lnum = 5;

    lsubj = Columns - cookie_size - 2; /* ident char + space */

    switch (fmt_linenum) {

     case -1:
	lsubj -= 9;
	so_printf("%-8.8s ", ah->sender);
	goto no_counters;
	
     case 0:
	lsubj -= NAME_LENGTH + 1 + 2 + lnum;  /* name. .subj. +.lines */
	so_printf("%-*s ", NAME_LENGTH, ah->sender);
	break;

     case 4:
	if (ah->subj_length > (lsubj - NAME_LENGTH - 5))
	    if (fmt_rptsubj || lno == firstl || (ah->flag & A_SAME) == 0) {
		so_printf("  ");
		lsubj -= 2;
		break;
	    }
	/* else use layout 1, so fall thru */

     case 1:
	lsubj -= NAME_LENGTH + 5;
	/* name.lines.  .subj (name may be shortened) */
	lname = NAME_LENGTH + 2 - lnum;
	so_printf("%-*.*s ", lname, lname, ah->sender);
	so_printf(ah->lines >= 0 ? "%d  " : "?  ", ah->lines);
	break;

     case 2:
	lsubj -= 6;
	so_printf("%5d ", ah->lines);
	break;

     case 3:
	break;
    }

 print_subj:
    if (cbuf[0]) {
	so_printf("%s", counter_delim_left);
	if (cattr == A_KILL) so_gotoxy(-1, -1, 0);
	so_printf("%s", cbuf);
	if (cattr == A_KILL) so_end();
	so_printf("%s", counter_delim_right);
	pad = strlen(cbuf) + strlen(counter_delim_left) + strlen(counter_delim_right);
	if (cattr == A_KILL) pad += cookie_size * 2;
	lsubj -= pad > subj_indent ? pad : subj_indent;
	pad = subj_indent - pad;
    } else {
	lsubj -= subj_indent;
	pad = subj_indent;
    }
    if (pad > 0) {
	if (pad > 20) pad = 20;
	so_printf("                    "+20-pad);
    }

 no_counters:
    if (!fmt_rptsubj && lno > firstl && ah->flag & A_SAME) {
	if (ah->replies == 0 || prt_replies(ah->replies) == 0)
	    so_printf("-");
    } else {
	lsubj -= prt_replies(ah->replies);
	if (lsubj >= ah->subj_length)
	    so_printf("%s", ah->subject);
	else
	if (collapse_subject < 0)
	    so_printf("%-.*s", lsubj, ah->subject);
	else {
	    if (collapse_subject > 0)
		so_printf("%-.*s", collapse_subject, ah->subject);
	    lsubj -= 2 + collapse_subject;
	    so_printf("<>%s", ah->subject + ah->subj_length - lsubj);
	}
    }

    if (fmt_linenum == 0)
	so_printf(ah->lines >= 0 ? " +%d" : " +?", ah->lines);

    so_end();
    if (ah->flag & A_CLOSED) clrline_noflush();

 out:
    ah->disp_attr = last_attr;
    return;
}

static new_mark(how)
attr_type how;
{
    articles[firsta + cura]->attr = how;
    mark();
}

static toggle()
{
    last_attr = articles[firsta + cura]->attr =
	articles[firsta + cura]->attr & A_SELECT ? 0 : A_SELECT;
}

static do_auto_kill()
{
    register article_number i;
    register article_header *ah, **ahp;
    int any = 0;

    for (i = 0, ahp = articles; i < n_articles; i++, ahp++) {
	ah = *ahp;
	if (auto_select_article(ah, 0)) {
	    ah->attr = A_KILL;
	    any = 1;
	}
    }
    return any;
}

/*
 *	perform auto selections that are not already selected
 *	if article is in range firsta..firsta+numa (incl) mark article
 */

static do_auto_select(re, mode)
regexp *re;
int mode;
{
    register article_number i;
    register article_header *ah, **ahp;
    int count = 0, o_cura;
    extern int kill_file_loaded;

    if (mode == 1 && re == NULL) 
	if (!kill_file_loaded && !init_kill()) return 0;

    o_cura = cura;
    cura = -1;

    for (i = 0, ahp = articles; i < n_articles; i++, ahp++) {
	ah = *ahp;
	if (cura >= 0 && (ah->flag & A_ROOT_ART)) {
	    mark();
	    cura = -1;
	}
	if (re != NULL) {
	    if (!regexec_cf(re, select_on_sender ? ah->sender : ah->subject)) continue;
	} else
	    if (!auto_select_article(ah, mode)) continue;

	count++;
	if (ah->attr & A_SELECT) continue;
	ah->attr = A_SELECT;
	if (firsta <= i && i <= (firsta+numa)) {
	    cura = i - firsta;
	    if ((ah->flag & A_CLOSED) == 0) {
		mark();
		cura = -1;
	    }
	}
    }

    if (cura >= 0) mark();

    if (count)
	msg("Selected %d article%s", count, plural((long)count));
    else
	msg("No selections");
    
    cura = o_cura;
}

static quit_preview(cmd)
int cmd;
{
    int op;

    if ((firsta + cura) >= n_articles) return 1;
    op = preview_continuation;
    if (cmd == MC_PREVIEW_NEXT)	op /= 10;
    op %= 10;
    switch (op) {
     case 0:
	return 1;
     case 1:
	return 0;
     case 2:
	return articles[firsta+cura]->flag & A_ROOT_ART;
    }
    return 0;
}

export long n_selected;
export int show_art_next_invalid;

static count_selected_articles()
{
    register article_number cur;

    n_selected = 0;
    for (cur = 0; cur < n_articles; cur++) {
	if (articles[cur]->attr & A_SELECT) n_selected++;
    }
}

static show_articles()
{
    register article_number cur, next, temp;
    register article_header *ah;
    article_number elim_list[1];
    register int mode;
    int cmd, prev = -1, again;
    attr_type o_attr;
    import int new_read_prompt;

    do {
	for (cur = 0; cur < n_articles; cur++) {
	    if (articles[cur]->attr & A_SELECT) break;
	}

	while (cur < n_articles) {

	    for (next = cur+1; next < n_articles; next++) {
		if (articles[next]->attr & A_SELECT) break;
	    }
	    show_art_next_invalid = 0;

	 show:
	    ah = articles[cur];
	    o_attr = ah->attr;
	    ah->attr = 0;

	    if (new_read_prompt)
		count_selected_articles();

	    mode = 0;
	    if (prev >= 0) mode |= MM_PREVIOUS;
	    if (next == n_articles) mode |= MM_LAST_SELECTED;
	    if ((cur + 1) >= n_articles) mode |= MM_LAST_ARTICLE;
	    if (cur == 0) mode |= MM_FIRST_ARTICLE;

	    cmd = more(ah, mode, 0);

	    switch (cmd) {

	     case MC_DO_KILL:
		if (do_auto_kill()) {
		    elim_list[0] = next;
		    elim_articles(elim_list, 1);
		    cur = elim_list[0];
		    /* if next was n_articles, cur will be 0 */
		    if (cur >= n_articles || cur < 0
			|| (articles[cur]->attr & A_SELECT) == 0)
			cur = n_articles;
		    continue;
		}
		break;

	     case MC_DO_SELECT:
		break;

	     case MC_PREV:
		ah->attr = o_attr;
		if (prev == next) break;

		next = cur; cur = prev; prev = next;
		goto show;

	     case MC_NEXTSUBJ:
		ah->attr = A_READ;
		for (next = cur+1; next < n_articles; next++) {
		    if ((ah = articles[next])->flag & A_ROOT_ART) break;
		    ah->attr = A_READ;
		}
		for (; next < n_articles; next++) {
		    if (articles[next]->attr & A_SELECT) break;
		}
		break;

	     case MC_ALLSUBJ:
		ah->attr = A_READ;
		for (next = cur+1; next < n_articles; next++) {
		    ah = articles[next];
		    if (ah->flag & A_ROOT_ART) break;
		    ah->attr = A_SELECT;
		}
		for (next = cur+1; next < n_articles; next++)
		    if (articles[next]->attr & A_SELECT) break;
		break;

	     case MC_MENU:
		ah->attr = o_attr;
		if (nexta - firsta < n_articles)
		    if ((firsta = cur - 5) < 0) firsta = 0;
		next_cura = cur - firsta;

		return MC_MENU;

	     case MC_NEXT:
		if (ah->attr == 0)	/* Not set by more (sufficient ???) */
		    ah->attr = A_READ;
		break;

	     case MC_BACK_ART:
		ah->attr = o_attr ? o_attr : A_SEEN;
		next = cur - 1;
		break;

	     case MC_FORW_ART:
		ah->attr = o_attr ? o_attr : A_SEEN;
		next = cur + 1;
		break;

	     case MC_NEXTGROUP:
	     case MC_REENTER_GROUP:
	     case MC_QUIT:
		ah->attr = o_attr;
		return cmd;

	     case MC_READGROUP:
		for (cur = 0; cur < n_articles; cur++) {
		    ah = articles[cur];
		    if (ah->attr == 0 || (ah->attr & A_SELECT))
			ah->attr = A_READ;
		}
		return MC_NEXTGROUP;
	    }

	    if (show_art_next_invalid)
		for (next = cur+1; next < n_articles; next++) {
		    if (articles[next]->attr & A_SELECT) break;
		}
	    prev = cur; cur = next;
	}

	for (cur = 0; cur < n_articles; cur++)
	    if (articles[cur]->attr & A_SELECT) break;
	if (cur < n_articles) continue;

	again = 0;
	for (cur = 0; cur < n_articles; cur++) {
	    ah = articles[cur];
	    if (ah->attr == A_LEAVE) {
		if (again == 0) {
		    prompt("Show left over articles again now? ");
		    if (yes(0) <= 0) break;
		}
		ah->attr = A_SELECT;
		again++;
	    }
	}

	if (again > 1)
	    sprintf(delayed_msg, "Showing %ld articles again", again);
    } while (again);

    return MC_READGROUP;
}

static int article_id;
static int cur_key;
static int is_k_select;	/* set when K_ARTICLE_ID was really K_SELECT */

static int get_k_cmd_1()
{
    extern int any_message;
    register int c, map;
    int *key_map = menu_key_map;

    if (flush_typeahead) flush_input();

 loop:

    article_id = -1;

    if ((c = get_c()) & GETC_COMMAND) {
	cur_key = K_interrupt;
	map = c & ~GETC_COMMAND;
    } else {
	cur_key = c;
	map = key_map[c];
    }
    if (s_hangup) map = K_QUIT;

    if (map & K_PREFIX_KEY) {
	key_map = keymaps[map & ~K_PREFIX_KEY].km_map;
	if (echo_prefix_key) msg("%s", key_name(cur_key));
	goto loop;
    }

    is_k_select = 0;
    if (map == K_SELECT) {
	map = K_ARTICLE_ID;
	article_id = cura;
	is_k_select = 1;
    } else
    if (map & K_ARTICLE_ID) {
	article_id = map & ~K_ARTICLE_ID;
	map = K_ARTICLE_ID;

	if (article_id < 0 || article_id > menu_articles) {
	    ding();
	    goto loop;
	}
	article_id = art_id_to_mi[article_id]->mi_cura;
    }

    if (any_message && map != K_LAST_MESSAGE) clrmsg(-1);
    return map;
}

static int get_k_cmd()
{
    register int map;

    map = get_k_cmd_1();
    if (map & K_MACRO)
	map = orig_menu_map[cur_key];
    return map;
}


char *pct(start, end, first, last)
long start, end, first, last;
{
    long n = end - start;
    static char buf[16];
    char *fmt;

    if (first <= start || n <= 0)
	if (last >= end || n <= 0)
	    return "All";
	else
	    fmt = "Top %d%%";
    else
	if (last >= end)
	    return "Bot";
	else
	    fmt = "%d%%";

    sprintf(buf, fmt, ((last - start) * 100)/n);
    return buf;
}

static repl_attr(first, last, old, new, update)
register article_number first, last;
register attr_type old, new;
int update;
{
    int any;
    register article_header *ah;
    article_number ocura = cura;

    if (new == old) return 0;
    if (new == A_KILL) update = 0;

    any = 0;
    cura = -1;
    while (first < last) {
	ah = articles[first];
	if (cura >= 0 && ah->flag & A_ROOT_ART) {
	    set_root_if_closed();
	    mark();
	    cura = -1;
	}
	if (old == A_KILL || ah->attr == old) {
	    ah->attr = new;
	    if (update && first >= firsta && first < nexta) {
		cura = first - firsta;
		if ((ah->flag & A_CLOSED) == 0) {
		    mark();
		    cura = -1;
		}
	    }
	    any = 1;
	}

	first++;
    }

    if (cura >= 0) {
	set_root_if_closed();
	mark();
    }
    cura = update ? last - firsta : ocura;
    return any;
}

static repl_attr_subject(old, new, update)
attr_type old, new;
int update;
{
    return repl_attr(root_article(firsta+cura), next_root_article(firsta+cura),
		     old, new, update);
}

static repl_attr_all(old, new, update)
attr_type old, new;
int update;
{
    return repl_attr((article_number)0, n_articles, old, new, update);
}

static get_purpose(purpose)
char *purpose;
{
    FILE *f, *open_purpose_file();
    char line[256], *group;
    register char *cp, *pp;
    register int len;

    if (current_group == NULL) return;
    if ((current_group->master_flag & M_VALID) == 0) return;
    if (current_group->group_flag & G_FAKED) return;
    
    if ((f = open_purpose_file()) == NULL) return;

    group = current_group->group_name;
    len = current_group->group_name_length;

    while (fgets(line, 256, f) != NULL) {
	if (!isascii(line[len]) || !isspace(line[len])) continue;
	if (strncmp(line, group, len)) continue;
	cp = line + len;
	while (*cp && isspace(*cp)) cp++;
	for (pp = purpose, len = 76; --len >= 0 && *cp && *cp != NL; )
	    *pp++ = *cp++;
	*pp = NUL;
    }
}

/*
 * bypass_consolidation may be set to temporarily overrule the global
 * consolidated_menu variable:
 *	1:	don't consolidate	(e.g. for G=...  )
 *	2:	do consolidate
 *	3:	use consolidated_mode
 */

export int bypass_consolidation = 0;
static int cur_bypass = 0;	/* current bypass status (see below) */

static do_consolidation()
{
    int consolidate;

    switch (bypass_consolidation) {
     case 0:
	break;
     case 1:
	cur_bypass = -1;	/* no consolidation */
	break;
     case 2:
	cur_bypass = 1;		/* force consolidation */
	break;
     case 3:
	cur_bypass = 0;		/* reset bypass to use consolidated_menu */
	break;
    }
    bypass_consolidation = 0;

    if (cur_bypass)
	consolidate = cur_bypass > 0;
    else
	consolidate = consolidated_menu;

    if (consolidate)
	for (nexta = 0; nexta < n_articles; nexta++)
	    articles[nexta]->flag |= A_CLOSED;
    else
	for (nexta = 0; nexta < n_articles; nexta++)
	    articles[nexta]->flag &= ~A_CLOSED;

    return consolidate;
}

menu(print_header)
fct_type print_header;
{
    register 		k_cmd, cur_k_cmd;
    register		article_header *ah;
    register struct menu_info *mi;
    int			consolidate, o_bypass;
    int			last_k_cmd;
    int 		menu_cmd, temp;
    int 		save_selected;
    article_number	last_save;
    attr_type		orig_attr, junk_attr;
    int			doing_unshar, did_unshar, junk_prompt;
    char 		*fname, *savemode, *init_save();
    int 		maxa;	/* max no of articles per menu page */
    article_number	o_firsta, temp1, temp2;
    int			o_mode;	/* for recursive calls */
    static		menu_level = 0;
    char		purpose[80], pr_fmt[60];
    extern int 		enable_stop, file_completion();
    extern int		alt_cmd_key, in_menu_mode;
    article_number	elim_list[3];
    int			entry_check;
    int			auto_read;
    long		o_selected;

#define	menu_return(cmd) \
    { menu_cmd = (cmd); goto menu_exit; }

    flush_input();

    o_firsta = firsta;
    o_mode = in_menu_mode;
    o_selected = n_selected;

    in_menu_mode = 1;

    menu_level++;

    if (menu_level == 1) {
	if ((current_group->group_flag & G_COUNTED)
	    && n_articles != current_group->unread_count) {
	    add_unread(current_group, -1);
	    current_group->unread_count = n_articles;
	    add_unread(current_group, 0);
	}
	entry_check = conf_group_entry && n_articles > conf_entry_limit;
	auto_read = auto_read_limit < 0 || n_articles <= auto_read_limit;
    } else {
	entry_check = 0;
	auto_read = 0;
    }

    sprintf(pr_fmt,
	    menu_level == 1 ?
	      "\1\2-- SELECT %s-----%%s-----\1" :
	      "\1\2-- SELECT %s-----%%s-----<%s%d>--\1",
	      novice ? "-- help:? " : "",
	      novice ? "level " : "",
	      menu_level);

    purpose[0] = NUL;
    if (!merged_menu)
	switch (show_purpose_mode) {
	 case 0: break;
	 case 1: if ((current_group->group_flag & G_NEW) == 0) break;
	 case 2: get_purpose(purpose);
	         if (purpose[0]) strcpy(delayed_msg, purpose);
	}

    o_bypass = cur_bypass;
    cur_bypass = 0;
    
    consolidate = do_consolidation();

    firsta = 0;
    while (firsta < n_articles && articles[firsta]->attr == A_SEEN)
	firsta++;

    if (firsta == n_articles) firsta = 0;

    next_cura = -1;
    cur_k_cmd = K_UNBOUND;

#ifdef HAVE_JOBCONTROL
#define	REDRAW_CHECK	if (s_redraw) goto do_redraw

 do_redraw:
    /* safe to clear here, because we are going to redraw anyway */
    s_redraw = 0;
#else
#define REDRAW_CHECK
#endif

 redraw:
    s_keyboard = 0;

 empty_menu_hack:	/* do: "s_keyboard=1; goto empty_menu_hack;" */
    if (!slow_mode) s_keyboard = 0;

    nexta = firsta;

    clrdisp();

    if (entry_check) {
	prompt_line = firstl = CALL(print_header)();
	prompt("\1Enter?\1 ");
	if ((temp = yes(0)) <= 0) {
	    if (temp < 0) {
		prompt("\1Mark as read?\1 ");
		if ((temp = yes(0)) < 0) menu_return(ME_QUIT);
		if (temp > 0) repl_attr_all(A_KILL, A_READ, 0);
	    }
	    menu_return(ME_NEXT);
	}

	gotoxy(0, firstl);
	clrline();
    }

    if (auto_read) {
	auto_read = entry_check = 0;
	if (repl_attr_all(0, A_AUTO_SELECT, 0)) {
	    k_cmd = K_READ_GROUP_UPDATE;
	    if (purpose[0])
		strcpy(delayed_msg, purpose);
	    else
	    sprintf(delayed_msg, "Entering %s, %ld articles",
		    current_group->group_name, (long)n_articles);
	    goto do_auto_read;
	}
    }

    if (!entry_check)
	firstl = CALL(print_header)();
    entry_check = 0;


    maxa = Lines - preview_window - firstl - 2;
    if (!long_menu) firstl++, maxa -= 2;

    if (maxa > (INTERVAL1 + INTERVAL2))
	maxa = INTERVAL1 + INTERVAL2;

 nextmenu:

    no_raw();
    gotoxy(0, firstl);
    clrpage(firstl);

    if (articles[nexta]->flag & A_CLOSED)
	nexta = root_article(nexta);
    firsta = nexta;
    cura = 0;

    REDRAW_CHECK;

    menu_length = 0;
    menu_articles = 0;
    goto draw_menu;
    
 partial_redraw:
    next_cura = cura;
 partial_redraw_nc:
    nexta = firsta + cura;
    menu_length = articles[nexta]->menu_line;
    menu_articles = menu_info[menu_length].mi_art_id;
    gotoxy(0, firstl + menu_length);
    clrpage(firstl + menu_length);

 draw_menu:
    subj_indent = consolidate ? counter_padding : 0;

    if (!s_keyboard) {
	int first_menu_line = menu_length;

	mi = menu_info + menu_length;
	while (nexta < n_articles) {
	    REDRAW_CHECK;

	    ah = articles[nexta];
	    if (ah->flag & A_HIDE) {
		ah->menu_line = menu_length;	/* just in case.... */
		continue;
	    }
	    if (menu_length > first_menu_line) {
		switch (menu_spacing) {
		 case 0:
		    break;
		 case 1:
		    if ((ah->flag & A_ROOT_ART) == 0) break;
		 case 2:
		    menu_length++;
		    mi++;
		    break;
		}
		if (menu_length >= maxa) break;
	    }
	    
	    ah->menu_line = menu_length;
	    art_id_to_mi[menu_articles] = mi;
	    mi->mi_art_id = menu_articles++;
	    
	    mi->mi_cura = cura = nexta - firsta;
	    if (ah->flag & A_CLOSED) {	/* skip rest of thread */
		nexta = thread_counters(nexta);
		if (nexta == firsta + cura + 1)
		    ah->flag &= ~A_CLOSED;
		else {
		    article_number tmpa = firsta + cura;
		    while (++tmpa < nexta)
			articles[tmpa]->menu_line = menu_length;
		}
	    } else
		nexta++;
	    ah->disp_attr = A_NOT_DISPLAYED;
	    mark();
	    if (++menu_length >= maxa) break;
	    mi++;
	}
    }

    if (menu_length > maxa) menu_length = maxa;

    fl;
    s_keyboard = 0;

    prompt_line = firstl + menu_length;
    if (!long_menu || menu_length < maxa) prompt_line++;

    numa = nexta - firsta - 1;
     if (numa < 0) prompt_line++;

     if (next_cura >= 0) {
	 cura = next_cura;
	 set_root_if_closed();
	 next_cura = -1;
     } else {
	 cura = 0;
	 for (article_id = firsta; cura < numa; article_id++, cura++)
	 {
	     if ((articles[article_id]->attr & A_SELECT) == 0) break;	/*???*/
	     if (!IS_VISIBLE(articles[article_id])) continue;
	 }
	 if (!IS_VISIBLE(articles[article_id]))
	     cura = root_article(article_id) - firsta;
     }

  build_prompt:

     raw();

  Prompt:

     prompt(pr_fmt,
	    pct(0L, (long)(n_articles-1), (long)firsta, (long)(firsta+numa)));

     if (delayed_msg[0] != NUL) {
	 msg(delayed_msg);
	 delayed_msg[0] = NUL;
     }

  same_prompt:

    if (cura < 0 || cura > numa) cura = 0;

    if (!IS_VISIBLE(articles[firsta+cura])) {
	cura = next_root_article(firsta + cura) - firsta;
	if (cura > numa) cura = 0;
    }

     if (numa >= 0) {
	 cursor_at_id();
     }

     last_k_cmd = cur_k_cmd;

  get_next_k_cmd:
     k_cmd = get_k_cmd_1();

  alt_key:
    if (k_cmd & K_MACRO) {
	m_invoke(k_cmd & ~K_MACRO);
	goto get_next_k_cmd;
    }

     switch (cur_k_cmd = k_cmd) {

      case K_UNBOUND:
	 ding();
	 flush_input();
      case K_INVALID:
	 goto same_prompt;

      case K_REDRAW:
	 next_cura = cura;
	 goto redraw;

      case K_LAST_MESSAGE:
	 msg((char *)NULL);
	 goto same_prompt;

      case K_HELP:
	 if (numa < 0)  goto nextmenu;	/* give specific help here */
	 display_help("menu");
	 goto redraw;

      case K_SHELL:
	 if (group_file_name) *group_file_name = NUL;
	 if (shell_escape()) goto redraw;
	 goto Prompt;

      case K_VERSION:
	 prompt(P_VERSION);
	 goto same_prompt;

      case K_EXTENDED_CMD:
	 temp = consolidated_menu;
	 switch (alt_command()) {

	  case AC_UNCHANGED:
	     goto same_prompt;

	  case AC_QUIT:
	     menu_return( ME_QUIT );

	  case AC_PROMPT:
	     goto Prompt;

	  case AC_REORDER:
	     firsta = 0;
	     consolidate = do_consolidation();
	     goto redraw;

	  case AC_REDRAW:
	     if (temp != consolidated_menu)
		 consolidate = do_consolidation();
	     goto redraw;

	  case AC_KEYCMD:
	     k_cmd = alt_cmd_key;
	     goto alt_key;

	  case AC_REENTER_GROUP:
	     menu_return(ME_REENTER_GROUP);
	 }

      case K_QUIT:
	 menu_return(ME_QUIT);

      case K_CANCEL:
	 savemode = "Cancel";
	 fname = "";
	 goto cancel1;

      case K_SAVE_NO_HEADER:
      case K_SAVE_SHORT_HEADER:
      case K_SAVE_FULL_HEADER:
      case K_PRINT:
      case K_UNSHAR:
      case K_PATCH:
      case K_UUDECODE:

	 if (numa < 0) goto nextmenu;

	 fname = init_save(k_cmd, &savemode);
	 if (fname == NULL) goto Prompt;

      cancel1:
	 enable_stop = 0;
	 save_selected = 0;
	 doing_unshar = k_cmd == K_UNSHAR || k_cmd == K_PATCH;
	 did_unshar = 0;

	 m_startinput();

	 if (novice)
	     msg(" * selected articles on this page, + all selected articles");

	 while (!save_selected && !did_unshar) {
	     prompt("\1%s\1 %.*s Article (* +): ",
		    savemode, Columns - 25, fname);

	     k_cmd = get_k_cmd();

	     if (k_cmd == K_SELECT_SUBJECT) {
		 save_selected = 1;
		 cura = 0;
		 article_id = firsta;
		 last_save = firsta + numa;
	     } else
	     if (k_cmd == K_AUTO_SELECT) {
		 save_selected = 1;
		 cura = -firsta;
		 article_id = 0;
		 last_save = n_articles - 1;
	     } else
	     if (k_cmd == K_ARTICLE_ID) {
		 cura = article_id;
		 article_id += firsta;
		 last_save = article_id;
		 if (articles[article_id]->flag & A_CLOSED) {
		     int n = save_closed_mode % 10;
		     int c;
		     
		     if (article_id == n_articles - 1) 
			 goto save_it;
		     if (articles[article_id + 1]->flag & A_ROOT_ART)
			 goto save_it;

		     if (save_closed_mode >= 10) {
			 prompt("\1%s thread\1 (r)oot (s)elected (u)nread (b)oth (a)ll  (%c)",
				savemode, "rsuba"[n]);
			 switch (get_c()) {
			  case 'r':	n = 0; break;
			  case 's':	n = 1; break;
			  case 'u':	n = 2; break;
			  case 'b':	n = 3; break;
			  case 'a':	n = 4; break;
			  case SP:
			  case CR:
			  case NL:	break;
			  default:	ding(); goto Prompt;
			 }
		     }
		     switch (n) {
		      case 0:		/* save root only */
			 break;
		      case 1:		/* save all selected */
			 save_selected = 1;
			 break;
		      case 2:		/* save all unread */
			 save_selected = 2;
			 break;
		      case 3:		/* save all selected + unread */
			 save_selected = 3;
			 break;
		      case 4: 		/* save all articles */
			 break;
		     }
		     if (n) last_save = next_root_article(article_id) - 1;
		     save_selected |= 8;	/* closed subject */
		     temp1 = cura;
		 }
	     } else
		 break;

	  save_it:
	     for ( ; article_id <= last_save ; article_id++, cura++) {
		 ah = articles[article_id];
		 switch (save_selected & 0x3) {
		  case 0:
		     break;
		  case 3:
		     if (ah->attr == 0) break;
		  case 1:
		     if ((ah->attr & A_SELECT) == 0) continue;
		     break;
		  case 2:
		     if (ah->attr == 0) break;
		     continue;
		 }		     

		 if (cur_k_cmd == K_CANCEL) {
		     if (current_group->group_flag & G_FOLDER) {
			 fcancel(ah);
		     } else
			 switch (cancel(ah)) {
			  case -1:
			     did_unshar = 1;
			     continue;
			  case 0:
			     ah->attr = A_CANCEL;
			     break;
			  default:
			     continue;
			 }

		     if (!did_unshar)
			 mark();

		     continue;
		 }

		 if (doing_unshar) {
		     did_unshar++;
		 } else
		 if (ah->subject != NULL)
		     prompt("Processing '%.50s'...", ah->subject);
		 else if (cura >= 0 && cura <= numa)
		     prompt("Processing %c...", ident[cura]);
		 else
		     prompt("Processing entry %d...", article_id);

		 if (save(ah)) {
		     ah->attr = A_READ;
		     if (doing_unshar) continue;

		     if (cura >= 0 && cura <= numa)
			 mark();
		 }
	     }
	     if (save_selected & 8) {
		 save_selected = 0;	/* select closed */
		 cura = temp1;
		 mark();
	     }
	 }

	 if (save_selected) cura = 0;

	 m_endinput();

	 enable_stop = 1;
	 if (cur_k_cmd != K_CANCEL)
	     end_save();

	 if (did_unshar) {
	     printf("\r\n");
	     any_key(0);
	     goto redraw;
	 }
	 goto Prompt;

      case K_FOLLOW_UP:
#ifdef NNTP_POST
	 if (use_nntp && nntp_no_post()) goto same_prompt;
#endif
      case K_REPLY:
	 if (numa < 0) goto nextmenu;

	 prompt(k_cmd == K_REPLY ?
		"\1Reply to author\1 of article: " :
		"\1Follow Up\1 to article: ");

	 if (get_k_cmd() == K_ARTICLE_ID)
	     if (answer(articles[firsta+article_id], k_cmd, -1))
		 goto redraw;

	 goto Prompt;

      case K_POST:

#ifdef NNTP_POST
	 if (use_nntp && nntp_no_post())
	     goto same_prompt;
#endif
	 if (post_menu()) goto redraw;
	 goto Prompt;

      case K_MAIL_OR_FORWARD:
	 if (numa < 0) goto nextmenu;

	 prompt("\1Article to be forwarded\1 (SP if none): ");

	 if ((k_cmd = get_k_cmd()) == K_ARTICLE_ID) {
	     if (answer(articles[firsta+article_id], K_MAIL_OR_FORWARD, 1))
		 goto redraw;
	 } else
	 if (k_cmd == K_CONTINUE)
	     if (answer((article_header *)NULL, K_MAIL_OR_FORWARD, 0))
		 goto redraw;

	 goto Prompt;
/*
      case K_CANCEL:
	 if (numa < 0) goto nextmenu;

	 if (current_group->group_flag & G_FOLDER) {
	     prompt("\1Cancel Folder\1 Article: ");
	     if (get_k_cmd() == K_ARTICLE_ID) {
		 cura = article_id;
		 fcancel(articles[firsta+article_id]);
		 mark();
	     }
	     goto Prompt;
	 }

	 prompt("\1Cancel\1 Article: ");

	 if (get_k_cmd() == K_ARTICLE_ID)
	     if (cancel(articles[firsta+article_id]) & 1) goto redraw;
	 goto Prompt;
*/
      case K_UNSUBSCRIBE:
	 if (unsubscribe(current_group)) {
	     if (current_group->group_flag & G_UNSUBSCRIBED)
		 menu_return(ME_NEXT);
	     home();
	     CALL(print_header)();
	 }
	 goto Prompt;

      case K_GROUP_OVERVIEW:
	 group_overview(-1);
	 goto redraw;

      case K_KILL_HANDLING:
	 switch (kill_menu((article_header *)NULL)) {
	  case 0:		/* select */
	     do_auto_select((regexp *)NULL, 2);
	     break;
	  case 1:		/* kill */
	     if (!do_auto_kill()) break;
	     goto junk_killed_articles;
	  default:
	     break;
	 }
	 goto Prompt;

      case K_CONTINUE:	/* goto next menu page or show the articles */
	 repl_attr(firsta, nexta, 0, A_SEEN, 0);
	 /* fall thru */
      case K_CONTINUE_NO_MARK:	/* but don't mark unselected articles */
	 if (nexta < n_articles) goto nextmenu;
	 break;

      case K_READ_GROUP_THEN_SAME:
	 if (temp = mark_read_return) goto do_marked_by;
	 break;

      case K_NEXT_GROUP_NO_UPDATE:
	 if (temp = mark_next_group) goto do_marked_by;
	 menu_return(ME_NEXT);

      case K_READ_GROUP_UPDATE:
	 if ((temp = mark_read_skip) == 0) break;
      do_marked_by:
	 temp1 = 0; temp2 = n_articles;
	 switch (temp) {
	  case 1:
	     temp1 = firsta;
	  case 3:
	     temp2 = nexta;
	     break;
	  case 2:
	     temp2 = firsta - 1;
	     break;
	  case 4:
	     break;
	 }
	 repl_attr(temp1, temp2, 0, A_SEEN, 0);
	 if (k_cmd != K_NEXT_GROUP_NO_UPDATE) break;
	 menu_return(ME_NEXT);

      case K_PREVIOUS:
	 menu_return(ME_PREV);

      case K_ADVANCE_GROUP:
      case K_BACK_GROUP:
	 if (merged_menu) {
	     msg("No possible on merged menu");
	     goto same_prompt;
	 }
	 /* FALL THRU */

      case K_GOTO_GROUP:
	 temp1 = n_articles;

	 switch (goto_group(k_cmd, (article_header *)NULL, (flag_type)0)) {

	  case ME_REDRAW:
	     firsta = 0;
	     if (temp1 != n_articles && consolidate)
		 consolidate = do_consolidation();
	     goto redraw;

	  case ME_NO_ARTICLES:
	     msg("No selections made.");

	  case ME_NO_REDRAW:
	     goto Prompt;

	  case ME_QUIT:
	     menu_return( ME_QUIT );

	  case ME_PREV:
	     goto redraw;

	  case ME_NEXT:
	     s_keyboard = 1;
	     goto empty_menu_hack;
	 }

      case K_OPEN_SUBJECT:
	 if (numa < 0) goto nextmenu;
	 prompt("\1Open subject\1");

	 k_cmd = get_k_cmd();
	 if (k_cmd != K_ARTICLE_ID) {
	     if (k_cmd != cur_k_cmd) goto Prompt;
	     article_id = cura;
	 }

      open_subject:
	 ah = articles[firsta+article_id];
	 if (!(ah->flag & A_CLOSED) ||
	     (ah->flag & (A_ROOT_ART | A_NEXT_SAME)) == A_ROOT_ART)
	     goto Prompt;

	 cura = article_id = root_article(firsta + article_id) - firsta;
	 while (cura + firsta < n_articles) {
	     ah = articles[firsta+cura];
	     if (cura > article_id && ah->flag & A_ROOT_ART)
		 break;
	     ah->flag &= ~A_CLOSED;
	     cura++;
	 }
	 cura = article_id;
	 goto partial_redraw;
	 
      case K_CLOSE_SUBJECT:
	 if (numa < 0) goto nextmenu;
	 prompt("\1Close subject\1");

	 k_cmd = get_k_cmd();
	 if (k_cmd != K_ARTICLE_ID) {
	     if (k_cmd != cur_k_cmd) goto Prompt;
	     article_id = cura;
	 }

	 ah = articles[firsta+article_id];
	 if ((ah->flag & A_CLOSED) ||
	     (ah->flag & (A_ROOT_ART | A_NEXT_SAME)) == A_ROOT_ART) {
	     cura = next_root_article(firsta + cura) - firsta;
	     goto Prompt;
	 }

	 cura = article_id = root_article(firsta + article_id) - firsta;
	 while (cura + firsta < n_articles) {
	     ah = articles[firsta+cura];
	     if (cura > article_id && ah->flag & A_ROOT_ART)
		 break;
	     ah->flag |= A_CLOSED;
	     cura++;
	 }
	 cura = article_id;
	 next_cura = next_root_article(firsta + cura) - firsta;
	 if (cura >= 0) goto partial_redraw_nc;
	 articles[cura + firsta]->menu_line = articles[firsta]->menu_line;
	 firsta += cura;
	 goto redraw;

      case K_LEAVE_NEXT:
      case K_JUNK_ARTICLES:
	 junk_prompt = cur_k_cmd == K_JUNK_ARTICLES ? 1 : 5;

	 for (;;) {
	     switch (junk_prompt) {
	      case 1:
		 if (novice) msg("Use J repeatedly to select other functions");
		 prompt("\1Mark read\1 S)een U)nmarked A)ll *+)selected a-z . [LN]");
		 junk_attr = A_READ;
		 break;
	      case 2:
		 prompt("\1Unmark\1 S)een R)ead a-z [*+LAN.J] ");
		 junk_attr = 0;
		 break;
	      case 3:
		 prompt("\1Select\1 L)eft-over, N(leave-next) [USRa-z.J]");
		 junk_attr = A_SELECT;
		 break;
	      case 4:
		 prompt("\1Kill\1 R)ead S)een [LANU*+a-z.J]");
		 junk_attr = A_KILL;
		 break;
	      case 5:
		 prompt("\1Leave\1 a-z .,/ * + U)nmarked [LANRSJ]");
		 junk_attr = A_LEAVE_NEXT;
		 break;
	      default:
		 junk_prompt = 1;
		 continue;
	     }

	  junk_another:
	     if (cura < 0 || cura > numa) cura = 0;
	     cursor_at_id();

	     switch (get_k_cmd()) {
	      case K_JUNK_ARTICLES:
		 junk_prompt++;	/* can be 0 */
		 continue;

	      case K_ARTICLE_ID:
		 cura = article_id;
		 if (junk_attr == A_KILL) junk_attr = A_READ;
		 if (auto_select_closed > 0 && articles[firsta + cura]->flag & A_CLOSED)
		     repl_attr_subject(A_KILL, junk_attr, 1);
		 else {
		     articles[firsta + cura]->attr = junk_attr;
		     mark();
		     cura++;
		 }
		 goto junk_another;

	      case K_NEXT_LINE:
		 cura++;
		 goto junk_another;
		 
	      case K_PREV_LINE:
		 --cura;
		 goto junk_another;

	      case K_SELECT_SUBJECT:
		 if (junk_attr == A_KILL) junk_attr = A_READ;
		 repl_attr(firsta, nexta, A_AUTO_SELECT, A_SELECT, 0);
		 repl_attr(firsta, nexta, A_SELECT, junk_attr, 1);
		 goto Prompt;

	      case K_AUTO_SELECT:
		 repl_attr_all(A_AUTO_SELECT, A_SELECT, 0);
		 orig_attr = A_SELECT;
		 break;

	      default:
		 switch (cur_key) {
		  case 'S':
		     orig_attr = A_SEEN;
		     break;

		  case 'U':
		     orig_attr = 0;
		     break;

		  case 'L':
		     if (junk_attr == A_KILL) junk_attr = A_READ;
		     orig_attr = A_LEAVE;
		     break;

		  case 'A':
		     orig_attr = A_KILL;
		     break;

		  case 'N':
		     orig_attr = A_LEAVE_NEXT;
		     break;

		  case 'R':		/* kill read articles */
		     orig_attr = A_READ;
		     break;

		  default:
		     goto Prompt;
		 }
		 break;
	     }
	     break;
	 }
	 if (nexta - firsta < n_articles) {
	     prompt("On all menu pages? ");
	     switch (yes(1)) {
	      case -1:
		 goto Prompt;
	      case 0:
		 if (!repl_attr(firsta, nexta, orig_attr, junk_attr, 1))
		     goto Prompt;
		 break;
	      case 1:
		 if (!repl_attr_all(orig_attr, junk_attr, 1))
		     goto Prompt;
		 break;
	     }
	 } else
	     if (!repl_attr(firsta, nexta, orig_attr, junk_attr, 1))
		 goto Prompt;

	 if (junk_attr != A_KILL) goto Prompt;

      junk_killed_articles:
	 elim_list[0] = firsta;
	 elim_list[1] = firsta + cura;
	 elim_list[2] = nexta;
	 if (elim_articles(elim_list, 3)) {
	     firsta = elim_list[0];
	     goto redraw;
	 }
	 firsta = elim_list[0];
	 cura   = elim_list[1] - firsta;
	 nexta  = elim_list[2];
	 goto Prompt;

      case K_ARTICLE_ID:
	 if (numa < 0) goto nextmenu;

	 if (!is_k_select && auto_preview_mode) goto auto_preview;

	 cura = article_id;
	 if (!auto_select_closed || !(articles[firsta+cura]->flag & A_CLOSED)) {
	     toggle();
	     if (!is_k_select && auto_select_subject) goto select_subject;
	     mark();
	     cura++;
	     goto same_prompt;
	 }
	 
	 if (auto_select_closed < 0) {
	     article_id = cura;
	     goto open_subject;
	 }

	 mi = menu_info + articles[firsta+cura]->menu_line;
	 if (mi->mi_unread == 0)
	     repl_attr_subject(A_KILL, A_SELECT, 1);
	 else if (mi->mi_selected == mi->mi_unread)
	     repl_attr_subject(auto_select_closed == 2 ? A_KILL : A_SELECT, 0, 1);
	 else
	     repl_attr_subject(auto_select_closed == 2 ? A_KILL : 0, A_SELECT, 1);
	 goto same_prompt;

      case K_SELECT_INVERT:
	 if (numa < 0) goto nextmenu;

	 temp = cura;

	 no_raw();	/* for x-on/x-off */
	 for (cura = 0; cura <= numa; cura++) {
	     toggle();
	 }
	 for (cura = 0; cura <= numa; cura++) {
	     if (IS_VISIBLE(articles[firsta+cura])) mark();
	 }
	 fl;

	 REDRAW_CHECK;
	 raw();

	 cura = temp;
	 goto same_prompt;

      case K_UNSELECT_ALL:
	 if (last_k_cmd == K_UNSELECT_ALL)
	     repl_attr_all(A_SELECT, 0, 1);
	 else
	     repl_attr_all(A_AUTO_SELECT, 0, 1);
	 fl;
	 cura = 0;
	 goto same_prompt;

      case K_NEXT_LINE:
	 if (numa < 0) goto nextmenu;

	 cura++;
	 goto same_prompt;

      case K_PREV_LINE:
	 if (numa < 0) goto nextmenu;

	 if (--cura < 0) cura = numa;
	 set_root_if_closed();
	 goto same_prompt;

      case K_SELECT_SUBJECT:
	 if (numa < 0) goto nextmenu;

	 if (last_k_cmd != K_ARTICLE_ID)
	     toggle();

      select_subject:
	 repl_attr_subject(A_KILL, last_attr, 1);
	 goto same_prompt;

      case K_SELECT_RANGE:
	 if (numa < 0) goto nextmenu;

	 if (last_k_cmd == K_ARTICLE_ID) {
	     cura--;
	     if (cura < 0) cura = numa;
	 } else
	     last_attr = (articles[firsta+cura]->attr & A_SELECT) ? 0 : A_SELECT;

      range_again:

	 prompt("\1%select range\1 %c-", last_attr ? "S" : "Des", ident[cura]);

	 k_cmd = get_k_cmd();
	 if (k_cmd == K_SELECT_RANGE) {
	     last_attr = last_attr ? 0 : A_SELECT;
	     goto range_again;
	 }

	 if (k_cmd != K_ARTICLE_ID) goto Prompt;

	 if (article_id > cura) {
	     article_number tmp = cura;
	     cura  = article_id;
	     article_id = tmp;
	 }

	 repl_attr(firsta+article_id, firsta+cura+1, A_KILL, last_attr, 1);
	 goto Prompt;

      case K_AUTO_SELECT:
	 do_auto_select((regexp *)NULL, 1);
	 goto same_prompt;

     case K_GOTO_MATCH:
	 prompt("\1Select regexp\1 ");
	 if ((fname = get_s(NONE, NONE, NONE, NULL_FCT)) == NULL)
	     goto Prompt;

	 if (*fname != NUL) {
	     if (regular_expr) freeobj(regular_expr);
	     if (case_fold_search) fold_string(fname);
	     regular_expr = regcomp(fname);
	 }

	 if (regular_expr == NULL)
	     msg("No previous expression");
	 else
	     do_auto_select(regular_expr, 2);

	 goto Prompt;

      case K_NEXT_PAGE:
	 if (nexta < n_articles) goto nextmenu;
	 if (firsta == 0) goto same_prompt;

	 nexta = 0;
	 goto nextmenu;

      case K_PREV_PAGE:
	 if (firsta == 0 && nexta == n_articles) goto same_prompt;

      prevmenu:
	 nexta = (firsta > 0 ? firsta : n_articles);
	 
	 for (menu_length = maxa; menu_length > 0 && --nexta >= 0; ) {
	    if (!IS_VISIBLE(articles[nexta])) continue;
	    if (--menu_length > 0) {
		switch (menu_spacing) {
		 case 0:
		    break;
		 case 1:
		    if ((articles[nexta]->flag & A_ROOT_ART) == 0) break;
		 case 2:
		    --menu_length;
		    break;
		}
	    }
	 }
	 if (nexta < 0) nexta = 0;
	 goto nextmenu;

      case K_FIRST_PAGE:
	 if (firsta == 0) goto same_prompt;

	 nexta = 0;
	 goto nextmenu;

      case K_LAST_PAGE:
	 if (nexta == n_articles) goto same_prompt;
	 firsta = 0;
	 goto prevmenu;

      case K_PREVIEW:
	 if (numa < 0) goto nextmenu;

      preview_other:

	 prompt("\1Preview article\1");
	 k_cmd = get_k_cmd();

	 if (k_cmd != K_ARTICLE_ID) {
	     if (k_cmd != K_PREVIEW)
		 goto Prompt;
	     article_id = cura;
	 }

      auto_preview:
	 temp = prompt_line;

      preview_next:
	 cura = article_id;
	 ah = articles[firsta+cura];

	 no_raw();
	 orig_attr = ah->attr;
	 ah->attr = 0;
	 menu_cmd = more(ah, MM_PREVIEW, prompt_line);
	 if (menu_cmd == MC_MENU) {
	     next_cura = cura;
	     if (ah->attr == 0) ah->attr = orig_attr;
	     if (prompt_line < 0) goto redraw;
	     mark();
	     prompt_line = temp;
	     goto build_prompt;
	 }

	 if (ah->attr == 0)
	     ah->attr = preview_mark_read ? A_READ : orig_attr;

	 if (prompt_line >= 0)
	     mark();
	 next_cura = ++cura;

	 switch (menu_cmd) {

	  case MC_DO_KILL:
	     if (!do_auto_kill()) break;
	     elim_list[0] = firsta;
	     elim_list[1] = firsta + cura;
	     elim_articles(elim_list, 2);
	     firsta = elim_list[0];
	     next_cura = elim_list[1] - firsta;
	     goto redraw;

	  case MC_DO_SELECT:
	     if (prompt_line >= 0) { /* not redrawn */
		 do_auto_select((regexp *)NULL, 2);
		 break;
	     }
	     numa = -1;
	     do_auto_select((regexp *)NULL, 2);
	     /* FALL THRU */

	  case MC_QUIT:
	     menu_return( ME_QUIT );

	  case MC_REENTER_GROUP:
	     menu_return( ME_REENTER_GROUP );

	  case MC_NEXT:
	  case MC_PREVIEW_NEXT:
	     if (prompt_line < 0) {	/* redrawn screen ! */
		 if (quit_preview(menu_cmd)) goto redraw;
		 prompt_line = Lines;
	     } else {
/*		 if (ah->attr == A_LEAVE || ah->attr == A_LEAVE_NEXT) {
		     cura--;
		     mark();
		     cura++;
		 }
*/		 if (quit_preview(menu_cmd)) break;
		 prompt_line = temp;
	     }
	     article_id = cura;
	     goto preview_next;

	  case MC_PREVIEW_OTHER:
	     prompt_line = temp;
	     raw();
	     goto preview_other;

	  default:
	     if (prompt_line < 0) goto redraw;
	     break;
	 }

	 prompt_line = temp;
	 goto build_prompt;

      case K_LAYOUT:
	 if (++fmt_linenum > 4) fmt_linenum = 0;
	 goto redraw;

      default:
	 msg("Command %d not supported", k_cmd);
	 goto same_prompt;
     }

    no_raw();

 do_auto_read:
    switch (show_articles()) {

     case MC_MENU:
	goto redraw;

     case MC_READGROUP:
	if (k_cmd == K_READ_GROUP_THEN_SAME) {
	    if (read_ret_next_page && nexta < n_articles)
		firsta = nexta;
	    goto redraw;
	}
	
     case MC_NEXTGROUP:
	menu_cmd = ME_NEXT;
	break;

     case MC_REENTER_GROUP:
	menu_cmd = ME_REENTER_GROUP;
	break;

     case MC_QUIT:
	menu_cmd = ME_QUIT;
	break;

     default:
	sys_error("show_articles returned improper value");
    }

 menu_exit:

    cur_bypass = o_bypass;
    n_selected = o_selected;
    firsta = o_firsta;
    in_menu_mode = o_mode;
    menu_level--;

    no_raw();
    return menu_cmd;
}


/*
 *	return article header for article on menu
 */

article_header *get_menu_article()
{
    register article_header *ah;

    fputs(" from article: ", stdout); fl;

    if (get_k_cmd() == K_ARTICLE_ID) {
	ah = articles[firsta + article_id];
	if (ah->a_group) init_group(ah->a_group);
	return ah;
    }

    return NULL;
}



/*
 *	read command from command line
 */

alt_command()
{
    int ok_val, macro_cmd;
    char *cmd, brkchars[10];
    extern key_type erase_key;
    extern int get_from_macro;
    extern int alt_completion();

    if (get_from_macro)
	ok_val = AC_UNCHANGED;
    else {
	prompt(":");
	ok_val = AC_PROMPT;
    }

 again:

    sprintf(brkchars, "?%c ", erase_key);

    cmd = get_s(NONE, NONE, brkchars, alt_completion);
    if (cmd == NULL ||
	*cmd == NUL || *cmd == SP || *cmd == erase_key)
	return ok_val;

    macro_cmd = get_from_macro;

    if (*cmd == '?') {
	display_file("help.extended", CLEAR_DISPLAY);
	ok_val = AC_REDRAW;
	goto new_prompt;
    }

    ok_val = parse_command(cmd, ok_val, (FILE *)NULL);
    if (ok_val != AC_REDRAW || !delay_redraw) return ok_val;

 new_prompt:
    if (macro_cmd) return ok_val;

    prompt_line = -1;
    printf("\n\r:");
    fl;
    goto again;
}

These are the contents of the former NiCE NeXT User Group NeXTSTEP/OpenStep software archive, currently hosted by Netfuture.ch.