-
-
Notifications
You must be signed in to change notification settings - Fork 31.5k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
gh-77578: IDLE Help - let users control font size #6665
base: main
Are you sure you want to change the base?
Conversation
csabella
commented
Apr 30, 2018
•
edited by terryjreedy
Loading
edited by terryjreedy
- Get initial font size from configuration.
- Allow for dynamically changing font size via keyboard or mousewheel.
- Issue: IDLE help viewer: let users control font size #77578
Lib/idlelib/idle_test/test_help.py
Outdated
@@ -30,5 +40,75 @@ def test_line1(self): | |||
text = self.frame.text | |||
self.assertEqual(text.get('1.0', '1.end'), ' IDLE ') | |||
|
|||
|
|||
class HelpTestTest(unittest.TestCase): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
typo - HelpTextTest
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Done
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
check
Lib/idlelib/help.py
Outdated
def create_fonts(self): | ||
"Create fonts to be used with tags." | ||
self.base_size = idleConf.GetOption('main', 'EditorWindow', | ||
'font-size', type='int') |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This sets base_size to 10 unless user overrides. Test must not depend on any particular value.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks. I've changed the test to set this value.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Set to 12 in the test
Lib/idlelib/help.py
Outdated
self.bind('<<zoom-text-out>>', lambda e: self.zoom(e, 'out')) | ||
|
||
self.event_add('<<zoom-text-in>>', '<Control-Key-minus>') | ||
self.bind('<<zoom-text-in>>', lambda e: self.zoom(e, 'in')) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't think we need the indirection of a pseudoevent or the lambda.
self.bind('', self.zoom)
etc should work as zoom can switch on event.type (keypress, button, or wheel).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I originally did this because I wasn't sure about hardcoding the checks for the 'plus' and 'minus' keys, if those would be configurable at some point. I've changed it to look for the keys and I suppose it can be changed later to use the keys attached to the binding.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think, if anything, we have too many key configurations. zoom has been changed. See new review comment.
Lib/idlelib/help.py
Outdated
self.event_add('<<zoom-text-in>>', '<Control-Key-minus>') | ||
self.bind('<<zoom-text-in>>', lambda e: self.zoom(e, 'in')) | ||
|
||
# Windows and Mac use MouseWheel. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
But they use opposite signs and magnitudes. From turtledemo.main
# For wheel up, event.delta = 120 on Windows, -1 on darwin.
I want to use the debugged code from that file (which I helped write ;-) as a starting point, but it can be simplified to one event handler (zoom) and one new size handler (here scale fontsize).
I also want to consider that the basic handling should be the same for editor windows. Perhaps we can factor out a FontSizer class, with the event handlers, that is initialized with the widget to bind methods to and a method to call with the new font size. This would be easier with standardized attribute names.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I hadn't seen the turtledemo code before I made my PR, but it was helpful to look at it now.
Created the FontSizer class. I tried to do it in a way as to not rely on the names that the widget uses. The only thing that is required is that the font used by the widget be an actual Font()
and not just defined on the fly.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Still need to review.
Lib/idlelib/help.py
Outdated
for tag in self.fonts: | ||
self.fonts[tag]['size'] = int(base * scale[tag]) | ||
|
||
def zoom(self, event, how='out'): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Delete 'how' and switch on event.type.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Done. I like it much better this way. :-)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
check
Lib/idlelib/help.py
Outdated
"Scale tag sizes based on the size of normal text." | ||
# Scale percentages are from Sphinx classic.css. | ||
scale = {'normal': 1.0, 'h6': 1.0, 'h5': 1.1, 'h4': 1.2, 'h3': 1.4, | ||
'h2': 1.6, 'h1': 2.0, 'em': 1.0, 'pre': 1.0, 'preblock': 0.9} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
h6 to h4 are not needed. 'h1': 20 seems too big as current 20/12 = 5/3 = 1.6 seems sufficient. ditto for h2 18/12 = 3/2 = 1.5, h3 = 15/12 = 5/4 = 1.2. The current ratios result in text the nearly matches the Firefox version.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I had originally added the extra values just to show the progression, even though they weren't needed. Now I've removed them and updated the ratios.
Lib/idlelib/help.py
Outdated
if how == 'wheel': | ||
# Button 5 or negative delta is when mouse wheel is scrolled down. | ||
how = 'in' if event.num == 5 or event.delta < 0 else 'out' | ||
factor = 1.1 if how == 'out' else 0.9 |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I would rather just increase/decrease the integer size by 1, but within bounds.
self.scale_fontsize(max(self.base_size - 1, MINIMUM_FONT_SIZE) if <decrease> else
min(self.base_size - 1, MAXIMUM_FONT_SIZE))
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Done
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
check
Lib/idlelib/idle_test/test_help.py
Outdated
('<Control-Button-4>', 0, zoomout), | ||
('<Control-Button-5>', 0, base)) | ||
|
||
# Base size starts at 12. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You set it to the default editor text size, which is 10 on buildbots.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fixed to set this value during the test.
Lib/idlelib/idle_test/test_help.py
Outdated
|
||
# Base size starts at 12. | ||
sizes = [text.fonts[tag]['size'] for tag in tags] | ||
eq(text.base_size, base[0]) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
10 is not 12.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Done
normalfont = self.findfont(['TkDefaultFont', 'arial', 'helvetica']) | ||
fixedfont = self.findfont(['TkFixedFont', 'monaco', 'courier']) | ||
|
||
self.fonts = fonts = {} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't really like the doubling of the configuration code, but I have not thought of anything better yet.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I didn't make any changes for this.
I have made the requested changes; please review again. |
Thanks for making the requested changes! @terryjreedy: please review the changes made to this pull request. |
Lib/idlelib/help.py
Outdated
# Zoom out works with or without shift. | ||
self.widget.event_add( | ||
'<<zoom-text-out>>', | ||
*[f'<{shortcut}-Key-equal>', f'<{shortcut}-Key-plus>']) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
f(*[1,2]) is exactly the same as f(1,2). Using * in a call is for when one has a name, not a display.
so pass f'<{shortcut}-Key-equal>', f'<{shortcut}-Key-plus>'
.
I would still like to remove the pseudoevents.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The unneeded *[..] is still present here and on line 185.
turtledemo.main does not have these pseudoevents. The only reason I can think of now to add them is if they were to make testing significantly easier and more dependable.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I tried back in May and I've been trying for a few days to get the tests to work without the virtual events. Even though it seems like the text widget has focus, doing a text.event_generate('<Control-Key-plus>')
just won't activate the binding. I'm sure I'm just missing something obvious, but I cannot figure out what it is. :-(
Lib/idlelib/help.py
Outdated
self.bind('<<zoom-text-out>>', lambda e: self.zoom(e, 'out')) | ||
|
||
self.event_add('<<zoom-text-in>>', '<Control-Key-minus>') | ||
self.bind('<<zoom-text-in>>', lambda e: self.zoom(e, 'in')) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think, if anything, we have too many key configurations. zoom has been changed. See new review comment.
Lib/idlelib/help.py
Outdated
self.event_add('<<zoom-text-in>>', '<Control-Key-minus>') | ||
self.bind('<<zoom-text-in>>', lambda e: self.zoom(e, 'in')) | ||
|
||
# Windows and Mac use MouseWheel. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Still need to review.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You can leave the one suggested code change until I retest and recheck code. 3.7.0c1 is scheduled next Tues, but I don't want to rush this without test on Mac. I really must try out devguide instructions with a small MacBook portable I was given.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We started with this code, extracted from turtledemo.main.
# Menu bindings (which we might use for context menu)
menu.add_command(label="Decrease (C-'-')", command=self.decrease_size,
font=menufont)
menu.add_command(label="Increase (C-'+')", command=self.increase_size,
font=menufont)
# Key/mouse bindings
text.bind_all('<%s-minus>' % shortcut, self.decrease_size)
text.bind_all('<%s-underscore>' % shortcut, self.decrease_size)
text.bind_all('<%s-equal>' % shortcut, self.increase_size)
text.bind_all('<%s-plus>' % shortcut, self.increase_size)
text.bind('<Control-MouseWheel>', self.update_mousewheel)
text.bind('<Control-Button-4>', self.increase_size)
text.bind('<Control-Button-5>', self.decrease_size)
# Event/action functions
darwin = sys.platform == 'darwin'
def set_txtsize(self, size):
txtfont[1] = size
self.text['font'] = tuple(txtfont)
self.output_lbl['text'] = 'Font size %d' % size
def decrease_size(self, dummy=None):
self.set_txtsize(max(txtfont[1] - 1, MINIMUM_FONT_SIZE))
return 'break'
def increase_size(self, dummy=None):
self.set_txtsize(min(txtfont[1] + 1, MAXIMUM_FONT_SIZE))
return 'break'
def update_mousewheel(self, event):
# For wheel up, event.delta = 120 on Windows, -1 on darwin.
# X-11 sends Control-Button-4 event instead.
if (event.delta < 0) == (not darwin):
return self.decrease_size()
else:
return self.increase_size()
As far as I know, menu, keyboard, and mouse methods work on all three major systems. There are no pseudoevents. To the extent possible, the meaning of actions is determined by their bindings.
Note: the code above is extracted from 4 different locations. This is the first time I have looked at it altogether.
At my request, Cheryl combined and consolidated the functions into one zoom handler. At Tal's request, she partly undid what I had requested. Zoom() has been separated into multiple functions again. However, I prefer the original code to the result. If we don't agree on reverting to the original design, lets discuss.
Lib/idlelib/help.py
Outdated
# Zoom out works with or without shift. | ||
self.widget.event_add( | ||
'<<zoom-text-out>>', | ||
*[f'<{shortcut}-Key-equal>', f'<{shortcut}-Key-plus>']) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The unneeded *[..] is still present here and on line 185.
turtledemo.main does not have these pseudoevents. The only reason I can think of now to add them is if they were to make testing significantly easier and more dependable.
Lib/idlelib/help.py
Outdated
|
||
def zoom_mousewheel(self, event): | ||
"Adjust font size based on mouse wheel direction." | ||
wheel_up = {EventType.MouseWheel: event.delta > 0, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This leaves out '== darwin', which I believe is correct for font resizing even if not correct for scrolling.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I wasn't sure when that logic needed to be applied and when it didn't. So, it's only adjusted on scrolling and not on other uses of the mousewheel?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I tested turtledemo a month ago and font resizing works the same as on Windows. Moving the top of the wheeI up increases size. I don't actually know if this is the correct convention on Mac, but in the absence of information otherwise, including the absence of complaint about turtledemo, I will consider it correct. I may have tested command-wheel-up in Safari and will recheck tomorrow.
When we copied to logic for scrolling, the result was buggy since the convention on Mac, which I know I verified with Safari, is the opposite of Windows (and Linux, I believe). I will try to find and point you to the tracker discussion.
Lib/idlelib/help.py
Outdated
@@ -176,31 +176,39 @@ def bind_events(self): | |||
shortcut = 'Command' if darwin else 'Control' | |||
# Zoom out works with or without shift. | |||
self.widget.event_add( | |||
'<<zoom-text-out>>', | |||
'<<zoom-text-in>>', |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I guess the pseudoevent error was masked since the following line has the same error and the direction was recalculated (in former version of zoom).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
While refamiliarizing myself with this code, I kept getting confused on what 'zoom out' and 'zoom in' meant and added a comment to the new methods in help.py to say which means larger text and which means smaller text. Once I did that, I realized I had it backwards in the definitions.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I agree that these names are potentially confusing and propose above (May 17 comment on line 180) 'make_font_bigger' and 'make_font_smaller'.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
github threw away or hid my my review summary. Will try to find or recreate.
Now it is visible again.
When I looked back through my original commits and Terry's comments, I thought the main issue was that I was using lambda and sending another parameter to the |
My apologies, @terryjreedy and @csabella. I wasn't aware of Terry's request to consolidate the zoom handling into a single handler. Cheryl, please do as Terry has requested, as he has the final say on these matters. |
Tal, I assumed that you had not gone through all the hidden 'outdated' comments. In the future, perhaps I should give better summary reviews as they should be easier to find. In any case, I welcome your review of all IDLE issues. Your dislike of zoom got me to take another look. I partly like it, but partly not. It is okay with me if we occasionally try a refactoring and decide against it. I responded about pseudoevents and handler names on the May 17 comment on line 180. |
It conflicted with my requested changes.
@terryjreedy, on the contrary, I should just take more care to read all of the previous discussion. You've got more than enough on your plate, you shouldn't bother to write summaries when the reviews are there to be read. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Current requested changes: Reverse previous request to combine turtledemo functions into zoom(). However, add psuedoevents '<<increase_font_size>>' and '<<decrease_font_size>>', use the same names for the handler functions. Don't use *[] in event add. Use original mouse logic.
Use pseudoevents for testing. Don't worry about key-mouse events now.
* Remove use of callback to adjust fonts in tags. * Adjust tag font sizes the same as the main font. * TODO: Tags don't retain ratio, so after they all go to min size, they don't recover original ratios.
* Originally resized Font objects, but now can resize fonts defined as tuples.
Ping, @csabella? |
I summarized 'remaining issues' on Sept 8, 2018, and added more on Sept 23 and Dec 14. I need to check all against current code updated since then. |
As far as I know, I had made all the requested changes and I had tried to keep up with rebasing. |
This missed the boat for inclusion in Python 3.9 which accepts security fixes only as of today. |
This PR is stale because it has been open for 30 days with no activity. |
This PR is stale because it has been open for 30 days with no activity. |