Gtk2::Ex::WidgetBits - miscellaneous Gtk widget helpers


Gtk2-Ex-WidgetBits documentation Contained in the Gtk2-Ex-WidgetBits distribution.

Index


Code Index:

NAME

Top

Gtk2::Ex::WidgetBits - miscellaneous Gtk widget helpers

SYNOPSIS

Top

 use Gtk2::Ex::WidgetBits;

FUNCTIONS

Top

Widget Position

($x,$y) = Gtk2::Ex::WidgetBits::get_root_position ($widget)

Return the position of the top left corner of $widget in root window coordinates. If $widget is unrealized the return is an empty list ().

This uses Gtk2::Ex::GdkBits::window_get_root_position so takes the most recently recorded window positions rather than making an X server round-trip.

Gtk2::Ex::WidgetBits::warp_pointer ($widget, $x, $y)

Warp, ie. forcibly move, the mouse pointer to $x,$y in $widget coordinates (ie. relative to the widget's top-left corner). $widget must be realized, since otherwise it doesn't have a screen position.

See Gtk2::Gdk::Display for the basic warp_pointer in root window coordinates. The code here converts using get_root_position above, so there's no server round-trip. Warping is available in Gtk 2.2 up.

The underlying XWarpPointer operates relative to any window, not just the root, but Gdk doesn't make that feature available.

($x,$y) = Gtk2::Ex::WidgetBits::xy_root_to_widget ($widget, $root_x,$root_y)

Convert a root window X,Y position to widget coordinates. If $widget is not realized then it doesn't have a screen position and the return is an empty list.

    if (my ($x,$y) = Gtk2::Ex::WidgetBits::xy_root_to_widget
                        ($widget, $x_root, $y_root)) {
       # widget $x,$y
    } else {
       # $widget is not realized, no values
    }

Widget Distances

In the following functions, sizes in millimetres come from the screen (Gtk2::Gdk::Screen). A widget has a screen when it's been added as a child somewhere under a toplevel Gtk2::Window etc. Or in Gtk 2.0.x there's only ever one screen and its size is always used (Gtk2->init required).

($width_mm, $height_mm) = Gtk2::Ex::WidgetBits::pixel_size_mm ($widget)

Return the width and height in millimetres of a pixel in $widget. If $widget doesn't have a screen then return no values.

   my ($xmm, $ymm) = Gtk2::Ex::WidgetBits::pixel_size_mm ($widget)
     or print "no screen yet";

$ratio = Gtk2::Ex::WidgetBits::pixel_aspect_ratio ($widget)

Return the ratio width/height of pixel size in millimetres in $widget. For example if a pixel is 3mm wide by 2mm high then the ratio would be 1.5. If $widget doesn't have a screen then return undef.

This ratio is the same way around as Gtk2::AspectFrame. Setting the ratio property to this pixel ratio makes the child square on the monitor.

$mm = Gtk2::Ex::WidgetBits::xy_distance_mm ($widget, $x1,$y1, $x2,$y2)

Return the distance in millimetres between pixel points $x1,$y1 and $x2,$y2 in $widget. If $widget doesn't have a screen then return undef.

SEE ALSO

Top

Gtk2::Ex::AdjustmentBits, Gtk2::Ex::EntryBits, Gtk2::Ex::GdkBits, Gtk2::Ex::MenuBits, Gtk2::Ex::TextBufferBits, Gtk2::Ex::TreeModelBits, Gtk2::Ex::TreeModel::ImplBits, Gtk2::Ex::TreeViewBits, Gtk2::Ex::Units

Gtk2::Ex::ActionTooltips, Gtk2::Ex::KeySnooper, Gtk2::Ex::SyncCall, Gtk2::Ex::Statusbar::MessageUntilKey, Gtk2::Ex::TreeModelFilter::Change, Test::Weaken::Gtk2

Gtk2::Widget, Gtk2::Ex::WidgetCursor.

HOME PAGE

Top

http://user42.tuxfamily.org/gtk2-ex-widgetbits/index.html

LICENSE

Top

Copyright 2008, 2009, 2010, 2011 Kevin Ryde

Gtk2-Ex-WidgetBits is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3, or (at your option) any later version.

Gtk2-Ex-WidgetBits is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.

You should have received a copy of the GNU General Public License along with Gtk2-Ex-WidgetBits. If not, see http://www.gnu.org/licenses/.


Gtk2-Ex-WidgetBits documentation Contained in the Gtk2-Ex-WidgetBits distribution.

# Copyright 2007, 2008, 2009, 2010, 2011 Kevin Ryde

# This file is part of Gtk2-Ex-WidgetBits.
#
# Gtk2-Ex-WidgetBits is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License as published
# by the Free Software Foundation; either version 3, or (at your option) any
# later version.
#
# Gtk2-Ex-WidgetBits is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General
# Public License for more details.
#
# You should have received a copy of the GNU General Public License along
# with Gtk2-Ex-WidgetBits.  If not, see <http://www.gnu.org/licenses/>.

package Gtk2::Ex::WidgetBits;
use 5.008;
use strict;
use warnings;
use Carp;
use Gtk2;

# uncomment this to run the ### lines
#use Smart::Comments;

our $VERSION = 43;

# get_root_position() might be done as
#
#     my $toplevel = $widget->get_toplevel;
#     my $window = $toplevel->window || return; # if unrealized
#     return $widget->translate_coordinates ($toplevel,
#                                            $window->get_position);
#
# if it can be assumed the toplevel widget of a hierarchy has its window a
# child of the root window (after possible window manager frame).  Or
# alternately just get_toplevel() could eliminate the 'no-window' test if it
# can be assumed the toplevel is a windowed widget.  All of which is of
# course true of GtkWindow, and probably ends up right for GtkPlug too, but
# could a custom toplevel widget do something tricky?
#
sub get_root_position {
  my ($widget) = @_;
  my $window = $widget->window || return; # if unrealized
  require Gtk2::Ex::GdkBits;
  my ($x, $y) = Gtk2::Ex::GdkBits::window_get_root_position ($window);
  if ($widget->flags & 'no-window') {
    my $alloc = $widget->allocation;
    $x += $alloc->x;
    $y += $alloc->y;
  }
  return ($x, $y);
}
sub warp_pointer {
  my ($widget, $x, $y) = @_;
  my ($origin_x, $origin_y) = get_root_position ($widget)
    or croak "Cannot warp on unrealized $widget";
  my $screen = $widget->get_screen;
  my $display = $widget->get_display;
  $display->warp_pointer ($screen, $origin_x + $x, $origin_y + $y);
}

sub xy_root_to_widget {
  my ($widget, $root_x, $root_y) = @_;
  ### _xy_root_to_widget(): "$widget", $root_x, $root_y
  my ($x, $y) = Gtk2::Ex::WidgetBits::get_root_position ($widget);
  if (! defined $x) {
    ### widget unrealized
    return;
  } else {
    return ($root_x - $x, $root_y - $y);
  }
}

#------------------------------------------------------------------------------

sub xy_distance_mm {
  my ($widget, $x1, $y1, $x2, $y2) = @_;
  my ($xmm, $ymm) = pixel_size_mm ($widget)
    or return undef;
  return _hypot ($xmm * ($x1 - $x2),
                 $ymm * ($y1 - $y2));
}

sub pixel_aspect_ratio {
  my ($widget) = @_;
  my ($xmm, $ymm) = pixel_size_mm ($widget)
    or return undef;
  return ($xmm / $ymm);
}

if (Gtk2::Gdk::Screen->can('get_width')) {
  eval "#line ".(__LINE__+1)." \"".__FILE__."\"\n" . <<'HERE' or die;
  ### using widget->screen

sub pixel_size_mm {
  my ($widget) = @_;

  # gtk_widget_get_screen() returns the default screen if the widget is not
  # yet in a toplevel, so test gtk_widget_has_screen() first.  Often just
  # that default screen would be good enough, but best not to be sloppy with
  # that sort of thing if potentially multi-display.
  #
  $widget->has_screen
    or return; # no values
  my $screen = $widget->get_screen;
  return ($screen->get_width_mm / $screen->get_width,
          $screen->get_height_mm / $screen->get_height);

  # Pointless in Gtk 2.14, the monitor sizes are always -1.
  # Xinerama 1.1 doesn't give monitor sizes in millimetres, only pixel areas.
  #
  # For a multi-monitor screen an individual monitor size is used if
  # available.  Currently the calculation only looks at a single monitor
  # containing or nearest to the widget, using C<get_monitor_at_window>.  No
  # attempt is made to tell if the line x1,y1 to x2,y2 crosses multiple
  # monitors.
  #
  #   # xinerama new in Gtk 2.14 (2008) and Gtk2-Perl 1.191 (Aug 2008)
  #   if (my $func = $screen->can('get_monitor_at_window')) {
  #     # no position on the screen until realized
  #     if (my $win = $widget->window) {
  #       my $mnum = $func->($screen, $win);
  #       my $rect = $screen->get_monitor_geometry ($mnum);
  #       my $width_mm = $screen->get_monitor_width_mm ($mnum);
  #       my $height_mm = $screen->get_monitor_height_mm ($mnum);
  #       # sizes -1 if not known
  #       if ($width_mm != -1 && $height_mm != -1) {
  #         return ($width_mm / $rect->width,
  #                 $height_mm / $rect->height);
  #       }
  #     }
  #   }
}
1
HERE

} else {
  eval "#line ".(__LINE__+1)." \"".__FILE__."\"\n" . <<'HERE' or die;
  ### using Gtk 2.0.x single-screen size

# Gtk 2.0.x single-screen sizes
sub pixel_size_mm {
  return (Gtk2::Gdk->screen_width_mm / Gtk2::Gdk->screen_width,
          Gtk2::Gdk->screen_height_mm / Gtk2::Gdk->screen_height);
}
1
HERE
}

#------------------------------------------------------------------------------
# generic

# cf Math::Libm hypot()
sub _hypot {
  my ($x, $y) = @_;
  return sqrt ($x ** 2 + $y ** 2);
}

1;
__END__