- Notifications
You must be signed in to change notification settings - Fork 31.7k
/
Copy pathcodecontext.py
270 lines (233 loc) · 11.2 KB
/
codecontext.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
"""codecontext - display the block context above the edit window
Once code has scrolled off the top of a window, it can be difficult to
determine which block you are in. This extension implements a pane at the top
of each IDLE edit window which provides block structure hints. These hints are
the lines which contain the block opening keywords, e.g. 'if', for the
enclosing block. The number of hint lines is determined by the maxlines
variable in the codecontext section of config-extensions.def. Lines which do
not open blocks are not shown in the context hints pane.
For EditorWindows, <<toggle-code-context>> is bound to CodeContext(self).
toggle_code_context_event.
"""
importre
fromsysimportmaxsizeasINFINITY
fromtkinterimportFrame, Text, TclError
fromtkinter.constantsimportNSEW, SUNKEN
fromidlelib.configimportidleConf
BLOCKOPENERS= {'class', 'def', 'if', 'elif', 'else', 'while', 'for',
'try', 'except', 'finally', 'with', 'async'}
defget_spaces_firstword(codeline, c=re.compile(r"^(\s*)(\w*)")):
"Extract the beginning whitespace and first word from codeline."
returnc.match(codeline).groups()
defget_line_info(codeline):
"""Return tuple of (line indent value, codeline, block start keyword).
The indentation of empty lines (or comment lines) is INFINITY.
If the line does not start a block, the keyword value is False.
"""
spaces, firstword=get_spaces_firstword(codeline)
indent=len(spaces)
iflen(codeline) ==indentorcodeline[indent] =='#':
indent=INFINITY
opener=firstwordinBLOCKOPENERSandfirstword
returnindent, codeline, opener
classCodeContext:
"Display block context above the edit window."
UPDATEINTERVAL=100# millisec
def__init__(self, editwin):
"""Initialize settings for context block.
editwin is the Editor window for the context block.
self.text is the editor window text widget.
self.context displays the code context text above the editor text.
Initially None, it is toggled via <<toggle-code-context>>.
self.topvisible is the number of the top text line displayed.
self.info is a list of (line number, indent level, line text,
block keyword) tuples for the block structure above topvisible.
self.info[0] is initialized with a 'dummy' line which
starts the toplevel 'block' of the module.
self.t1 and self.t2 are two timer events on the editor text widget to
monitor for changes to the context text or editor font.
"""
self.editwin=editwin
self.text=editwin.text
self._reset()
def_reset(self):
self.context=None
self.cell00=None
self.t1=None
self.topvisible=1
self.info= [(0, -1, "", False)]
@classmethod
defreload(cls):
"Load class variables from config."
cls.context_depth=idleConf.GetOption("extensions", "CodeContext",
"maxlines", type="int",
default=15)
def__del__(self):
"Cancel scheduled events."
ifself.t1isnotNone:
try:
self.text.after_cancel(self.t1)
exceptTclError: # pragma: no cover
pass
self.t1=None
deftoggle_code_context_event(self, event=None):
"""Toggle code context display.
If self.context doesn't exist, create it to match the size of the editor
window text (toggle on). If it does exist, destroy it (toggle off).
Return 'break' to complete the processing of the binding.
"""
ifself.contextisNone:
# Calculate the border width and horizontal padding required to
# align the context with the text in the main Text widget.
#
# All values are passed through getint(), since some
# values may be pixel objects, which can't simply be added to ints.
widgets=self.editwin.text, self.editwin.text_frame
# Calculate the required horizontal padding and border width.
padx=0
border=0
forwidgetinwidgets:
info= (widget.grid_info()
ifwidgetisself.editwin.text
elsewidget.pack_info())
padx+=widget.tk.getint(info['padx'])
padx+=widget.tk.getint(widget.cget('padx'))
border+=widget.tk.getint(widget.cget('border'))
context=self.context=Text(
self.editwin.text_frame,
height=1,
width=1, # Don't request more than we get.
highlightthickness=0,
padx=padx, border=border, relief=SUNKEN, state='disabled')
self.update_font()
self.update_highlight_colors()
context.bind('<ButtonRelease-1>', self.jumptoline)
# Get the current context and initiate the recurring update event.
self.timer_event()
# Grid the context widget above the text widget.
context.grid(row=0, column=1, sticky=NSEW)
line_number_colors=idleConf.GetHighlight(idleConf.CurrentTheme(),
'linenumber')
self.cell00=Frame(self.editwin.text_frame,
bg=line_number_colors['background'])
self.cell00.grid(row=0, column=0, sticky=NSEW)
menu_status='Hide'
else:
self.context.destroy()
self.context=None
self.cell00.destroy()
self.cell00=None
self.text.after_cancel(self.t1)
self._reset()
menu_status='Show'
self.editwin.update_menu_label(menu='options', index='*ode*ontext',
label=f'{menu_status} Code Context')
return"break"
defget_context(self, new_topvisible, stopline=1, stopindent=0):
"""Return a list of block line tuples and the 'last' indent.
The tuple fields are (linenum, indent, text, opener).
The list represents header lines from new_topvisible back to
stopline with successively shorter indents > stopindent.
The list is returned ordered by line number.
Last indent returned is the smallest indent observed.
"""
assertstopline>0
lines= []
# The indentation level we are currently in.
lastindent=INFINITY
# For a line to be interesting, it must begin with a block opening
# keyword, and have less indentation than lastindent.
forlinenuminrange(new_topvisible, stopline-1, -1):
codeline=self.text.get(f'{linenum}.0', f'{linenum}.end')
indent, text, opener=get_line_info(codeline)
ifindent<lastindent:
lastindent=indent
ifopenerin ("else", "elif"):
# Also show the if statement.
lastindent+=1
ifopenerandlinenum<new_topvisibleandindent>=stopindent:
lines.append((linenum, indent, text, opener))
iflastindent<=stopindent:
break
lines.reverse()
returnlines, lastindent
defupdate_code_context(self):
"""Update context information and lines visible in the context pane.
No update is done if the text hasn't been scrolled. If the text
was scrolled, the lines that should be shown in the context will
be retrieved and the context area will be updated with the code,
up to the number of maxlines.
"""
new_topvisible=self.editwin.getlineno("@0,0")
ifself.topvisible==new_topvisible: # Haven't scrolled.
return
ifself.topvisible<new_topvisible: # Scroll down.
lines, lastindent=self.get_context(new_topvisible,
self.topvisible)
# Retain only context info applicable to the region
# between topvisible and new_topvisible.
whileself.info[-1][1] >=lastindent:
delself.info[-1]
else: # self.topvisible > new_topvisible: # Scroll up.
stopindent=self.info[-1][1] +1
# Retain only context info associated
# with lines above new_topvisible.
whileself.info[-1][0] >=new_topvisible:
stopindent=self.info[-1][1]
delself.info[-1]
lines, lastindent=self.get_context(new_topvisible,
self.info[-1][0]+1,
stopindent)
self.info.extend(lines)
self.topvisible=new_topvisible
# Last context_depth context lines.
context_strings= [x[2] forxinself.info[-self.context_depth:]]
showfirst=0ifcontext_strings[0] else1
# Update widget.
self.context['height'] =len(context_strings) -showfirst
self.context['state'] ='normal'
self.context.delete('1.0', 'end')
self.context.insert('end', '\n'.join(context_strings[showfirst:]))
self.context['state'] ='disabled'
defjumptoline(self, event=None):
""" Show clicked context line at top of editor.
If a selection was made, don't jump; allow copying.
If no visible context, show the top line of the file.
"""
try:
self.context.index("sel.first")
exceptTclError:
lines=len(self.info)
iflines==1: # No context lines are showing.
newtop=1
else:
# Line number clicked.
contextline=int(float(self.context.index('insert')))
# Lines not displayed due to maxlines.
offset=max(1, lines-self.context_depth) -1
newtop=self.info[offset+contextline][0]
self.text.yview(f'{newtop}.0')
self.update_code_context()
deftimer_event(self):
"Event on editor text widget triggered every UPDATEINTERVAL ms."
ifself.contextisnotNone:
self.update_code_context()
self.t1=self.text.after(self.UPDATEINTERVAL, self.timer_event)
defupdate_font(self):
ifself.contextisnotNone:
font=idleConf.GetFont(self.text, 'main', 'EditorWindow')
self.context['font'] =font
defupdate_highlight_colors(self):
ifself.contextisnotNone:
colors=idleConf.GetHighlight(idleConf.CurrentTheme(), 'context')
self.context['background'] =colors['background']
self.context['foreground'] =colors['foreground']
ifself.cell00isnotNone:
line_number_colors=idleConf.GetHighlight(idleConf.CurrentTheme(),
'linenumber')
self.cell00.config(bg=line_number_colors['background'])
CodeContext.reload()
if__name__=="__main__":
fromunittestimportmain
main('idlelib.idle_test.test_codecontext', verbosity=2, exit=False)
# Add htest.