[comp.os.vms] TPU, and searching for end of line

gregg@a.cs.okstate.edu (Gregg Wonderly) (03/05/88)

There has been muchado about the fact the EVE does not allow you to search
for text in context.  The VMS_SHAR file below contains my regular expression
code from TPU-VI, modified so that you can EXTEND EVE with the RE.TPU file.
The result is that EVE_FIND and EVE_REPLACE will recognize RE's.  RE.RNO is
runoff source for HELP file entries to document the change in behavior of
the FIND and REPLACE commands.

Do with these as you wish...

-----
Gregg Wonderly
Department of Computing and Information Sciences
Oklahoma State University

UUCP:      {cbosgd, ihnp4, rutgers}!okstate!gregg
Internet:  gregg@A.CS.OKSTATE.EDU

........................ Cut between dotted lines and save ......................
$!..............................................................................
$! VAX/VMS archive file created by VMS_SHAR V-4.03 05-Aug-1987
$! which was written by Michael Bednarek (U3369429@ucsvc.dn.mu.oz.au)
$! To unpack, simply save and execute (@) this file.
$!
$! This archive was created by GREGG
$!      on Wednesday 2-MAR-1988 19:35:29.78
$!
$! It contains the following 2 files:
$! RE.RNO RE.TPU
$!==============================================================================
$ Set Symbol/Scope=(NoLocal,NoGlobal)
$ Version=F$GetSYI("VERSION") ! See what VMS version we have here:
$ If Version.ges."V4.4" then goto Version_OK
$ Write SYS$Output "Sorry, you are running VMS ",Version, -
                ", but this procedure requires V4.4 or higher."
$ Exit 44
$Version_OK: CR[0,8]=13
$ Pass_or_Failed="failed!,passed."
$ Goto Start
$Convert_File:
$ Read/Time_Out=0/Error=No_Error1/Prompt="creating ''File_is'" SYS$Command ddd
$No_Error1: Define/User_Mode SYS$Output NL:
$ Edit/TPU/NoSection/NoDisplay/Command=SYS$Input/Output='File_is' -
        VMS_SHAR_DUMMY.DUMMY
