| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | /* parens.c -- Implementation of matching parentheses feature. */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Copyright (C) 1987, 1989, 1992 Free Software Foundation, Inc.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |    This file is part of the GNU Readline Library, a library for | 
					
						
							|  |  |  |    reading lines of text with interactive input and history editing. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |    The GNU Readline Library 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 1, or | 
					
						
							|  |  |  |    (at your option) any later version. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |    The GNU Readline Library 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. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |    The GNU General Public License is often shipped with GNU software, and | 
					
						
							|  |  |  |    is generally kept in a file called COPYING or LICENSE.  If you do not | 
					
						
							|  |  |  |    have a copy of the license, write to the Free Software Foundation, | 
					
						
							|  |  |  |    675 Mass Ave, Cambridge, MA 02139, USA. */ | 
					
						
							|  |  |  | #define READLINE_LIBRARY
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "rlconf.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if !defined (PAREN_MATCHING)
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | extern int rl_insert (); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | int | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | rl_insert_close (count, invoking_key) | 
					
						
							|  |  |  |      int count, invoking_key; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return (rl_insert (count, invoking_key)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #else /* PAREN_MATCHING */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (HAVE_CONFIG_H)
 | 
					
						
							|  |  |  | #  include <config.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <sys/types.h>
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if defined (FD_SET) && !defined (HAVE_SELECT)
 | 
					
						
							|  |  |  | #  define HAVE_SELECT
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (HAVE_SELECT)
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | #  include <sys/time.h>
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #endif /* HAVE_SELECT */
 | 
					
						
							|  |  |  | #if defined (HAVE_SYS_SELECT_H)
 | 
					
						
							|  |  |  | #  include <sys/select.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (HAVE_STRING_H)
 | 
					
						
							|  |  |  | #  include <string.h>
 | 
					
						
							|  |  |  | #else /* !HAVE_STRING_H */
 | 
					
						
							|  |  |  | #  include <strings.h>
 | 
					
						
							|  |  |  | #endif /* !HAVE_STRING_H */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if !defined (strchr) && !defined (__STDC__)
 | 
					
						
							|  |  |  | extern char *strchr (), *strrchr (); | 
					
						
							|  |  |  | #endif /* !strchr && !__STDC__ */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | #include "readline.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | extern int rl_explicit_arg; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Non-zero means try to blink the matching open parenthesis when the
 | 
					
						
							|  |  |  |    close parenthesis is inserted. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (HAVE_SELECT)
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | int rl_blink_matching_paren = 1; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #else /* !HAVE_SELECT */
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | int rl_blink_matching_paren = 0; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #endif /* !HAVE_SELECT */
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static int find_matching_open (); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | int | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | rl_insert_close (count, invoking_key) | 
					
						
							|  |  |  |      int count, invoking_key; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (rl_explicit_arg || !rl_blink_matching_paren) | 
					
						
							|  |  |  |     rl_insert (count, invoking_key); | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (HAVE_SELECT)
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       int orig_point, match_point, ready; | 
					
						
							|  |  |  |       struct timeval timer; | 
					
						
							|  |  |  |       fd_set readfds; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       rl_insert (1, invoking_key); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       (*rl_redisplay_function) (); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       match_point = | 
					
						
							|  |  |  | 	find_matching_open (rl_line_buffer, rl_point - 2, invoking_key); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* Emacs might message or ring the bell here, but I don't. */ | 
					
						
							|  |  |  |       if (match_point < 0) | 
					
						
							|  |  |  | 	return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       FD_ZERO (&readfds); | 
					
						
							|  |  |  |       FD_SET (fileno (rl_instream), &readfds); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       timer.tv_sec = 0; | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  |       timer.tv_usec = 500000; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       orig_point = rl_point; | 
					
						
							|  |  |  |       rl_point = match_point; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       (*rl_redisplay_function) (); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       ready = select (1, &readfds, (fd_set *)NULL, (fd_set *)NULL, &timer); | 
					
						
							|  |  |  |       rl_point = orig_point; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #else /* !HAVE_SELECT */
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       rl_insert (count, invoking_key); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #endif /* !HAVE_SELECT */
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | find_matching_open (string, from, closer) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int from, closer; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register int i; | 
					
						
							|  |  |  |   int opener, level, delimiter; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch (closer) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     case ']': opener = '['; break; | 
					
						
							|  |  |  |     case '}': opener = '{'; break; | 
					
						
							|  |  |  |     case ')': opener = '('; break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       return (-1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   level = 1;			/* The closer passed in counts as 1. */ | 
					
						
							|  |  |  |   delimiter = 0;		/* Delimited state unknown. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (i = from; i > -1; i--) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (delimiter && (string[i] == delimiter)) | 
					
						
							|  |  |  | 	delimiter = 0; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       else if (rl_basic_quote_characters && strchr (rl_basic_quote_characters, string[i])) | 
					
						
							|  |  |  | 	delimiter = string[i]; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       else if (!delimiter && (string[i] == closer)) | 
					
						
							|  |  |  | 	level++; | 
					
						
							|  |  |  |       else if (!delimiter && (string[i] == opener)) | 
					
						
							|  |  |  | 	level--; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (!level) | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return (i); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif /* PAREN_MATCHING */
 |