| 
									
										
										
										
											2000-07-11 10:38:24 +00:00
										 |  |  | """Simple textbox editing widget with Emacs-like keybindings.""" | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | import sys, curses, ascii | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def rectangle(win, uly, ulx, lry, lrx): | 
					
						
							|  |  |  |     "Draw a rectangle." | 
					
						
							|  |  |  |     win.vline(uly+1, ulx, curses.ACS_VLINE, lry - uly - 1) | 
					
						
							|  |  |  |     win.hline(uly, ulx+1, curses.ACS_HLINE, lrx - ulx - 1) | 
					
						
							|  |  |  |     win.hline(lry, ulx+1, curses.ACS_HLINE, lrx - ulx - 1) | 
					
						
							|  |  |  |     win.vline(uly+1, lrx, curses.ACS_VLINE, lry - uly - 1) | 
					
						
							|  |  |  |     win.addch(uly, ulx, curses.ACS_ULCORNER) | 
					
						
							|  |  |  |     win.addch(uly, lrx, curses.ACS_URCORNER) | 
					
						
							|  |  |  |     win.addch(lry, lrx, curses.ACS_LRCORNER) | 
					
						
							|  |  |  |     win.addch(lry, ulx, curses.ACS_LLCORNER) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  | class Textbox: | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |     """Editing widget using the interior of a window object.
 | 
					
						
							|  |  |  |      Supports the following Emacs-like key bindings: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Ctrl-A      Go to left edge of window. | 
					
						
							|  |  |  |     Ctrl-B      Cursor left, wrapping to previous line if appropriate. | 
					
						
							|  |  |  |     Ctrl-D      Delete character under cursor. | 
					
						
							| 
									
										
										
										
											2000-08-04 07:33:18 +00:00
										 |  |  |     Ctrl-E      Go to right edge (stripspaces off) or end of line (stripspaces on). | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |     Ctrl-F      Cursor right, wrapping to next line when appropriate. | 
					
						
							|  |  |  |     Ctrl-G      Terminate, returning the window contents. | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |     Ctrl-H      Delete character backward. | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |     Ctrl-J      Terminate if the window is 1 line, otherwise insert newline. | 
					
						
							|  |  |  |     Ctrl-K      If line is blank, delete it, otherwise clear to end of line. | 
					
						
							| 
									
										
										
										
											2000-08-04 07:33:18 +00:00
										 |  |  |     Ctrl-L      Refresh screen. | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |     Ctrl-N      Cursor down; move down one line. | 
					
						
							|  |  |  |     Ctrl-O      Insert a blank line at cursor location. | 
					
						
							|  |  |  |     Ctrl-P      Cursor up; move up one line. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Move operations do nothing if the cursor is at an edge where the movement | 
					
						
							|  |  |  |     is not possible.  The following synonyms are supported where possible: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     KEY_LEFT = Ctrl-B, KEY_RIGHT = Ctrl-F, KEY_UP = Ctrl-P, KEY_DOWN = Ctrl-N | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |     KEY_BACKSPACE = Ctrl-h | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |     """
 | 
					
						
							|  |  |  |     def __init__(self, win): | 
					
						
							|  |  |  |         self.win = win | 
					
						
							|  |  |  |         (self.maxy, self.maxx) = win.getmaxyx() | 
					
						
							|  |  |  |         self.maxy = self.maxy - 1 | 
					
						
							|  |  |  |         self.maxx = self.maxx - 1 | 
					
						
							|  |  |  |         self.stripspaces = 1 | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |         self.lastcmd = None | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |         win.keypad(1) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-04 07:33:18 +00:00
										 |  |  |     def _end_of_line(self, y): | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |         "Go to the location of the first blank on the given line." | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |         last = self.maxx | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |         while 1: | 
					
						
							|  |  |  |             if ascii.ascii(self.win.inch(y, last)) != ascii.SP: | 
					
						
							|  |  |  |                 last = last + 1 | 
					
						
							|  |  |  |                 break | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |             elif last == 0: | 
					
						
							|  |  |  |                 break | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             last = last - 1 | 
					
						
							|  |  |  |         return last | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def do_command(self, ch): | 
					
						
							|  |  |  |         "Process a single editing command." | 
					
						
							|  |  |  |         (y, x) = self.win.getyx() | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |         self.lastcmd = ch | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |         if ascii.isprint(ch): | 
					
						
							|  |  |  |             if y < self.maxy or x < self.maxx: | 
					
						
							|  |  |  |                 # The try-catch ignores the error we trigger from some curses | 
					
						
							|  |  |  |                 # versions by trying to write into the lowest-rightmost spot | 
					
						
							| 
									
										
										
										
											2001-08-13 13:47:23 +00:00
										 |  |  |                 # in the window. | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |                 try: | 
					
						
							|  |  |  |                     self.win.addch(ch) | 
					
						
							| 
									
										
										
										
											2001-08-13 13:47:23 +00:00
										 |  |  |                 except curses.error: | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |                     pass | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |         elif ch == ascii.SOH:				# ^a | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             self.win.move(y, 0) | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |         elif ch in (ascii.STX,curses.KEY_LEFT, ascii.BS,curses.KEY_BACKSPACE): | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             if x > 0: | 
					
						
							|  |  |  |                 self.win.move(y, x-1) | 
					
						
							|  |  |  |             elif y == 0: | 
					
						
							|  |  |  |                 pass | 
					
						
							|  |  |  |             elif self.stripspaces: | 
					
						
							| 
									
										
										
										
											2000-08-04 07:33:18 +00:00
										 |  |  |                 self.win.move(y-1, self._end_of_line(y-1)) | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             else: | 
					
						
							|  |  |  |                 self.win.move(y-1, self.maxx) | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |             if ch in (ascii.BS, curses.KEY_BACKSPACE): | 
					
						
							|  |  |  |                 self.win.delch() | 
					
						
							|  |  |  |         elif ch == ascii.EOT:				# ^d | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             self.win.delch() | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |         elif ch == ascii.ENQ:				# ^e | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             if self.stripspaces: | 
					
						
							| 
									
										
										
										
											2000-08-04 07:33:18 +00:00
										 |  |  |                 self.win.move(y, self._end_of_line(y)) | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             else: | 
					
						
							|  |  |  |                 self.win.move(y, self.maxx) | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |         elif ch in (ascii.ACK, curses.KEY_RIGHT):	# ^f | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             if x < self.maxx: | 
					
						
							|  |  |  |                 self.win.move(y, x+1) | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |             elif y == self.maxy: | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |                 pass | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 self.win.move(y+1, 0) | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |         elif ch == ascii.BEL:				# ^g | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             return 0 | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |         elif ch == ascii.NL:				# ^j | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             if self.maxy == 0: | 
					
						
							|  |  |  |                 return 0 | 
					
						
							|  |  |  |             elif y < self.maxy: | 
					
						
							|  |  |  |                 self.win.move(y+1, 0) | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |         elif ch == ascii.VT:				# ^k | 
					
						
							| 
									
										
										
										
											2000-08-04 07:33:18 +00:00
										 |  |  |             if x == 0 and self._end_of_line(y) == 0: | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |                 self.win.deleteln() | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 self.win.clrtoeol() | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |         elif ch == ascii.FF:				# ^l | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             self.win.refresh() | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |         elif ch in (ascii.SO, curses.KEY_DOWN):		# ^n | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             if y < self.maxy: | 
					
						
							|  |  |  |                 self.win.move(y+1, x) | 
					
						
							| 
									
										
										
										
											2000-08-04 07:33:18 +00:00
										 |  |  |                 if x > self._end_of_line(y+1): | 
					
						
							|  |  |  |                     self.win.move(y+1, self._end_of_line(y+1)) | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |         elif ch == ascii.SI:				# ^o | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             self.win.insertln() | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |         elif ch in (ascii.DLE, curses.KEY_UP):		# ^p | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             if y > 0: | 
					
						
							|  |  |  |                 self.win.move(y-1, x) | 
					
						
							| 
									
										
										
										
											2000-08-04 07:33:18 +00:00
										 |  |  |                 if x > self._end_of_line(y-1): | 
					
						
							|  |  |  |                     self.win.move(y-1, self._end_of_line(y-1)) | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |         return 1 | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |     def gather(self): | 
					
						
							|  |  |  |         "Collect and return the contents of the window." | 
					
						
							|  |  |  |         result = "" | 
					
						
							|  |  |  |         for y in range(self.maxy+1): | 
					
						
							|  |  |  |             self.win.move(y, 0) | 
					
						
							| 
									
										
										
										
											2000-08-04 07:33:18 +00:00
										 |  |  |             stop = self._end_of_line(y) | 
					
						
							|  |  |  |             #sys.stderr.write("y=%d, _end_of_line(y)=%d\n" % (y, stop)) | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             if stop == 0 and self.stripspaces: | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             for x in range(self.maxx+1): | 
					
						
							|  |  |  |                 if self.stripspaces and x == stop: | 
					
						
							|  |  |  |                     break | 
					
						
							|  |  |  |                 result = result + chr(ascii.ascii(self.win.inch(y, x))) | 
					
						
							|  |  |  |             if self.maxy > 0: | 
					
						
							|  |  |  |                 result = result + "\n" | 
					
						
							|  |  |  |         return result | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def edit(self, validate=None): | 
					
						
							|  |  |  |         "Edit in the widget window and collect the results." | 
					
						
							|  |  |  |         while 1: | 
					
						
							|  |  |  |             ch = self.win.getch() | 
					
						
							|  |  |  |             if validate: | 
					
						
							|  |  |  |                 ch = validate(ch) | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |             if not ch: | 
					
						
							|  |  |  |                 continue | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             if not self.do_command(ch): | 
					
						
							|  |  |  |                 break | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |             self.win.refresh() | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |         return self.gather() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if __name__ == '__main__': | 
					
						
							|  |  |  |     def test_editbox(stdscr): | 
					
						
							|  |  |  |         win = curses.newwin(4, 9, 15, 20) | 
					
						
							|  |  |  |         rectangle(stdscr, 14, 19, 19, 29) | 
					
						
							|  |  |  |         stdscr.refresh() | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |         return Textbox(win).edit() | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     str = curses.wrapper(test_editbox) | 
					
						
							|  |  |  |     print str |