Commit 87e59ac1 authored by Tal Einat's avatar Tal Einat Committed by GitHub

bpo-33839: refactor IDLE's tooltips & calltips, add docstrings and tests (GH-7683)

* make CallTip and ToolTip sub-classes of a common abstract base class
* remove ListboxToolTip (unused and ugly)
* greatly increase test coverage
* tested on Windows, Linux and macOS
parent 2e5566d9
...@@ -51,7 +51,7 @@ class Calltip: ...@@ -51,7 +51,7 @@ class Calltip:
self.open_calltip(False) self.open_calltip(False)
def refresh_calltip_event(self, event): def refresh_calltip_event(self, event):
if self.active_calltip and self.active_calltip.is_active(): if self.active_calltip and self.active_calltip.tipwindow:
self.open_calltip(False) self.open_calltip(False)
def open_calltip(self, evalfuncs): def open_calltip(self, evalfuncs):
......
This diff is collapsed.
...@@ -80,11 +80,14 @@ AboutDialog_spec = { ...@@ -80,11 +80,14 @@ AboutDialog_spec = {
"are correctly displayed.\n [Close] to exit.", "are correctly displayed.\n [Close] to exit.",
} }
# TODO implement ^\; adding '<Control-Key-\\>' to function does not work.
_calltip_window_spec = { _calltip_window_spec = {
'file': 'calltip_w', 'file': 'calltip_w',
'kwds': {}, 'kwds': {},
'msg': "Typing '(' should display a calltip.\n" 'msg': "Typing '(' should display a calltip.\n"
"Typing ') should hide the calltip.\n" "Typing ') should hide the calltip.\n"
"So should moving cursor out of argument area.\n"
"Force-open-calltip does not work here.\n"
} }
_module_browser_spec = { _module_browser_spec = {
......
...@@ -23,7 +23,7 @@ class CallTipWindowTest(unittest.TestCase): ...@@ -23,7 +23,7 @@ class CallTipWindowTest(unittest.TestCase):
del cls.text, cls.root del cls.text, cls.root
def test_init(self): def test_init(self):
self.assertEqual(self.calltip.widget, self.text) self.assertEqual(self.calltip.anchor_widget, self.text)
if __name__ == '__main__': if __name__ == '__main__':
unittest.main(verbosity=2) unittest.main(verbosity=2)
from idlelib.tooltip import TooltipBase, Hovertip
from test.support import requires
requires('gui')
from functools import wraps
import time
from tkinter import Button, Tk, Toplevel
import unittest
def setUpModule():
global root
root = Tk()
def root_update():
global root
root.update()
def tearDownModule():
global root
root.update_idletasks()
root.destroy()
del root
def add_call_counting(func):
@wraps(func)
def wrapped_func(*args, **kwargs):
wrapped_func.call_args_list.append((args, kwargs))
return func(*args, **kwargs)
wrapped_func.call_args_list = []
return wrapped_func
def _make_top_and_button(testobj):
global root
top = Toplevel(root)
testobj.addCleanup(top.destroy)
top.title("Test tooltip")
button = Button(top, text='ToolTip test button')
button.pack()
testobj.addCleanup(button.destroy)
top.lift()
return top, button
class ToolTipBaseTest(unittest.TestCase):
def setUp(self):
self.top, self.button = _make_top_and_button(self)
def test_base_class_is_unusable(self):
global root
top = Toplevel(root)
self.addCleanup(top.destroy)
button = Button(top, text='ToolTip test button')
button.pack()
self.addCleanup(button.destroy)
with self.assertRaises(NotImplementedError):
tooltip = TooltipBase(button)
tooltip.showtip()
class HovertipTest(unittest.TestCase):
def setUp(self):
self.top, self.button = _make_top_and_button(self)
def test_showtip(self):
tooltip = Hovertip(self.button, 'ToolTip text')
self.addCleanup(tooltip.hidetip)
self.assertFalse(tooltip.tipwindow and tooltip.tipwindow.winfo_viewable())
tooltip.showtip()
self.assertTrue(tooltip.tipwindow and tooltip.tipwindow.winfo_viewable())
def test_showtip_twice(self):
tooltip = Hovertip(self.button, 'ToolTip text')
self.addCleanup(tooltip.hidetip)
self.assertFalse(tooltip.tipwindow and tooltip.tipwindow.winfo_viewable())
tooltip.showtip()
self.assertTrue(tooltip.tipwindow and tooltip.tipwindow.winfo_viewable())
orig_tipwindow = tooltip.tipwindow
tooltip.showtip()
self.assertTrue(tooltip.tipwindow and tooltip.tipwindow.winfo_viewable())
self.assertIs(tooltip.tipwindow, orig_tipwindow)
def test_hidetip(self):
tooltip = Hovertip(self.button, 'ToolTip text')
self.addCleanup(tooltip.hidetip)
tooltip.showtip()
tooltip.hidetip()
self.assertFalse(tooltip.tipwindow and tooltip.tipwindow.winfo_viewable())
def test_showtip_on_mouse_enter_no_delay(self):
tooltip = Hovertip(self.button, 'ToolTip text', hover_delay=None)
self.addCleanup(tooltip.hidetip)
tooltip.showtip = add_call_counting(tooltip.showtip)
root_update()
self.assertFalse(tooltip.tipwindow and tooltip.tipwindow.winfo_viewable())
self.button.event_generate('<Enter>', x=0, y=0)
root_update()
self.assertTrue(tooltip.tipwindow and tooltip.tipwindow.winfo_viewable())
self.assertGreater(len(tooltip.showtip.call_args_list), 0)
def test_showtip_on_mouse_enter_hover_delay(self):
tooltip = Hovertip(self.button, 'ToolTip text', hover_delay=50)
self.addCleanup(tooltip.hidetip)
tooltip.showtip = add_call_counting(tooltip.showtip)
root_update()
self.assertFalse(tooltip.tipwindow and tooltip.tipwindow.winfo_viewable())
self.button.event_generate('<Enter>', x=0, y=0)
root_update()
self.assertFalse(tooltip.tipwindow and tooltip.tipwindow.winfo_viewable())
time.sleep(0.1)
root_update()
self.assertTrue(tooltip.tipwindow and tooltip.tipwindow.winfo_viewable())
self.assertGreater(len(tooltip.showtip.call_args_list), 0)
def test_hidetip_on_mouse_leave(self):
tooltip = Hovertip(self.button, 'ToolTip text', hover_delay=None)
self.addCleanup(tooltip.hidetip)
tooltip.showtip = add_call_counting(tooltip.showtip)
root_update()
self.button.event_generate('<Enter>', x=0, y=0)
root_update()
self.button.event_generate('<Leave>', x=0, y=0)
root_update()
self.assertFalse(tooltip.tipwindow and tooltip.tipwindow.winfo_viewable())
self.assertGreater(len(tooltip.showtip.call_args_list), 0)
def test_dont_show_on_mouse_leave_before_delay(self):
tooltip = Hovertip(self.button, 'ToolTip text', hover_delay=50)
self.addCleanup(tooltip.hidetip)
tooltip.showtip = add_call_counting(tooltip.showtip)
root_update()
self.button.event_generate('<Enter>', x=0, y=0)
root_update()
self.button.event_generate('<Leave>', x=0, y=0)
root_update()
time.sleep(0.1)
root_update()
self.assertFalse(tooltip.tipwindow and tooltip.tipwindow.winfo_viewable())
self.assertEqual(tooltip.showtip.call_args_list, [])
if __name__ == '__main__':
unittest.main(verbosity=2)
# general purpose 'tooltip' routines - currently unused in idlelib """Tools for displaying tool-tips.
# (although the 'calltips' extension is partly based on this code)
# may be useful for some purposes in (or almost in ;) the current project scope
# Ideas gleaned from PySol
This includes:
* an abstract base-class for different kinds of tooltips
* a simple text-only Tooltip class
"""
from tkinter import * from tkinter import *
class ToolTipBase:
def __init__(self, button): class TooltipBase(object):
self.button = button """abstract base class for tooltips"""
self.tipwindow = None
self.id = None
self.x = self.y = 0
self._id1 = self.button.bind("<Enter>", self.enter)
self._id2 = self.button.bind("<Leave>", self.leave)
self._id3 = self.button.bind("<ButtonPress>", self.leave)
def enter(self, event=None): def __init__(self, anchor_widget):
self.schedule() """Create a tooltip.
def leave(self, event=None): anchor_widget: the widget next to which the tooltip will be shown
self.unschedule()
self.hidetip()
def schedule(self): Note that a widget will only be shown when showtip() is called.
self.unschedule() """
self.id = self.button.after(1500, self.showtip) self.anchor_widget = anchor_widget
self.tipwindow = None
def unschedule(self): def __del__(self):
id = self.id self.hidetip()
self.id = None
if id:
self.button.after_cancel(id)
def showtip(self): def showtip(self):
"""display the tooltip"""
if self.tipwindow: if self.tipwindow:
return return
# The tip window must be completely outside the button; self.tipwindow = tw = Toplevel(self.anchor_widget)
# show no border on the top level window
tw.wm_overrideredirect(1)
try:
# This command is only needed and available on Tk >= 8.4.0 for OSX.
# Without it, call tips intrude on the typing process by grabbing
# the focus.
tw.tk.call("::tk::unsupported::MacWindowStyle", "style", tw._w,
"help", "noActivates")
except TclError:
pass
self.position_window()
self.showcontents()
self.tipwindow.update_idletasks() # Needed on MacOS -- see #34275.
self.tipwindow.lift() # work around bug in Tk 8.5.18+ (issue #24570)
def position_window(self):
"""(re)-set the tooltip's screen position"""
x, y = self.get_position()
root_x = self.anchor_widget.winfo_rootx() + x
root_y = self.anchor_widget.winfo_rooty() + y
self.tipwindow.wm_geometry("+%d+%d" % (root_x, root_y))
def get_position(self):
"""choose a screen position for the tooltip"""
# The tip window must be completely outside the anchor widget;
# otherwise when the mouse enters the tip window we get # otherwise when the mouse enters the tip window we get
# a leave event and it disappears, and then we get an enter # a leave event and it disappears, and then we get an enter
# event and it reappears, and so on forever :-( # event and it reappears, and so on forever :-(
x = self.button.winfo_rootx() + 20 #
y = self.button.winfo_rooty() + self.button.winfo_height() + 1 # Note: This is a simplistic implementation; sub-classes will likely
self.tipwindow = tw = Toplevel(self.button) # want to override this.
tw.wm_overrideredirect(1) return 20, self.anchor_widget.winfo_height() + 1
tw.wm_geometry("+%d+%d" % (x, y))
self.showcontents()
def showcontents(self, text="Your text here"): def showcontents(self):
# Override this in derived class """content display hook for sub-classes"""
label = Label(self.tipwindow, text=text, justify=LEFT, # See ToolTip for an example
background="#ffffe0", relief=SOLID, borderwidth=1) raise NotImplementedError
label.pack()
def hidetip(self): def hidetip(self):
"""hide the tooltip"""
# Note: This is called by __del__, so careful when overriding/extending
tw = self.tipwindow tw = self.tipwindow
self.tipwindow = None self.tipwindow = None
if tw: if tw:
tw.destroy() try:
tw.destroy()
except TclError:
pass
class OnHoverTooltipBase(TooltipBase):
"""abstract base class for tooltips, with delayed on-hover display"""
def __init__(self, anchor_widget, hover_delay=1000):
"""Create a tooltip with a mouse hover delay.
anchor_widget: the widget next to which the tooltip will be shown
hover_delay: time to delay before showing the tooltip, in milliseconds
class ToolTip(ToolTipBase): Note that a widget will only be shown when showtip() is called,
def __init__(self, button, text): e.g. after hovering over the anchor widget with the mouse for enough
ToolTipBase.__init__(self, button) time.
"""
super(OnHoverTooltipBase, self).__init__(anchor_widget)
self.hover_delay = hover_delay
self._after_id = None
self._id1 = self.anchor_widget.bind("<Enter>", self._show_event)
self._id2 = self.anchor_widget.bind("<Leave>", self._hide_event)
self._id3 = self.anchor_widget.bind("<Button>", self._hide_event)
def __del__(self):
try:
self.anchor_widget.unbind("<Enter>", self._id1)
self.anchor_widget.unbind("<Leave>", self._id2)
self.anchor_widget.unbind("<Button>", self._id3)
except TclError:
pass
super(OnHoverTooltipBase, self).__del__()
def _show_event(self, event=None):
"""event handler to display the tooltip"""
if self.hover_delay:
self.schedule()
else:
self.showtip()
def _hide_event(self, event=None):
"""event handler to hide the tooltip"""
self.hidetip()
def schedule(self):
"""schedule the future display of the tooltip"""
self.unschedule()
self._after_id = self.anchor_widget.after(self.hover_delay,
self.showtip)
def unschedule(self):
"""cancel the future display of the tooltip"""
after_id = self._after_id
self._after_id = None
if after_id:
self.anchor_widget.after_cancel(after_id)
def hidetip(self):
"""hide the tooltip"""
try:
self.unschedule()
except TclError:
pass
super(OnHoverTooltipBase, self).hidetip()
class Hovertip(OnHoverTooltipBase):
"A tooltip that pops up when a mouse hovers over an anchor widget."
def __init__(self, anchor_widget, text, hover_delay=1000):
"""Create a text tooltip with a mouse hover delay.
anchor_widget: the widget next to which the tooltip will be shown
hover_delay: time to delay before showing the tooltip, in milliseconds
Note that a widget will only be shown when showtip() is called,
e.g. after hovering over the anchor widget with the mouse for enough
time.
"""
super(Hovertip, self).__init__(anchor_widget, hover_delay=hover_delay)
self.text = text self.text = text
def showcontents(self):
ToolTipBase.showcontents(self, self.text)
class ListboxToolTip(ToolTipBase):
def __init__(self, button, items):
ToolTipBase.__init__(self, button)
self.items = items
def showcontents(self): def showcontents(self):
listbox = Listbox(self.tipwindow, background="#ffffe0") label = Label(self.tipwindow, text=self.text, justify=LEFT,
listbox.pack() background="#ffffe0", relief=SOLID, borderwidth=1)
for item in self.items: label.pack()
listbox.insert(END, item)
def _tooltip(parent): # htest # def _tooltip(parent): # htest #
top = Toplevel(parent) top = Toplevel(parent)
...@@ -83,14 +170,17 @@ def _tooltip(parent): # htest # ...@@ -83,14 +170,17 @@ def _tooltip(parent): # htest #
top.geometry("+%d+%d" % (x, y + 150)) top.geometry("+%d+%d" % (x, y + 150))
label = Label(top, text="Place your mouse over buttons") label = Label(top, text="Place your mouse over buttons")
label.pack() label.pack()
button1 = Button(top, text="Button 1") button1 = Button(top, text="Button 1 -- 1/2 second hover delay")
button2 = Button(top, text="Button 2")
button1.pack() button1.pack()
Hovertip(button1, "This is tooltip text for button1.", hover_delay=500)
button2 = Button(top, text="Button 2 -- no hover delay")
button2.pack() button2.pack()
ToolTip(button1, "This is tooltip text for button1.") Hovertip(button2, "This is tooltip\ntext for button2.", hover_delay=None)
ListboxToolTip(button2, ["This is","multiple line",
"tooltip text","for button2"])
if __name__ == '__main__': if __name__ == '__main__':
from unittest import main
main('idlelib.idle_test.test_tooltip', verbosity=2, exit=False)
from idlelib.idle_test.htest import run from idlelib.idle_test.htest import run
run(_tooltip) run(_tooltip)
IDLE: refactor ToolTip and CallTip and add documentation and tests
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