cpython/Lib/idlelib/IdleHistory.py

121 lines
4.5 KiB
Python
Raw Normal View History

"Implement Idle Shell history mechanism with History class"
from idlelib.configHandler import idleConf
2000-08-14 22:13:23 -03:00
class History:
''' Implement Idle Shell history mechanism.
2000-08-14 22:13:23 -03:00
store - Store source statement (called from PyShell.resetoutput).
fetch - Fetch stored statement matching prefix already entered.
history_next - Bound to <<history-next>> event (default Alt-N).
history_prev - Bound to <<history-prev>> event (default Alt-P).
'''
2000-08-14 22:13:23 -03:00
def __init__(self, text, output_sep = "\n"):
'''Initialize data attributes and bind event methods.
.text - Idle wrapper of tk Text widget, with .bell().
.history - source statements, possibly with multiple lines.
.prefix - source already entered at prompt; filters history list.
.pointer - index into history.
.cyclic - wrap around history list (or not).
'''
2000-08-14 22:13:23 -03:00
self.text = text
self.history = []
self.prefix = None
self.pointer = None
2000-08-14 22:13:23 -03:00
self.output_sep = output_sep
self.cyclic = idleConf.GetOption("main", "History", "cyclic", 1, "bool")
2000-08-14 22:13:23 -03:00
text.bind("<<history-previous>>", self.history_prev)
text.bind("<<history-next>>", self.history_next)
def history_next(self, event):
"Fetch later statement; start with ealiest if cyclic."
self.fetch(reverse=False)
2000-08-14 22:13:23 -03:00
return "break"
def history_prev(self, event):
"Fetch earlier statement; start with most recent."
self.fetch(reverse=True)
2000-08-14 22:13:23 -03:00
return "break"
def _get_source(self, start, end):
# Get source code from start index to end index. Lines in the
# text control may be separated by sys.ps2 .
lines = self.text.get(start, end).split(self.output_sep)
return "\n".join(lines)
2000-08-14 22:13:23 -03:00
def _put_source(self, where, source):
output = self.output_sep.join(source.split("\n"))
2000-08-14 22:13:23 -03:00
self.text.insert(where, output)
def fetch(self, reverse):
'''Fetch statememt and replace current line in text widget.
Set prefix and pointer as needed for successive fetches.
Reset them to None, None when returning to the start line.
Sound bell when return to start line or cannot leave a line
because cyclic is False.
'''
2000-08-14 22:13:23 -03:00
nhist = len(self.history)
pointer = self.pointer
prefix = self.prefix
2000-08-14 22:13:23 -03:00
if pointer is not None and prefix is not None:
if self.text.compare("insert", "!=", "end-1c") or \
self._get_source("iomark", "end-1c") != self.history[pointer]:
pointer = prefix = None
if pointer is None or prefix is None:
prefix = self._get_source("iomark", "end-1c")
if reverse:
pointer = nhist # will be decremented
2000-08-14 22:13:23 -03:00
else:
if self.cyclic:
pointer = -1 # will be incremented
else: # abort history_next
self.text.bell()
return
2000-08-14 22:13:23 -03:00
nprefix = len(prefix)
while 1:
if reverse:
pointer = pointer - 1
else:
pointer = pointer + 1
if pointer < 0 or pointer >= nhist:
self.text.bell()
if not self.cyclic and pointer < 0: # abort history_prev
return
else:
if self._get_source("iomark", "end-1c") != prefix:
self.text.delete("iomark", "end-1c")
self._put_source("iomark", prefix)
pointer = prefix = None
2000-08-14 22:13:23 -03:00
break
item = self.history[pointer]
if item[:nprefix] == prefix and len(item) > nprefix:
self.text.delete("iomark", "end-1c")
self._put_source("iomark", item)
break
self.text.mark_set("insert", "end-1c")
self.text.see("insert")
self.text.tag_remove("sel", "1.0", "end")
self.pointer = pointer
self.prefix = prefix
2000-08-14 22:13:23 -03:00
def store(self, source):
"Store Shell input statement into history list."
source = source.strip()
2000-08-14 22:13:23 -03:00
if len(source) > 2:
# avoid duplicates
try:
self.history.remove(source)
except ValueError:
pass
self.history.append(source)
self.pointer = None
self.prefix = None
if __name__ == "__main__":
from test import support
support.use_resources = ['gui']
from unittest import main
main('idlelib.idle_test.test_idlehistory', verbosity=2, exit=False)