123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186 |
- """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:
- """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: # pragma: no cover
- 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().__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) # pragma: no cover
- self.anchor_widget.unbind("<Button>", self._id3) # pragma: no cover
- except TclError:
- pass
- super().__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: # pragma: no cover
- pass
- super().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().__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)
|