| 
									
										
										
										
											2000-07-11 10:38:24 +00:00
										 |  |  | """Simple textbox editing widget with Emacs-like keybindings.""" | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-05 20:39:59 +00:00
										 |  |  | import curses | 
					
						
							|  |  |  | import curses.ascii | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | def rectangle(win, uly, ulx, lry, lrx): | 
					
						
							| 
									
										
										
										
											2004-10-19 19:21:20 +00:00
										 |  |  |     """Draw a rectangle with corners at the provided upper-left
 | 
					
						
							|  |  |  |     and lower-right coordinates. | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |     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
										 |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2008-01-20 00:00:38 +00:00
										 |  |  |     def __init__(self, win, insert_mode=False): | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |         self.win = win | 
					
						
							| 
									
										
										
										
											2008-01-20 00:00:38 +00:00
										 |  |  |         self.insert_mode = insert_mode | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |         (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): | 
					
						
							| 
									
										
										
										
											2008-01-20 00:00:38 +00:00
										 |  |  |         """Go to the location of the first blank on the given line,
 | 
					
						
							|  |  |  |         returning the index of the last non-blank character."""
 | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |         last = self.maxx | 
					
						
							| 
									
										
										
										
											2008-01-20 00:00:38 +00:00
										 |  |  |         while True: | 
					
						
							| 
									
										
										
										
											2008-07-05 20:39:59 +00:00
										 |  |  |             if curses.ascii.ascii(self.win.inch(y, last)) != curses.ascii.SP: | 
					
						
							| 
									
										
										
										
											2005-06-02 00:10:04 +00:00
										 |  |  |                 last = min(self.maxx, last+1) | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |                 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 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-01-20 00:00:38 +00:00
										 |  |  |     def _insert_printable_char(self, ch): | 
					
						
							|  |  |  |         (y, x) = self.win.getyx() | 
					
						
							|  |  |  |         if y < self.maxy or x < self.maxx: | 
					
						
							|  |  |  |             if self.insert_mode: | 
					
						
							|  |  |  |                 oldch = self.win.inch() | 
					
						
							|  |  |  |             # The try-catch ignores the error we trigger from some curses | 
					
						
							|  |  |  |             # versions by trying to write into the lowest-rightmost spot | 
					
						
							|  |  |  |             # in the window. | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 self.win.addch(ch) | 
					
						
							|  |  |  |             except curses.error: | 
					
						
							|  |  |  |                 pass | 
					
						
							|  |  |  |             if self.insert_mode: | 
					
						
							|  |  |  |                 (backy, backx) = self.win.getyx() | 
					
						
							| 
									
										
										
										
											2008-07-05 20:39:59 +00:00
										 |  |  |                 if curses.ascii.isprint(oldch): | 
					
						
							| 
									
										
										
										
											2008-01-20 00:00:38 +00:00
										 |  |  |                     self._insert_printable_char(oldch) | 
					
						
							|  |  |  |                     self.win.move(backy, backx) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |     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 | 
					
						
							| 
									
										
										
										
											2008-07-05 20:39:59 +00:00
										 |  |  |         if curses.ascii.isprint(ch): | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             if y < self.maxy or x < self.maxx: | 
					
						
							| 
									
										
										
										
											2008-01-20 00:00:38 +00:00
										 |  |  |                 self._insert_printable_char(ch) | 
					
						
							| 
									
										
										
										
											2008-07-05 20:39:59 +00:00
										 |  |  |         elif ch == curses.ascii.SOH:                           # ^a | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             self.win.move(y, 0) | 
					
						
							| 
									
										
										
										
											2008-07-05 20:39:59 +00:00
										 |  |  |         elif ch in (curses.ascii.STX,curses.KEY_LEFT, curses.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) | 
					
						
							| 
									
										
										
										
											2008-07-05 20:39:59 +00:00
										 |  |  |             if ch in (curses.ascii.BS, curses.KEY_BACKSPACE): | 
					
						
							| 
									
										
										
										
											2000-06-27 00:53:12 +00:00
										 |  |  |                 self.win.delch() | 
					
						
							| 
									
										
										
										
											2008-07-05 20:39:59 +00:00
										 |  |  |         elif ch == curses.ascii.EOT:                           # ^d | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             self.win.delch() | 
					
						
							| 
									
										
										
										
											2008-07-05 20:39:59 +00:00
										 |  |  |         elif ch == curses.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) | 
					
						
							| 
									
										
										
										
											2008-07-05 20:39:59 +00:00
										 |  |  |         elif ch in (curses.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) | 
					
						
							| 
									
										
										
										
											2008-07-05 20:39:59 +00:00
										 |  |  |         elif ch == curses.ascii.BEL:                           # ^g | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             return 0 | 
					
						
							| 
									
										
										
										
											2008-07-05 20:39:59 +00:00
										 |  |  |         elif ch == curses.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) | 
					
						
							| 
									
										
										
										
											2008-07-05 20:39:59 +00:00
										 |  |  |         elif ch == curses.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: | 
					
						
							| 
									
										
										
										
											2004-10-19 19:29:40 +00:00
										 |  |  |                 # first undo the effect of self._end_of_line | 
					
						
							|  |  |  |                 self.win.move(y, x) | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |                 self.win.clrtoeol() | 
					
						
							| 
									
										
										
										
											2008-07-05 20:39:59 +00:00
										 |  |  |         elif ch == curses.ascii.FF:                            # ^l | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             self.win.refresh() | 
					
						
							| 
									
										
										
										
											2008-07-05 20:39:59 +00:00
										 |  |  |         elif ch in (curses.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)) | 
					
						
							| 
									
										
										
										
											2008-07-05 20:39:59 +00:00
										 |  |  |         elif ch == curses.ascii.SI:                            # ^o | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             self.win.insertln() | 
					
						
							| 
									
										
										
										
											2008-07-05 20:39:59 +00:00
										 |  |  |         elif ch in (curses.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 | 
					
						
							| 
									
										
										
										
											2002-09-29 00:25:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |     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) | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             if stop == 0 and self.stripspaces: | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             for x in range(self.maxx+1): | 
					
						
							| 
									
										
										
										
											2008-01-20 00:00:38 +00:00
										 |  |  |                 if self.stripspaces and x > stop: | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |                     break | 
					
						
							| 
									
										
										
										
											2008-07-05 20:39:59 +00:00
										 |  |  |                 result = result + chr(curses.ascii.ascii(self.win.inch(y, x))) | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |             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): | 
					
						
							| 
									
										
										
										
											2004-10-19 19:21:20 +00:00
										 |  |  |         ncols, nlines = 9, 4 | 
					
						
							|  |  |  |         uly, ulx = 15, 20 | 
					
						
							| 
									
										
										
										
											2004-10-19 19:36:09 +00:00
										 |  |  |         stdscr.addstr(uly-2, ulx, "Use Ctrl-G to end editing.") | 
					
						
							| 
									
										
										
										
											2004-10-19 19:21:20 +00:00
										 |  |  |         win = curses.newwin(nlines, ncols, uly, ulx) | 
					
						
							|  |  |  |         rectangle(stdscr, uly-1, ulx-1, uly + nlines, ulx + ncols) | 
					
						
							| 
									
										
										
										
											2000-06-26 23:55:42 +00:00
										 |  |  |         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) | 
					
						
							| 
									
										
										
										
											2004-10-19 19:36:09 +00:00
										 |  |  |     print 'Contents of text box:', repr(str) |