f:=Get_Info(Command_Line,"File_Name");b:=Create_Buffer("",f);
o:=Get_Info(Command_Line,"Output_File");Set (Output_File,b,o);
Position (Beginning_of(b));Loop x:=Erase_Character(1); Loop ExitIf x<>"V";
Move_Vertical(1);x:=Erase_Character(1);Append_Line;Move_Horizontal
(-Current_Offset);EndLoop;Move_Vertical(1);ExitIf Mark(None)=End_of(b)
EndLoop;Exit;
$ Delete VMS_SHAR_DUMMY.DUMMY;*
$ Checksum 'File_is
$ Success=F$Element(Check_Sum_is.eq.CHECKSUM$CHECKSUM,",",Pass_or_Failed)+CR
$ Read/Time_Out=0/Error=No_Error2/Prompt=" CHECKSUM ''Success'" SYS$Command ddd
$No_Error2: Return
$Start:
$ File_is="RE.RNO"
$ Check_Sum_is=157006870
$ Copy SYS$Input VMS_SHAR_DUMMY.DUMMY
X.I-1
X3 FIND
X.br
XFIND
X.s
X.endif manual
XSearches for an occurrence of a pattern.  FIND uses a very powerful pattern
Xmatching mechanism known as a regular expression.  Regular expressions use
Xspecial characters to denote special patterns such as the beginning of the
Xline, and the end of the line.  The regular expressions recognized by FIND have
Xthe following form.
X.s
X.lm +5
XThe character '.' represents any character.  Thus, the pattern ".t" matches
X"it", "at", "ot", "lt", and other strings of length 2 that end in a "t".
X.s
XThe character '^' as the first character of a pattern indicates that the
Xfollowing characters should be looked for only at the beginning of the line.
X.s
XThe character '$' as the last character of a pattern indicates that the
Xpreceeding characters should be looked for only at the end of a line.  Anywhere
Xelse in the pattern it is taken literally.
X.s
XThe character '[' starts a list of characters that ends with a ']'.  The
Xcharacters between the '[' and ']' indicate all the possible characters that
Xcan occur ONCE at that position in the pattern.  Thus, the pattern "[ai]t"
Xmatches ONLY the strings "it" and "at".  If the first character following the
X'[' is a '^', then the complement of the set of characters that follows is
Xassumed.  Thus, the pattern "[^ia]t", never matches the strings "it" or "at".
XIt is possible to denote a sequence of characters by placing a '-' between two
Xcharacters.  Thus, "[0-9]" is short for "[0123456789]".
X.s
XThe character '*' is a multiplicative operator.  Its presence indicates that
Xthe preceeding character or [...] set might occur zero or more times.  Thus,
Xthe pattern "i[ai]*t" matches "it", "iat", "iit", "iaat", "iiit", "iait", and
Xall other strings starting with "i", ending with "t", and having only the
Xcharacters 'a' or 'i' between.
X.s
XBecause the characters '[', '.', '^', '$', and '*' are special, they must
Xbe preceed by a '\' if they are meant literally in a pattern.  Thus, '\'
Xitself must be indicated as "\\".
X.lm -5
X.s
XPress the FIND key and then enter the pattern from the main keyboard.  End the
Xstring by pressing the RETURN key.  To search for the previously specified
Xpattern, press the FIND key twice, or use the FIND NEXT key on the keypad
X(normally PF3).  You can press the FIND key, and then press the up-arrow
Xkey to edit the previously specified search string.
X.s
XFIND is case-insensitive if the string contains only lowercase letters; it is
Xcase-sensitive if the string contains any uppercase letters.
X.s
XThe direction of the search (Forward or Reverse) is determined by the current
Xdirection of the buffer, as shown in the status line at the bottom of the
Xbuffer.  If the pattern can be found only by searching in the opposite
Xdirection, GWEDIT asks you if you want to move the cursor in that direction.
X.s
XThe FIND command is bound to the FIND key, and the SHIFT-KEY, PF3 key sequence.
X.I-1
X3 REPLACE
X.br
XREPLACE
X.s
X.endif manual
XLets you repeatedly substitute one word or phrase for another
Xthroughout a buffer.  For example,
X.s
X.I+5
Xreplace good excellent
X.s
Xinstructs GWEDIT to replace "good" with "excellent" in the current
Xbuffer.
X.s
XEach occurrence of "good" within the current buffer will be highlighted in
Xreverse video, and you will be prompted for an action.  Typing "YES", "Y", "y",
X"yes" or just pressing RETURN replaces this occurrence.  "NO", "no", "n", or
X"N", skips this occurrence.  "All" replaces this occurrence and all future
Xoccurrences without any further prompting.  "Last" replaces this occurrence and
Xstops the REPLACE command.  "Quit" stops the REPLACE command without replacing
Xthis occurrence. 
X.s
XIn order to replace multi-word phrases, put the phrases in
Xquotation marks.  For example,
X.s
X.I+5
Xreplace "the first one" "the second one"
X.s
XSpecial notation in the first string allows you to specify regular expressions.
XSee the FIND command for more information.  You may also specify groups of
Xcharacters in the first string to be placed in the second string when the
Vsubstitution occurs.  An '&' in the second string will cause all text that was
X found
Xby the search pattern to be placed in the resultant string.
X.s
X.lm+5
Xe.g.  REPLACE "go[ <TAB>]*" "foo$&"
X.s
X.lm-5
Xwill cause the string "foo$" to be prepended to all occurrances of the string
X"go" followed by zero or more space or tab characters.
X.s
XIt is also possible to pick portions of the original text to be kept in the
Xresultant text.
X.s
X.lm+5
Xe.g.  REPLACE "foo$\(go[ <TAB>]\)" "\1"
X.lm-5
X.s
Xwould cause the prefix "foo$" to be removed from all occurances of the string
X"foo$go" followed by zero or more space or tab characters.  The notation
Xshown is derived
Xfrom the EX(1) editor under UNIX.  Up to 9 sets of \( \) pairs may appear
Xinside the search string.  These are refered to by specifying the relative
Xnumber from left to right.  E.g. to swap two columns of text that occur at the
Xbeginning of a line, you might say
X.lm+5
X.s
XREPLACE "^\([^ <TAB>]*\)\([ <TAB>]*\)\([^ <TAB>]*\)" "\3\2\1"
X.lm -5
X.s
Xwhich delimits three groups of characters.  The first is a sequence of
Xcharacters which are not spaces or tabs, followed by a group of characters
Xwhich are spaces or tabs, followed, again, by a sequence of characters which
Xare not spaces or tabs.  By swapping field 3 with field 1, the columns will be
Xswapped.
$ GoSub Convert_File
$ File_is="RE.TPU"
$ Check_Sum_is=657926407
$ Copy SYS$Input VMS_SHAR_DUMMY.DUMMY
X!
X! Top-level find command.  Calls eve$find, as does the replace command.
X! The two commands have slightly different requirements, so they both
X! call eve$find and pass it a parameter to indicate the caller.
X!
X! Parameters:
X!
X!	target			String to find - input
X
XPROCEDURE eve_find (target, compile_it)
X
X	LOCAL
X		comp_val;
X
X	comp_val := compile_it;
X
X	IF (GET_INFO (compile_it, "TYPE") = STRING) THEN
X		comp_val := 1;	! Always compile when EVE_FIND is called by EVE_PARSE.
X	ENDIF;
X
X	eve$find (target, 0, comp_val, 0);
X
XENDPROCEDURE;
X
X!
X! Search for target in the current direction.  If not found in the
X! current direction look in the opposite direction, but do not go
X! there without prompting the user.  Search is case-insensitive if
X! target is all lowercase; otherwise is case-sensitive.
X! Returns range if target found, otherwise returns false.
X!
X! Parameters:
X!
X!	target			String to find - input
X!	replacing		If true, called by eve_replace; allow a
X!				match at current cursor position - input
X
XPROCEDURE eve$find (target, replacing, compile_it, do_parens)
X
X	LOCAL
X		new_target,				! Local copy of target
X		lowercase_target,		! Lowercase version of eve$x_target
X		start_find_key,			! String describing key used to invoke 
find
X		old_str,
X		old_pos,
X		new_pat,
X		stop_find_key,			! String describing key used after prom
pt
X		this_position,			! Marker for current cursor position
X		how_exact,				! Keyword to indicate case-sens
itivity
X		find_range,				! Range returned by search
X		other_direction,		! Keyword for opposite direction
X		other_direction_string,	! String for message including other_direction
X		find_reply,				! Reply to inquiry about changi
ng direction
X		change_direction_key;	! Keyword for key used to end find_reply
X
X	ON_ERROR
X		IF ERROR = TPU$_STRNOTFOUND THEN
X		  	find_range := 0;
X		ENDIF;
X	ENDON_ERROR;
X
X	start_find_key := eve$lookup_comment (LAST_KEY);
X
X	IF target <> eve$kt_null THEN
X		new_target := target;
X	ELSE
X		IF CURRENT_DIRECTION = FORWARD THEN
X			new_target := READ_LINE ("Forward Find: ");
X		ELSE
X		  	new_target := READ_LINE ("Reverse Find: ");
X		ENDIF;
X	ENDIF;
X
X	IF (compile_it = 0) AND (GET_INFO (new_target, "TYPE") = STRING) THEN
X		eve$x_orig_target := new_target;
X	ENDIF;
X
X	IF (new_target <> "") AND (GET_INFO (new_target, "TYPE") = STRING) AND
X										
				(compile_it <> 0) THEN
