metasploit-framework/lib/msf/ui/gtk2/treeviewtooltips.rb

201 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