mirror of
https://github.com/python/cpython.git
synced 2025-07-25 12:14:38 +00:00

About 10 IDLE features were implemented as supposedly optional extensions. Their different behavior could be confusing or worse for users and not good for maintenance. Hence the conversion. The main difference for users is that user configurable key bindings for builtin features are now handled uniformly. Now, editing a binding in a keyset only affects its value in the keyset. All bindings are defined together in the system-specific default keysets in config- extensions.def. All custom keysets are saved as a whole in config- extension.cfg. All take effect as soon as one clicks Apply or Ok. The affected events are '<<force-open-completions>>', '<<expand-word>>', '<<force-open-calltip>>', '<<flash-paren>>', '<<format-paragraph>>', '<<run-module>>', '<<check-module>>', and '<<zoom-height>>'. Any (global) customizations made before 3.6.3 will not affect their keyset- specific customization after 3.6.3. and vice versa. Inital patch by Charles Wohlganger, revised by Terry Jan Reedy.
96 lines
3.1 KiB
Python
96 lines
3.1 KiB
Python
'''Complete the current word before the cursor with words in the editor.
|
|
|
|
Each menu selection or shortcut key selection replaces the word with a
|
|
different word with the same prefix. The search for matches begins
|
|
before the target and moves toward the top of the editor. It then starts
|
|
after the cursor and moves down. It then returns to the original word and
|
|
the cycle starts again.
|
|
|
|
Changing the current text line or leaving the cursor in a different
|
|
place before requesting the next selection causes AutoExpand to reset
|
|
its state.
|
|
|
|
There is only one instance of Autoexpand.
|
|
'''
|
|
import re
|
|
import string
|
|
|
|
|
|
class AutoExpand:
|
|
wordchars = string.ascii_letters + string.digits + "_"
|
|
|
|
def __init__(self, editwin):
|
|
self.text = editwin.text
|
|
self.bell = self.text.bell
|
|
self.state = None
|
|
|
|
def expand_word_event(self, event):
|
|
"Replace the current word with the next expansion."
|
|
curinsert = self.text.index("insert")
|
|
curline = self.text.get("insert linestart", "insert lineend")
|
|
if not self.state:
|
|
words = self.getwords()
|
|
index = 0
|
|
else:
|
|
words, index, insert, line = self.state
|
|
if insert != curinsert or line != curline:
|
|
words = self.getwords()
|
|
index = 0
|
|
if not words:
|
|
self.bell()
|
|
return "break"
|
|
word = self.getprevword()
|
|
self.text.delete("insert - %d chars" % len(word), "insert")
|
|
newword = words[index]
|
|
index = (index + 1) % len(words)
|
|
if index == 0:
|
|
self.bell() # Warn we cycled around
|
|
self.text.insert("insert", newword)
|
|
curinsert = self.text.index("insert")
|
|
curline = self.text.get("insert linestart", "insert lineend")
|
|
self.state = words, index, curinsert, curline
|
|
return "break"
|
|
|
|
def getwords(self):
|
|
"Return a list of words that match the prefix before the cursor."
|
|
word = self.getprevword()
|
|
if not word:
|
|
return []
|
|
before = self.text.get("1.0", "insert wordstart")
|
|
wbefore = re.findall(r"\b" + word + r"\w+\b", before)
|
|
del before
|
|
after = self.text.get("insert wordend", "end")
|
|
wafter = re.findall(r"\b" + word + r"\w+\b", after)
|
|
del after
|
|
if not wbefore and not wafter:
|
|
return []
|
|
words = []
|
|
dict = {}
|
|
# search backwards through words before
|
|
wbefore.reverse()
|
|
for w in wbefore:
|
|
if dict.get(w):
|
|
continue
|
|
words.append(w)
|
|
dict[w] = w
|
|
# search onwards through words after
|
|
for w in wafter:
|
|
if dict.get(w):
|
|
continue
|
|
words.append(w)
|
|
dict[w] = w
|
|
words.append(word)
|
|
return words
|
|
|
|
def getprevword(self):
|
|
"Return the word prefix before the cursor."
|
|
line = self.text.get("insert linestart", "insert")
|
|
i = len(line)
|
|
while i > 0 and line[i-1] in self.wordchars:
|
|
i = i-1
|
|
return line[i:]
|
|
|
|
|
|
if __name__ == '__main__':
|
|
import unittest
|
|
unittest.main('idlelib.idle_test.test_autoexpand', verbosity=2)
|