202 lines
5.0 KiB
Ruby
202 lines
5.0 KiB
Ruby
module Msf
|
|
module Ui
|
|
module Gtk2
|
|
|
|
##
|
|
# This class perform some tooltips for TreeView widget
|
|
# This class is a port of Daniel J. Popowich coded in Python to ruby-gtk2 style
|
|
##
|
|
class TreeViewTooltips < Gtk::Window
|
|
|
|
def initialize
|
|
super(Gtk::Window::POPUP)
|
|
self.set_resizable(false)
|
|
self.set_border_width(5)
|
|
self.set_app_paintable(true)
|
|
|
|
# create the label
|
|
@label = Gtk::Label.new
|
|
@label.set_wrap(true)
|
|
@label.set_alignment(0.5, 0.5)
|
|
@label.set_use_markup(true)
|
|
@label.show()
|
|
self.add(@label)
|
|
|
|
# by default, the tooltip is enabled
|
|
@enabled = true
|
|
# saves the current cell
|
|
@save = nil
|
|
# the timer id for the next tooltip to be shown
|
|
@next = nil
|
|
# flag on whether the tooltip window is shown
|
|
@shown = false
|
|
end
|
|
|
|
#
|
|
# Enable the tooltip
|
|
#
|
|
def enable
|
|
@enable = true
|
|
end
|
|
|
|
#
|
|
# Disable the tooltip
|
|
#
|
|
def disable
|
|
@enable = false
|
|
end
|
|
|
|
|
|
#
|
|
# Handler to be connected on to the Gtk::Treeview
|
|
#
|
|
def add_view(view)
|
|
|
|
# Enter
|
|
view.signal_connect('motion-notify-event') do |view, event|
|
|
motion_handler(view, event)
|
|
end
|
|
|
|
# Leave
|
|
view.signal_connect('leave-notify-event') do |view, event|
|
|
leave_handler(view, event)
|
|
end
|
|
end
|
|
|
|
|
|
#
|
|
# Given the x,y coordinates of the pointer and the width and
|
|
# height (w,h) demensions of the tooltip window, return the x, y
|
|
# coordinates of the tooltip window.
|
|
#
|
|
# The default location is to center the window on the pointer
|
|
# and 20 pixels below it.
|
|
#
|
|
def location(x, y, w, h)
|
|
return x - w/2, y + 20
|
|
end
|
|
|
|
|
|
#########
|
|
private #
|
|
#########
|
|
|
|
#
|
|
# show the tooltip popup with the text/markup given by
|
|
# tooltip.
|
|
#
|
|
# tooltip: the text/markup for the tooltip.
|
|
# x, y: the coord. (root window based) of the pointer.
|
|
#
|
|
def show(tooltip, x, y)
|
|
# set label
|
|
@label.set_label(tooltip)
|
|
|
|
# resize window
|
|
w, h = self.size_request()
|
|
|
|
# Display the window with a 1-pixel black border
|
|
self.style.paint_flat_box( self.window, # window: a Gdk::Window
|
|
Gtk::STATE_NORMAL, # state_type: a state type (GtkStateType)
|
|
Gtk::SHADOW_OUT, # shadow_type: a shadow type (GtkShadowType)
|
|
nil, # area: a Gdk::Rectangle to which the output is clipped
|
|
self, # widget: a Gtk::Widget
|
|
'tooltip', # detail: a String or nil
|
|
0, # x:
|
|
0, # y:
|
|
w, # width:
|
|
h # height:
|
|
)
|
|
|
|
# move the window
|
|
self.move(*location(x,y,w,h))
|
|
|
|
# show it
|
|
self.show_all
|
|
@shown = true
|
|
end
|
|
|
|
#
|
|
# hide the tooltip
|
|
#
|
|
def hide_me
|
|
queue_next()
|
|
self.hide()
|
|
@shown = false
|
|
end
|
|
|
|
#
|
|
# When the pointer leaves the view, hide the tooltip
|
|
#
|
|
def leave_handler(view, event)
|
|
hide_me()
|
|
end
|
|
|
|
#
|
|
# As the pointer moves across the view, show a tooltip.
|
|
#
|
|
def motion_handler(view, event)
|
|
|
|
path = view.get_path_at_pos(event.x, event.y)
|
|
|
|
if (@enabled and path)
|
|
path, col, x, y = path
|
|
tooltip = get_tooltip(view, col, path)
|
|
if tooltip
|
|
tooltip = tooltip.strip
|
|
queue_next( [path, col], tooltip, event.x_root, event.y_root)
|
|
return
|
|
end
|
|
end
|
|
|
|
hide_me()
|
|
end
|
|
|
|
#
|
|
# queue next request to show a tooltip
|
|
#
|
|
def queue_next(*args)
|
|
|
|
# if args is non-empty it means a request was made to show a
|
|
# tooltip. if empty, no request is being made, but any
|
|
# pending requests should be cancelled anyway.
|
|
|
|
cell = nil
|
|
|
|
# if called with args, break them out
|
|
if args
|
|
cell, tooltip, x, y = args
|
|
end
|
|
|
|
# if it's the same cell as previously shown, just return
|
|
if (@save == cell)
|
|
return
|
|
end
|
|
|
|
# if we have something queued up, cancel it
|
|
if @next:
|
|
Gtk.timeout_remove(@next)
|
|
@next = nil
|
|
end
|
|
|
|
# if there was a request...
|
|
if cell
|
|
# if the tooltip is already shown, show the new one
|
|
# immediately
|
|
if @shown
|
|
show(tooltip, x, y)
|
|
# else queue it up in 1/2 second
|
|
else
|
|
@next = Gtk.timeout_add(500) { show(tooltip, x, y) }
|
|
end
|
|
end
|
|
|
|
# save this cell
|
|
@save = cell
|
|
end
|
|
end
|
|
|
|
end
|
|
end
|
|
end
|