X		eve$x_orig_target := new_target;
X		eve$x_paren_cnt := 0;
X		new_pat := eve$pattern_gen (new_target, eve$x_paren_cnt, do_parens);
X
X		IF (new_pat = "") THEN
X			RETURN (0);
X		ENDIF;
X
X		EXECUTE (COMPILE ("eve$x_find_pat := " + new_pat));
X		new_target := eve$x_find_pat;
X	ENDIF;
X
X	stop_find_key := eve$lookup_comment (LAST_KEY);
X
X	IF new_target = eve$kt_null THEN
X		IF ((start_find_key = "find") AND
X				((stop_find_key = "find") OR (stop_find_key = "return")
)) THEN
X
X			IF eve$x_target = eve$kt_null THEN
X				MESSAGE ("No previous target to find");
X		  		RETURN (0);
X			ELSE
X				IF GET_INFO (eve$x_orig_target, "TYPE") = STRING THEN
X					MESSAGE (FAO ("Finding: !AS", eve$x_orig_target
));
X				ELSE
X					MESSAGE ("Finding previous target: ");
X				ENDIF;
X			ENDIF;
X		ELSE
X			MESSAGE ("Invalid terminator");
X			RETURN (0);
X		ENDIF;
X	ELSE
X		eve$x_target := new_target;
X		IF GET_INFO (eve$x_orig_target, "TYPE") = STRING THEN
X			MESSAGE (FAO ("Finding: !AS", eve$x_orig_target));
X		ELSE
X			MESSAGE ("Searching...");
X		ENDIF;
X	endif;
X
X	lowercase_target := eve$x_orig_target;
X
X	IF GET_INFO (lowercase_target, "TYPE") = STRING THEN
X		CHANGE_CASE (lowercase_target, LOWER);
X	ENDIF;
X
X	IF lowercase_target = eve$x_orig_target THEN
X		how_exact := NO_EXACT;
X	ELSE
X		how_exact := EXACT;
X	ENDIF;
X
X	this_position := MARK (NONE);
X
X	IF CURRENT_DIRECTION = FORWARD THEN
X		IF this_position <> END_OF (CURRENT_BUFFER) THEN
X			IF NOT replacing THEN
X				MOVE_HORIZONTAL (1);
X			ENDIF;
X			find_range := SEARCH (eve$x_target, FORWARD, how_exact);
X		ELSE
X			find_range := 0;
X		ENDIF;
X	ELSE
X		IF this_position <> BEGINNING_OF (CURRENT_BUFFER) THEN
X			MOVE_HORIZONTAL (-1);
X			find_range := SEARCH (eve$x_target, REVERSE, how_exact);
X		ELSE
X			find_range := 0;
X		ENDIF;
X	ENDIF;
X	
X	IF find_range = 0 THEN
X		IF CURRENT_DIRECTION = FORWARD THEN
X			other_direction := REVERSE;
X			other_direction_string := "reverse";
X		ELSE
X			other_direction := FORWARD;
X			other_direction_string := "forward";
X		ENDIF;
X
X		POSITION (this_position);
X
X		IF other_direction = FORWARD THEN
X			IF this_position <> END_OF (CURRENT_BUFFER) THEN
X				MOVE_HORIZONTAL (1);
X				find_range := SEARCH (eve$x_target, FORWARD, how_exact)
;
X			ELSE
X				find_range := 0;
X		  	ENDIF;
X		ELSE
X			IF this_position <> BEGINNING_OF (CURRENT_BUFFER) THEN
X				MOVE_HORIZONTAL (-1);
X				find_range := SEARCH (eve$x_target, REVERSE, how_exact)
;
X			ELSE
X				find_range := 0;
X			ENDIF;
X		ENDIF;
X
X		IF find_range = 0 THEN
X			IF GET_INFO (eve$x_orig_target, "TYPE") = STRING THEN
X				MESSAGE (FAO ("Could not find: !AS", eve$x_orig_target)
);
X			ELSE
X				MESSAGE ("Could not find pattern!");
X			ENDIF;
X
X			POSITION (this_position);
X			RETURN (0);
X		ELSE
X			find_reply :=
X				READ_LINE (FAO ("Found in !AS direction.  Go there? ",
X					other_direction_string));
X
X			! Hitting return or do means yes; hitting another non-typing
X			! key is probably a mistake, so interpret as no.
X
X			IF find_reply = eve$kt_null THEN
X				change_direction_key := eve$lookup_comment (LAST_KEY);
X
X				IF (change_direction_key = "return") OR
X									   	(change
_direction_key = "do") THEN
X					find_reply := "yes";
X				ELSE
X					find_reply := "no";
X				ENDIF;
X			ELSE
X				CHANGE_CASE (find_reply, LOWER);
X			ENDIF;
X
X			IF SUBSTR ("yes", 1, LENGTH (find_reply)) = find_reply THEN
X				SET (other_direction, CURRENT_BUFFER);
X				eve$update_status_lines;
X				eve$position_in_middle (BEGINNING_OF (find_range));
X				RETURN (find_range);
X			ELSE
X				POSITION (this_position);
X				RETURN (0);
X			ENDIF;
X		ENDIF;
X	ELSE
X		eve$position_in_middle (BEGINNING_OF (find_range));
X		RETURN (find_range);
X	ENDIF;
X
X	MESSAGE (eve$kt_null);
XENDPROCEDURE;
X
X!
X!
X! TPU pattern generator.  Generates a pattern string from the passed
X! regular expression.
X!
XPROCEDURE eve$pattern_gen (pat, paren_cnt, do_parens)
X
X	LOCAL
X		first,		! First pattern to be done
X		part_pat,
X		chno,
X		startchar,
X		haveany,
X		regular,
X		tstr,
X		endchar,
X		str_pat,
X		cur_pat,	! The current pattern to be extracted
X		cur_char,	! The current character in the regular
X			   		! expression being examined
X		new_pat,	! The output pattern
X		pos;       	! The position within the regular
X					! expression string that we are examining
X					! currently
X
X	paren_cnt := 0;
X
X	IF ((INDEX (pat, "$") <> 0) OR (INDEX (pat, "[") <> 0) OR
X					(INDEX (pat, "^") <> 0) OR (INDEX (pat, ".") <>
 0) OR
X						(INDEX (pat, "*") <> 0) OR (INDEX (pat,
 "\") <> 0) OR
X						(INDEX (pat, '"') <> 0)) THEN
X		new_pat := "";
X	ELSE
X		new_pat := '"'+pat+'"';
X		RETURN (new_pat);
X	ENDIF;
X
X	pos := 1;
X
X	IF SUBSTR (pat, pos, 1) = "^" THEN
X		new_pat := "line_begin";
X		pos := pos + 1;
X	ENDIF;
X
X	LOOP
X		EXITIF (pos > LENGTH (pat));
X
X		regular := 0;
X		cur_pat := "";
X		cur_char := substr (pat, pos, 1);
X		pat_str := "";
X
X		IF (cur_char = "$") THEN
X			cur_pat := "line_end";
X			IF (pos < LENGTH (pat)) THEN
X				cur_pat := "'$'";
X			ENDIF;
X		ELSE
X			IF cur_char = "[" THEN
X				pos := pos + 1;
X
X	 			IF SUBSTR (pat, pos, 1) = "^" THEN
X					pos := pos + 1;
X					part_pat := "notany('";
X				ELSE
X					part_pat := "any('";
X				ENDIF;
X
X				LOOP
X					EXITIF pos > LENGTH (pat);
X			   		EXITIF SUBSTR (pat, pos, 1) = "]";
X
X					IF SUBSTR (pat, pos, 1) = "\" THEN
X						pos := pos + 1;
X						IF pos > LENGTH (pat) THEN
X							MESSAGE ("Missing character aft
er \");
X							RETURN ("");
X						ENDIF;
X					ENDIF;
X
X	 				startchar := SUBSTR (pat, pos, 1);
X					pat_str := pat_str + startchar;
X
X					IF (SUBSTR (pat, pos+1, 1) = '-') THEN
X						pos := pos + 2;
X						IF (pos >= LENGTH (pat)) THEN
X							MESSAGE ("Missing character aft
er '-'");
X							RETURN ("");
X						ENDIF;
X
X						endchar := SUBSTR (pat, pos, 1);
X
X						chno := 1;
X						LOOP
X							EXITIF (ASCII(chno) = startchar
);
X							chno := chno + 1;
X						ENDLOOP;
X
X						LOOP
X							chno := chno + 1;
X							IF (chno > 255) THEN
X								MESSAGE ("Invalid chara
cter sequence for '-'");
X								RETURN ("");
X							ENDIF;
X
X							EXITIF (ASCII (chno-1) = endcha
r);
X							pat_str := pat_str + ASCII (chn
o);
X						ENDLOOP;
X					ENDIF;
X					pos := pos + 1;
X				ENDLOOP;
X
X				IF pat_str = "" THEN
X					MESSAGE ("No text found between []");
X					RETURN ("");
X				ENDIF;
X
X				IF (SUBSTR (pat, pos+1, 1) = "*") THEN
X					IF (part_pat = "notany('") THEN
X						cur_pat := cur_pat + "(scan('"+pat_str+
"')|"""")";
X					ELSE
X						cur_pat := cur_pat + "(span('"+pat_str+
"')|"""")";
X					ENDIF;
X					pos := pos + 1;
X				ELSE
X					cur_pat := part_pat + pat_str + "')";
X				ENDIF;
X			ELSE
X
X				!
X				!	Process a single character character, either '\
', '.',
X				!	or some literal character.  There might be a '*
' following
X				!	a character, in which case that is also handled
 here.
X				!
X
X				tstr := '"';
X				haveany := 0;
X				regular := 1;
X
X				LOOP
X					cur_char := SUBSTR (pat, pos, 1);
X					EXITIF (INDEX ("$^[", cur_char) > 0);
X					EXITIF (pos > LENGTH (pat));
X
X					IF cur_char = "\" THEN
X						pos := pos + 1;
X						cur_char := SUBSTR (pat, pos, 1);
X						IF (do_parens) THEN
X							IF (cur_char = "(") THEN
X								paren_cnt := paren_cnt 
+ 1;
X								IF tstr = '"' THEN
X									tstr := '""@o'+
STR(paren_cnt)+'&"';
X								ELSE
X									tstr := tstr + 
'"@o'+STR(paren_cnt)+'&"';
X								ENDIF;
X							ELSE
X								IF (cur_char = ")") THE
N
X									IF (paren_cnt =
 0) THEN
X										MESSAGE
 (
X										FAO ("N
o previous ""\("" near: !AS",
X										SUBSTR 
(pat, pos, LENGTH(pat)-pos))
X										);
X										RETURN 
(0);
X									ENDIF;
X
X									IF tstr = '"' T
HEN
X										tstr :=
 "@p"+STR(paren_cnt)+'&"';
X									ELSE
X										tstr :=
 tstr + '"@p' + 
X										
			STR(paren_cnt)+'&"';
X									ENDIF;
X								ELSE
X									tstr:=tstr+cur_
char;
X								ENDIF;
X							ENDIF;
X						ELSE
X							tstr:=tstr+cur_char;
X						ENDIF;
X					ELSE
X						IF (cur_char = ".") THEN
X							cur_char := "longer_than_1";
X						ENDIF;
X
X						! Check for zero or more
X
X						IF (SUBSTR (pat, pos+1, 1) = '*') THEN
X							pos := pos + 1;
X
X							IF (LENGTH (cur_char) > 1) THEN
X								cur_pat :=
X									"''&(span(eve$p
ch)|'')";
X							ELSE
X								cur_pat := "(span('"+cu
r_char+"')|"""")";
X							ENDIF;
X							tstr := tstr+'"'+"&"+cur_pat+"&
"+'"';
X							haveany := 0;
X						ELSE
X							IF (LENGTH (cur_char) > 1) THEN
X								IF (haveany) THEN
X									tstr := tstr +'
"'+"&"+"arb(1)"+"&"+'"';
X									haveany := 0;
X								ELSE
X									IF (LENGTH (tst
r)>0) and (tstr <> '"') THEN
X										tstr :=
 tstr +'"'+"&"+"arb(1)"+"&"+'"';
X									ELSE
X										tstr :=
 "arb(1)"+"&"+'"';
X									ENDIF
X								ENDIF;
X							ELSE
X								IF (cur_char = """") TH
EN
X									tstr := tstr + 
'""';
X									haveany := have
any + 2;
X								ELSE
X									tstr := tstr + 
cur_char;
X									haveany := have
any + 1;
X								ENDIF;
X							ENDIF;
X						ENDIF;
X					ENDIF;
X					pos := pos + 1;
X				ENDLOOP;
X				cur_pat := tstr + '"';
X				pos := pos - 1;
X			ENDIF;
X		ENDIF;
X
X		IF (regular) THEN
X			IF new_pat = "" THEN
X				new_pat := cur_pat;
X			ELSE
X				IF (LENGTH (tstr) > 1) THEN
X					new_pat := new_pat + "&" + cur_pat;
X				ENDIF;
X	       	ENDIF;
X		ELSE
X			IF new_pat = "" THEN
X				new_pat := cur_pat;
X			ELSE
X				new_pat := new_pat + "&" + cur_pat;
X			ENDIF;
X		ENDIF;
X		pos := pos + 1;
X	ENDLOOP;
X
X	pos := INDEX (new_pat, "&");
X	LOOP
X		EXITIF (pos = 0) OR (pos > LENGTH (new_pat));
X		IF (SUBSTR (new_pat, pos, 3) = '&""') AND 
X			(SUBSTR (new_pat, pos, 5) <> '&""""') THEN
X			new_pat := SUBSTR (new_pat, 1, pos-1) +
X										
		SUBSTR (new_pat, pos+3, 255);
X		ELSE
X			IF SUBSTR (new_pat, pos, 2) = '&@' THEN
X				new_pat := SUBSTR (new_pat, 1, pos-1) +
X										
		SUBSTR (new_pat, pos+1, 255);
X			ENDIF;
X		ENDIF;
X		pos := pos + 1;
X	ENDLOOP;
X
X	RETURN (new_pat);
XENDPROCEDURE;
X
X!
X!
X!
XPROCEDURE eve$do_substitution (source, dest)
X
X	LOCAL
X		cur_char,
X		result,
X		idx;
X
X	idx := 0;
X	result := "";
X
X	LOOP
X		EXITIF (idx > LENGTH(dest));
X
X		cur_char := SUBSTR (dest, idx, 1);
X		IF (cur_char = "&") THEN
X			result := result + source;
X			idx := idx + 1;
X		ELSE
X			IF (cur_char = '\') THEN
X				cur_char := SUBSTR(dest, idx+1, 1);
X				IF (INDEX (eve$x_digit_characters, cur_char) > 0) THEN
X					EXECUTE (COMPILE ("eve$x_glo_str := SUBSTR (p" 
+
X							cur_char +", LENGTH (o"+cur_cha
r+")+1,512);"));
X					result := result + eve$x_glo_str;
X				ELSE
X					result := result + "\" + cur_char;
X				ENDIF;
X				idx := idx + 2;
X			ELSE
X				result := result + cur_char;
X				idx := idx + 1;
X			ENDIF;
X		ENDIF;
X	ENDLOOP;
X
X	RETURN (result);
XENDPROCEDURE;
X
X!
X! Search and replace procedure.  Case-sensitivity of search is
X! same as for the find command.  If case-insensitive, replacements
X! are done to match case of current occurrence.
X!
X! Parameters:
X!
X!	replace_parameter_1	Old string - input
X!	replace_parameter_2	New string - input
X
XPROCEDURE eve_replace (replace_parameter_1, replace_parameter_2)
X
X	LOCAL
X		replace_text,
X		source_text,
X		target,	   				! Local copy of replace_paramet
er_1
X		replacement,			! Local copy of replace_parameter_2
X		this_buffer,		  	! Current buffer
X		this_mode,				! Keyword for current mode
X		lowercase_target,		! Lowercase version of target string
X		lowercase_replacement,	! Lowercase version of replacement string
X		uppercase_target,		! Uppercase version of target string
X		uppercase_replacement,	! Uppercase version of replacement string
X		capital_target,			! Capitalized version of target string
X		capital_replacement,	! Capitalized version of replacement string
X		how_exact,				! Keyword to indicate case-sens
itivity
X		replace_range,			! Range of current occurrence
X		highlight_range,		! Reverse-video version of replace_rang
e
X		replace_action,			! String reply to prompt
X		action_length,			! Length of replace_action
X		asking,					! True unless "all" option has 
been chosen
X		this_occurrence,		! String of replace_range
X		occurences;
X
X	this_buffer := CURRENT_BUFFER;
X	this_mode := GET_INFO (CURRENT_BUFFER, "MODE");
X	SET (INSERT, this_buffer);
X	asking := 1;
X
X	IF NOT (eve$prompt_string (replace_parameter_1, target,
X			   					"Old string: ", "No str
ing to replace")) THEN
X		RETURN;
X	ENDIF;
X
X	replacement := replace_parameter_2;
X
X	IF replacement = eve$kt_null THEN
X		replacement := READ_LINE ("New string: ");	! empty string is ok he
re
X	ENDIF;
X
X	lowercase_target := target;
X
X	IF GET_INFO (lowercase_target, "TYPE") = STRING THEN
X		CHANGE_CASE (lowercase_target, LOWER);
X	ENDIF;
X
X	lowercase_replacement := replacement;
X	CHANGE_CASE (lowercase_replacement, LOWER);
X
X	IF (lowercase_target = target) AND
X									(lowercase_repl
acement = replacement) THEN
X		how_exact := NO_EXACT;
X		uppercase_target := target;
X		IF GET_INFO (uppercase_target, "TYPE") = STRING THEN
X			CHANGE_CASE (uppercase_target, UPPER);
X		ENDIF;
X
X		capital_target := target;
X
X		IF GET_INFO (capital_target, "TYPE") = STRING THEN
X			eve$capitalize_string (capital_target);
X		ENDIF;
X
X		uppercase_replacement := replacement;
X		CHANGE_CASE (uppercase_replacement, UPPER);
X		capital_replacement := replacement;
X		eve$capitalize_string (capital_replacement);
X	ELSE
X		how_exact := EXACT;
X	ENDIF;
X
X	occurrences := 0;
X
X	LOOP
X		replace_range := eve$find (target, 1, 1, 1);
X
X		EXITIF replace_range = 0;
X		highlight_range :=
X			CREATE_RANGE (BEGINNING_OF (replace_range),
X								END_OF (replace_range),
 eve$x_highlighting);
X		POSITION (BEGINNING_OF (replace_range));
X		UPDATE (CURRENT_WINDOW);
X		source_text := 
X			CREATE_RANGE (BEGINNING_OF (replace_range),
X								END_OF (replace_range),
 NONE);
X
X		source_text := SUBSTR (source_text, 1, LENGTH (source_text));
X
X		LOOP
X			IF asking THEN
X				replace_action :=
X					READ_LINE ("Replace? Type yes, no, all, last, o
r quit: ");
X				CHANGE_CASE (replace_action, LOWER);
X			ELSE
X				replace_action := "yes";
X			ENDIF;
X
X			action_length := LENGTH (replace_action);
X			IF (replace_action = SUBSTR ("yes", 1, action_length)) OR
X	   				(replace_action = SUBSTR ("all", 1, action_leng
th)) OR
X	   				(replace_action = SUBSTR ("last", 1, action_len
gth)) OR
X	   									
			(action_length = 0) THEN
X				highlight_range := 0;
X
X				IF how_exact = EXACT THEN
X					replace_text := eve$do_substitution (
X										
			source_text, replacement);
X				ELSE
X					! Make sure non-alphabetic target is replaced b
y lowercase
X
X					IF this_occurrence = lowercase_target THEN
X						replace_text := eve$do_substitution (
X										source_
text, lowercase_replacement);
X					ELSE
X			 			IF this_occurrence = uppercase_target T
HEN
X							replace_text := eve$do_substitu
tion (
X										source_
text, uppercase_replacement);
X	  					ELSE
X							IF this_occurrence = capital_ta
rget THEN
X								replace_text := eve$do_
substitution (
X										source_
text, capital_replacement);
X							ELSE
X								replace_text := eve$do_
substitution (
X										source_
text, lowercase_replacement);
X							ENDIF;
X						ENDIF;
X					ENDIF;
X				ENDIF;
X				this_occurrence := ERASE_CHARACTER (LENGTH (replace_ran
ge));
X				COPY_TEXT (replace_text);
X
X				IF CURRENT_DIRECTION = REVERSE THEN
X					MOVE_HORIZONTAL (- LENGTH (replace_text));
X				ENDIF;
X
X				occurrences := occurrences + 1;
X				UPDATE (current_window);
X
X				IF (replace_action = SUBSTR ("all", 1, action_length)) 
AND
X		   								
			(action_length > 0) THEN
X					asking := 0;
X					MESSAGE ("Replacing all occurrences...");
X					SET (SCREEN_UPDATE, OFF);
X				ENDIF;
X				EXITIF 1;
X			ELSE
X				IF (replace_action = SUBSTR ("no", 1, action_length)) O
R
X		   			(replace_action = SUBSTR ("quit", 1, action_len
gth)) THEN
X					highlight_range := 0;
X					IF CURRENT_DIRECTION = FORWARD THEN
X						POSITION (END_OF (replace_range));
X						MOVE_HORIZONTAL (1);
X					ENDIF;
X					UPDATE (current_window);
X					EXITIF 1;
X				ENDIF;
X			ENDIF;
X		ENDLOOP;
X
X		EXITIF (action_length > 0) AND
X	   				((replace_action = SUBSTR ("quit", 1, action_le
ngth)) OR
X					(replace_action = SUBSTR ("last", 1, action_len
gth)));
X
X	ENDLOOP;
X
X	SET (SCREEN_UPDATE, ON);
X	MESSAGE (FAO ("Replaced !SL occurrence!%S", occurrences));
X	SET (this_mode, this_buffer);
X
XENDPROCEDURE;
X
X!
X!  Re and initialize some things.
X!
Xdefine_key ("eve_find ('',1)", pf1, " find", eve$x_vt100_keys);
Xdefine_key ("eve_find ('',1)", e1, " find", eve$x_standard_keys);
Xeve$arg2_find := eve$arg1_buffer;
Xeve$x_orig_target := eve$kt_null;
Xeve$x_paren_cnt := eve$kt_null;
Xeve$x_find_pat := eve$kt_null;
Xeve$x_glo_str := eve$kt_null;
X
Xeve$pch := 	ASCII(9); ! All printable characters.
X
Xi := 32;
XLOOP
X	EXITIF (i > 255);
X	eve$pch := eve$pch + ASCII (i);
X	i := i + 1;
XENDLOOP;
$ GoSub Convert_File
$ Exit