Commit 506bf73d authored by Andrew M. Kuchling's avatar Andrew M. Kuchling

Patch #1048820 from Stefan Wehr: add insert-mode editing to Textbox.

Fix an off-by-one error I noticed.
parent 46fbc66a
...@@ -868,16 +868,19 @@ complete list of changes, or look through the CVS logs for all the details. ...@@ -868,16 +868,19 @@ complete list of changes, or look through the CVS logs for all the details.
.. Revision 57769 .. Revision 57769
* A new method in the :mod:`curses` module: for a window, :meth:`chgat` changes * A new method in the :mod:`curses` module: for a window, :meth:`chgat` changes
the display characters for a certain number of characters on a single line. the display characters for a certain number of characters on a single line.
(Contributed by Fabian Kreutz.)
:: ::
# Boldface text starting at y=0,x=21 # Boldface text starting at y=0,x=21
# and affecting the rest of the line. # and affecting the rest of the line.
stdscr.chgat(0,21, curses.A_BOLD) stdscr.chgat(0,21, curses.A_BOLD)
(Contributed by Fabian Kreutz.) The :class:`Textbox` class in the :mod:`curses.textpad` module
now supports editing in insert mode as well as overwrite mode.
Insert mode is enabled by supplying a true value for the *insert_mode*
parameter when creating the :class:`Textbox` instance.
* The :mod:`decimal` module was updated to version 1.66 of * The :mod:`decimal` module was updated to version 1.66 of
`the General Decimal Specification <http://www2.hursley.ibm.com/decimal/decarith.html>`__. New features `the General Decimal Specification <http://www2.hursley.ibm.com/decimal/decarith.html>`__. New features
......
...@@ -39,8 +39,9 @@ class Textbox: ...@@ -39,8 +39,9 @@ class Textbox:
KEY_LEFT = Ctrl-B, KEY_RIGHT = Ctrl-F, KEY_UP = Ctrl-P, KEY_DOWN = Ctrl-N KEY_LEFT = Ctrl-B, KEY_RIGHT = Ctrl-F, KEY_UP = Ctrl-P, KEY_DOWN = Ctrl-N
KEY_BACKSPACE = Ctrl-h KEY_BACKSPACE = Ctrl-h
""" """
def __init__(self, win): def __init__(self, win, insert_mode=False):
self.win = win self.win = win
self.insert_mode = insert_mode
(self.maxy, self.maxx) = win.getmaxyx() (self.maxy, self.maxx) = win.getmaxyx()
self.maxy = self.maxy - 1 self.maxy = self.maxy - 1
self.maxx = self.maxx - 1 self.maxx = self.maxx - 1
...@@ -49,9 +50,10 @@ class Textbox: ...@@ -49,9 +50,10 @@ class Textbox:
win.keypad(1) win.keypad(1)
def _end_of_line(self, y): def _end_of_line(self, y):
"Go to the location of the first blank on the given line." """Go to the location of the first blank on the given line,
returning the index of the last non-blank character."""
last = self.maxx last = self.maxx
while 1: while True:
if ascii.ascii(self.win.inch(y, last)) != ascii.SP: if ascii.ascii(self.win.inch(y, last)) != ascii.SP:
last = min(self.maxx, last+1) last = min(self.maxx, last+1)
break break
...@@ -60,12 +62,11 @@ class Textbox: ...@@ -60,12 +62,11 @@ class Textbox:
last = last - 1 last = last - 1
return last return last
def do_command(self, ch): def _insert_printable_char(self, ch):
"Process a single editing command."
(y, x) = self.win.getyx() (y, x) = self.win.getyx()
self.lastcmd = ch
if ascii.isprint(ch):
if y < self.maxy or x < self.maxx: 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 # The try-catch ignores the error we trigger from some curses
# versions by trying to write into the lowest-rightmost spot # versions by trying to write into the lowest-rightmost spot
# in the window. # in the window.
...@@ -73,6 +74,19 @@ class Textbox: ...@@ -73,6 +74,19 @@ class Textbox:
self.win.addch(ch) self.win.addch(ch)
except curses.error: except curses.error:
pass pass
if self.insert_mode:
(backy, backx) = self.win.getyx()
if ascii.isprint(oldch):
self._insert_printable_char(oldch)
self.win.move(backy, backx)
def do_command(self, ch):
"Process a single editing command."
(y, x) = self.win.getyx()
self.lastcmd = ch
if ascii.isprint(ch):
if y < self.maxy or x < self.maxx:
self._insert_printable_char(ch)
elif ch == ascii.SOH: # ^a elif ch == ascii.SOH: # ^a
self.win.move(y, 0) self.win.move(y, 0)
elif ch in (ascii.STX,curses.KEY_LEFT, ascii.BS,curses.KEY_BACKSPACE): elif ch in (ascii.STX,curses.KEY_LEFT, ascii.BS,curses.KEY_BACKSPACE):
...@@ -139,7 +153,7 @@ class Textbox: ...@@ -139,7 +153,7 @@ class Textbox:
if stop == 0 and self.stripspaces: if stop == 0 and self.stripspaces:
continue continue
for x in range(self.maxx+1): for x in range(self.maxx+1):
if self.stripspaces and x == stop: if self.stripspaces and x > stop:
break break
result = result + chr(ascii.ascii(self.win.inch(y, x))) result = result + chr(ascii.ascii(self.win.inch(y, x)))
if self.maxy > 0: if self.maxy > 0:
......
...@@ -696,6 +696,7 @@ Bob Watson ...@@ -696,6 +696,7 @@ Bob Watson
Aaron Watters Aaron Watters
Henrik Weber Henrik Weber
Corran Webster Corran Webster
Stefan Wehr
Zack Weinberg Zack Weinberg
Edward Welbourne Edward Welbourne
Cliff Wells Cliff Wells
......
...@@ -390,6 +390,10 @@ Library ...@@ -390,6 +390,10 @@ Library
- #1837: The queue module now also supports a LIFO queue and a priority queue. - #1837: The queue module now also supports a LIFO queue and a priority queue.
- Patch #1048820: Add insert-mode editing to curses.textpad.Textbox
(patch by Stefan Wehr). Also, fix an off-by-one bug in
Textbox.gather().
- Issue #1831: ctypes now raises a TypeError if conflicting positional - Issue #1831: ctypes now raises a TypeError if conflicting positional
and named arguments are passed to a Structure or Union initializer. and named arguments are passed to a Structure or Union initializer.
When too many positional arguments are passed, also a TypeError is When too many positional arguments are passed, also a TypeError is
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment