# Copyright (c) 2003-2009, Mikhael Goikhman # # This program 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 2 of the License, or # (at your option) any later version. # # This program 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 this program; if not, write to the Free Software # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA package FVWM::EventNames; use strict; use FVWM::Constants; use constant number => 0; use constant bool => 1; use constant window => 2; use constant pixel => 3; use constant string => 4; use constant wflags => 5; use constant looped => 6; use vars qw($EVENTS_INFO); # ---------------------------------------------------------------------------- # start of the would-be-generated part $EVENTS_INFO = { &M_NEW_PAGE => { format => "l!3L!4", fields => [ vp_x => number, vp_y => number, desk_n => number, vp_width => number, vp_height => number, desk_pages_x => number, desk_pages_y => number, ], aliases => { desk => 'desk_n', }, }, &M_NEW_DESK => { format => "l!", fields => [ desk_n => number, ], aliases => { desk => 'desk_n', }, }, # &M_OLD_ADD_WINDOW => { # }, &M_RAISE_WINDOW => { format => "L!3", fields => [ win_id => window, frame_id => window, ptr => number, ], }, &M_LOWER_WINDOW => { format => "L!3", fields => [ win_id => window, frame_id => window, ptr => number, ], }, # &M_OLD_CONFIGURE_WINDOW => { # }, &M_FOCUS_CHANGE => { format => "L!5", fields => [ win_id => window, frame_id => window, flip => bool, focus_fg => pixel, focus_bg => pixel, ], }, &M_DESTROY_WINDOW => { format => "L!3", fields => [ win_id => window, frame_id => window, ptr => number, ], }, &M_ICONIFY => { format => "L!3l!8", fields => [ win_id => window, frame_id => window, ptr => number, icon_x => number, icon_y => number, icon_width => number, icon_height => number, frame_x => number, frame_y => number, frame_width => number, frame_height => number, ], }, &M_DEICONIFY => { format => "L!3l!8", fields => [ win_id => window, frame_id => window, ptr => number, icon_x => number, icon_y => number, icon_width => number, icon_height => number, frame_x => number, frame_y => number, frame_width => number, frame_height => number, ], }, &M_WINDOW_NAME => { format => "L!3a*", fields => [ win_id => window, frame_id => window, ptr => number, name => string, ], }, &M_ICON_NAME => { format => "L!3a*", fields => [ win_id => window, frame_id => window, ptr => number, name => string, ], }, &M_RES_CLASS => { format => "L!3a*", fields => [ win_id => window, frame_id => window, ptr => number, name => string, ], }, &M_RES_NAME => { format => "L!3a*", fields => [ win_id => window, frame_id => window, ptr => number, name => string, ], }, &M_END_WINDOWLIST => { format => "", fields => [ ], }, &M_ICON_LOCATION => { format => "L!3l!4", fields => [ win_id => window, frame_id => window, ptr => number, x => number, y => number, width => number, height => number, ], }, &M_MAP => { format => "L!3", fields => [ win_id => window, frame_id => window, ptr => number, ], }, &M_ERROR => { format => "L!3a*", fields => [ win_id => window, frame_id => window, ptr => number, text => string, ], }, &M_CONFIG_INFO => { format => "L!3a*", fields => [ win_id => window, frame_id => window, ptr => number, text => string, ], }, &M_END_CONFIG_INFO => { format => "", fields => [ ], }, &M_ICON_FILE => { format => "L!3a*", fields => [ win_id => window, frame_id => window, ptr => number, name => string, ], }, &M_DEFAULTICON => { format => "L!3a*", fields => [ win_id => window, frame_id => window, ptr => number, name => string, ], }, &M_STRING => { format => "L!3a*", fields => [ win_id => window, frame_id => window, ptr => number, text => string, ], }, &M_MINI_ICON => { format => "L!9a*", fields => [ win_id => window, frame_id => window, ptr => number, width => number, height => number, depth => number, icon_id => window, mask => number, alpha => number, name => string, ], }, &M_WINDOWSHADE => { format => "L!3", fields => [ win_id => window, frame_id => window, ptr => number, ], }, &M_DEWINDOWSHADE => { format => "L!3", fields => [ win_id => window, frame_id => window, ptr => number, ], }, &M_VISIBLE_NAME => { format => "L!3a*", fields => [ win_id => window, frame_id => window, ptr => number, name => string, ], }, # &M_SENDCONFIG => { # }, &M_RESTACK => { format => "L!3a*", fields => [ win_id => window, frame_id => window, ptr => number, low_windows => looped, ], loop_format => "L!3a*", loop_fields => [ win_id => window, frame_id => window, ptr => number, ], }, &M_ADD_WINDOW => { format => "L!3l!6l!8L!2l!L!2l!3S4a*", fields => [ win_id => window, frame_id => window, ptr => number, frame_x => number, frame_y => number, frame_width => number, frame_height => number, desk => number, layer => number, win_width => number, win_height => number, resize_width_inc => number, resize_height_inc => number, minimum_width => number, minimum_height => number, maximum_width => number, maximum_height => number, icon_title_id => window, icon_image_id => window, gravity => number, fore_color => pixel, back_color => pixel, ewmh_layer => number, ewmh_desktop => number, ewmh_window_type => number, title_height => number, border_width => number, dummy_zero_1 => number, dummy_zero_2 => number, window_flags => wflags, ], }, &M_CONFIGURE_WINDOW => { format => "L!3l!6l!8L!2l!L!2l!3S4a*", fields => [ win_id => window, frame_id => window, ptr => number, frame_x => number, frame_y => number, frame_width => number, frame_height => number, desk => number, layer => number, win_width => number, win_height => number, resize_width_inc => number, resize_height_inc => number, minimum_width => number, minimum_height => number, maximum_width => number, maximum_height => number, icon_title_id => window, icon_image_id => window, gravity => number, fore_color => pixel, back_color => pixel, ewmh_layer => number, ewmh_desktop => number, ewmh_window_type => number, title_height => number, border_width => number, dummy_zero_1 => number, dummy_zero_2 => number, window_flags => wflags, ], }, # &M_EXTENDED_MSG => { # }, &MX_VISIBLE_ICON_NAME => { format => "L!3a*", fields => [ win_id => window, frame_id => window, ptr => number, name => string, ], }, &MX_ENTER_WINDOW => { format => "L!3", fields => [ win_id => window, frame_id => window, ptr => number, ], }, &MX_LEAVE_WINDOW => { format => "L!3", fields => [ win_id => window, frame_id => window, ptr => number, ], }, &MX_PROPERTY_CHANGE => { format => "L!3a*", fields => [ type => number, value => number, win_id => window, text => string, ], info => { type_names => [ qw(NONE BACKGROUND SWALLOW) ], }, }, &MX_REPLY => { format => "L!3a*", fields => [ win_id => window, frame_id => window, ptr => number, text => string, ], }, "faked" => { format => "", fields => [ ], }, }; # end of the would-be-generated part # ---------------------------------------------------------------------------- use Exporter; use vars qw(@EXPORT @ISA $EVENT_TYPES $EVENT_NAMES $EVENT_TYPE_NAMES); @EXPORT = ( @FVWM::Constants::EXPORT, qw(event_name event_arg_names event_arg_types event_arg_values event_args), qw(event_loop_arg_names event_loop_arg_types), qw(event_arg_aliases all_event_names all_event_types) ); @ISA = qw(Exporter); sub all_event_type_names () { if (!defined $EVENT_TYPE_NAMES) { $EVENT_TYPES = []; $EVENT_NAMES = []; $EVENT_TYPE_NAMES = {}; my ($type, $name); foreach $name (@FVWM::Constants::EXPORT) { next unless $name =~ /^MX?_/; next if $name eq 'M_EXTENDED_MSG'; no strict 'refs'; $type = &$name(); next if $name =~ /^MX_/ && !($type & M_EXTENDED_MSG); push @$EVENT_TYPES, $type; push @$EVENT_NAMES, $name; $EVENT_TYPE_NAMES->{$type} = $name; } } return $EVENT_TYPE_NAMES; } sub all_event_names () { all_event_type_names(); return wantarray? @$EVENT_NAMES: $EVENT_NAMES; } sub all_event_types () { all_event_type_names(); return wantarray? @$EVENT_TYPES: $EVENT_TYPES; } sub event_name ($) { my $type = shift; return all_event_type_names()->{$type}; } sub event_type_to_binary ($) { my $type = shift || "no-event-type"; return $type unless $type =~ /^\d+$/; return sprintf("%b", $type); } sub event_info ($) { my $type = shift; unless (defined $EVENTS_INFO->{$type}) { die "FVWM::EventNames: Unknown event type (" . event_type_to_binary($type) . ")\n"; } return $EVENTS_INFO->{$type}; } sub calculate_internals ($) { my $type = shift; my $event_info = event_info($type); $event_info->{names} = []; $event_info->{types} = []; my $i = 0; foreach (@{$event_info->{fields}}) { push @{$event_info->{names}}, $_ if ($i % 2) == 0; push @{$event_info->{types}}, $_ if ($i % 2) == 1; $i++; } # handle loop args if any return unless exists $event_info->{loop_fields}; $event_info->{loop_names} = []; $event_info->{loop_types} = []; $i = 0; foreach (@{$event_info->{loop_fields}}) { push @{$event_info->{loop_names}}, $_ if ($i % 2) == 0; push @{$event_info->{loop_types}}, $_ if ($i % 2) == 1; $i++; } } sub event_arg_names ($$) { my $type = shift; my $arg_values = shift; my $event_info = event_info($type); my $arg_names = $event_info->{names}; return $arg_names if defined $arg_names; calculate_internals($type); return $event_info->{names}; } sub event_arg_types ($$) { my $type = shift; my $arg_values = shift; my $event_info = event_info($type); my $arg_types = $event_info->{types}; return $arg_types if defined $arg_types; calculate_internals($type); return $event_info->{types}; } sub event_arg_values ($$) { my $type = shift; my $packed_str = shift; my $event_info = event_info($type); my @arg_values = unpack($event_info->{format}, $packed_str); my $arg_fields = $event_info->{fields}; push @arg_values, (undef) x ((@$arg_fields / 2) - @arg_values); # process looped args if (@$arg_fields && $arg_fields->[@$arg_fields - 1] == looped) { my @loop_arg_values = (); my $rest_str = pop @arg_values; while ($rest_str ne "") { my @new_arg_values = unpack($event_info->{loop_format}, $rest_str); die "Internal error, no loop args unpacked ($type)\n" unless @new_arg_values > 1; $rest_str = pop @new_arg_values; push @loop_arg_values, @new_arg_values; } push @arg_values, \@loop_arg_values; } # strip everything past the first null (or newline) if needed if (@$arg_fields && $arg_fields->[-1] == string) { $arg_values[-1] =~ s/\n*\0.*//s; } return \@arg_values; } sub event_loop_arg_names ($$) { my $type = shift; my $arg_values = shift; my $event_info = event_info($type); my $arg_names = $event_info->{loop_names}; return $arg_names if defined $arg_names; calculate_internals($type); return $event_info->{loop_names}; } sub event_loop_arg_types ($$) { my $type = shift; my $arg_values = shift; my $event_info = event_info($type); my $arg_types = $event_info->{loop_types}; return $arg_types if defined $arg_types; calculate_internals($type); return $event_info->{loop_types}; } sub event_args ($$) { my $type = shift; my $arg_values = shift; my $arg_names = event_arg_names($type, $arg_values); die sprintf "Internal error, event type %s (%d names, %d values)\n", event_type_to_binary($type), scalar @$arg_names, scalar @$arg_values if @$arg_names != @$arg_values; my $loop_arg_names = event_loop_arg_names($type, $arg_values); die sprintf "Internal error, event type %s (%d loop names, non array)\n", event_type_to_binary($type), scalar @$loop_arg_names if $loop_arg_names && ref($loop_arg_names) ne 'ARRAY' && !@$loop_arg_names && ref($arg_values->[-1]) ne 'ARRAY'; my $i = 0; my %args = map { my $value = $_; $arg_names->[$i++], ref($value) ne 'ARRAY'? $value: do { my $loop_value = []; my $j = 0; while ($j < @$value) { my %loop_hash = map { $_, $value->[$j++] } @$loop_arg_names; push @$loop_value, \%loop_hash; } $loop_value } } @$arg_values; return \%args; } sub event_arg_aliases ($) { my $type = shift; return event_info($type)->{aliases} || {}; } # ---------------------------------------------------------------------------- =head1 NAME FVWM::EventNames - names and types of all fvwm event arguments =head1 SYNOPSIS use FVWM::EventNames; print "All event names: ", join(", ", @{all_event_names()}), "\n"; print "All event types: ", join(", ", @{all_event_types()}), "\n"; my $name = event_name (M_ICON_LOCATION); my $arg_values = event_arg_values(M_ICON_LOCATION, $packed_str); my $arg_names = event_arg_names (M_ICON_LOCATION, $arg_values); my $arg_types = event_arg_types (M_ICON_LOCATION, $arg_values); my $args = event_args (M_ICON_LOCATION, $arg_values); =head1 DESCRIPTION Every event send by I consist of arguments. The argument names and types vary from one event type to another. For example, event of the type B consists of only one argument I of type I. B consists of 5 numeric arguments, B consists of 10 arguments of different types. This class provides information about all fvwm events. It provides such services as listing all supported event types and their names, converting event type to event name, listing the event argument names/types, constructing event argument values from the plain packet data. Usually you do not need to work with this class directly, but, instead, with B objects. Hovewer, you may need this class source as a reference for the names of the event arguments and their types. =head1 PUBLIC FUNCTIONS =over 4 =item B I Returns the string representation of the numeric event I constant, like I or I. =item B I I Constructs array ref of argument values for the event I from the I (as received from I). If the last argument type of the event is string, for convenience, everything past the first null (or newline) is automatically stripped from the last argument value. =item B I I Returns array ref of argument names of the event type. I is either the real array ref of values (as returned by B) or a number of actual values. The returned array has the same number of elements. =item B I I Returns array ref of argument types of the event type. I is either the real array ref of values (as returned by B) or a number of actual values. The returned array has the same number of elements. =item B I I Returns array ref of looped argument names of the event type (or undef). =item B I I Returns array ref of looped argument types of the event type (or undef). =item B I I Constructs hash ref of the named arguments for the event I from the I array ref (as returned by B). =item B I This method is provided for backward compatibility when argument names are changed. For example, in the past the argument name of I was B, but now it is B. Using this method it is possible to make both names supported. Returns hash ref (old-name => new-name). =item B Returns array ref of all known event names (strings). In the list context returns list of these names. =item B Returns array ref of all known event types (numbers). In the list context returns list of these types. =item B Returns hash ref of all known event names and types (type => name). =back =head1 SEE ALSO L. =cut 1;