"""Tools for displaying tool-tips. This includes: * an abstract base-class for different kinds of tooltips * a simple text-only Tooltip class """ from tkinter import * class TooltipBase(object): """abstract base class for tooltips""" def __init__(self, anchor_widget): """Create a tooltip. anchor_widget: the widget next to which the tooltip will be shown Note that a widget will only be shown when showtip() is called. """ self.anchor_widget = anchor_widget self.tipwindow = None def __del__(self): self.hidetip() def showtip(self): """display the tooltip""" if self.tipwindow: return 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 # a leave event and it disappears, and then we get an enter # event and it reappears, and so on forever :-( # # Note: This is a simplistic implementation; sub-classes will likely # want to override this. return 20, self.anchor_widget.winfo_height() + 1 def showcontents(self): """content display hook for sub-classes""" # See ToolTip for an example raise NotImplementedError def hidetip(self): """hide the tooltip""" # Note: This is called by __del__, so careful when overriding/extending tw = self.tipwindow self.tipwindow = None if tw: 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 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(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 def showcontents(self): label = Label(self.tipwindow, text=self.text, justify=LEFT, background="#ffffe0", relief=SOLID, borderwidth=1) label.pack() def _tooltip(parent): # htest # top = Toplevel(parent) top.title("Test tooltip") x, y = map(int, parent.geometry().split('+')[1:]) top.geometry("+%d+%d" % (x, y + 150)) label = Label(top, text="Place your mouse over buttons") label.pack() button1 = Button(top, text="Button 1 -- 1/2 second hover delay") button1.pack() Hovertip(button1, "This is tooltip text for button1.", hover_delay=500) button2 = Button(top, text="Button 2 -- no hover delay") button2.pack() Hovertip(button2, "This is tooltip\ntext for button2.", hover_delay=None) if __name__ == '__main__': from unittest import main main('idlelib.idle_test.test_tooltip', verbosity=2, exit=False) from idlelib.idle_test.htest import run run(_tooltip)
Name | Type | Size | Permission | Actions |
---|---|---|---|---|
Icons | Folder | 0755 |
|
|
__pycache__ | Folder | 0755 |
|
|
idle_test | Folder | 0755 |
|
|
CREDITS.txt | File | 1.82 KB | 0644 |
|
ChangeLog | File | 55.04 KB | 0644 |
|
HISTORY.txt | File | 10.07 KB | 0644 |
|
NEWS.txt | File | 38.91 KB | 0644 |
|
NEWS2x.txt | File | 26.54 KB | 0644 |
|
README.txt | File | 9.37 KB | 0644 |
|
TODO.txt | File | 8.28 KB | 0644 |
|
__init__.py | File | 396 B | 0644 |
|
__main__.py | File | 159 B | 0644 |
|
_pyclbr.py | File | 14.84 KB | 0644 |
|
autocomplete.py | File | 9.11 KB | 0644 |
|
autocomplete_w.py | File | 19.36 KB | 0644 |
|
autoexpand.py | File | 3.14 KB | 0644 |
|
browser.py | File | 8.09 KB | 0644 |
|
calltip.py | File | 5.92 KB | 0644 |
|
calltip_w.py | File | 6.94 KB | 0644 |
|
codecontext.py | File | 10.24 KB | 0644 |
|
colorizer.py | File | 11.01 KB | 0644 |
|
config-extensions.def | File | 2.21 KB | 0644 |
|
config-highlight.def | File | 2.62 KB | 0644 |
|
config-keys.def | File | 10.52 KB | 0644 |
|
config-main.def | File | 3.05 KB | 0644 |
|
config.py | File | 37.97 KB | 0644 |
|
config_key.py | File | 13.09 KB | 0644 |
|
configdialog.py | File | 98.69 KB | 0644 |
|
debugger.py | File | 18.65 KB | 0644 |
|
debugger_r.py | File | 11.86 KB | 0644 |
|
debugobj.py | File | 3.96 KB | 0644 |
|
debugobj_r.py | File | 1.06 KB | 0644 |
|
delegator.py | File | 1.02 KB | 0644 |
|
dynoption.py | File | 1.97 KB | 0644 |
|
editor.py | File | 65.7 KB | 0644 |
|
extend.txt | File | 3.56 KB | 0644 |
|
filelist.py | File | 3.8 KB | 0644 |
|
grep.py | File | 6.58 KB | 0644 |
|
help.html | File | 53.82 KB | 0644 |
|
help.py | File | 11.06 KB | 0644 |
|
help_about.py | File | 8.77 KB | 0644 |
|
history.py | File | 3.95 KB | 0644 |
|
hyperparser.py | File | 12.58 KB | 0644 |
|
idle.py | File | 454 B | 0644 |
|
idle.pyw | File | 570 B | 0644 |
|
iomenu.py | File | 20.25 KB | 0644 |
|
macosx.py | File | 9.43 KB | 0644 |
|
mainmenu.py | File | 3.62 KB | 0644 |
|
multicall.py | File | 18.21 KB | 0644 |
|
outwin.py | File | 5.67 KB | 0644 |
|
paragraph.py | File | 7 KB | 0644 |
|
parenmatch.py | File | 7.04 KB | 0644 |
|
pathbrowser.py | File | 3.12 KB | 0644 |
|
percolator.py | File | 3.06 KB | 0644 |
|
pyparse.py | File | 19.65 KB | 0644 |
|
pyshell.py | File | 56.38 KB | 0755 |
|
query.py | File | 12.14 KB | 0644 |
|
redirector.py | File | 6.71 KB | 0644 |
|
replace.py | File | 7.33 KB | 0644 |
|
rpc.py | File | 20.64 KB | 0644 |
|
rstrip.py | File | 868 B | 0644 |
|
run.py | File | 16.87 KB | 0644 |
|
runscript.py | File | 7.66 KB | 0644 |
|
scrolledlist.py | File | 4.35 KB | 0644 |
|
search.py | File | 3.09 KB | 0644 |
|
searchbase.py | File | 7.28 KB | 0644 |
|
searchengine.py | File | 7.3 KB | 0644 |
|
squeezer.py | File | 13 KB | 0644 |
|
stackviewer.py | File | 4.35 KB | 0644 |
|
statusbar.py | File | 1.41 KB | 0644 |
|
textview.py | File | 5.98 KB | 0644 |
|
tooltip.py | File | 6.33 KB | 0644 |
|
tree.py | File | 14.74 KB | 0644 |
|
undo.py | File | 10.79 KB | 0644 |
|
window.py | File | 2.53 KB | 0644 |
|
zoomheight.py | File | 1.31 KB | 0644 |
|
zzdummy.py | File | 961 B | 0644 |
|