Projekt

Allgemein

Profil

Herunterladen (3,67 KB) Statistiken
| Zweig: | Markierung: | Revision:
6b293028 Moritz Bunkus
package SL::MoreCommon;

require Exporter;
76c486e3 Sven Schöling
our @ISA = qw(Exporter);
6b293028 Moritz Bunkus
76c486e3 Sven Schöling
our @EXPORT = qw(save_form restore_form compare_numbers any cross);
our @EXPORT_OK = qw(ary_union ary_intersect ary_diff listify);
6b293028 Moritz Bunkus
use YAML;

8c7e4493 Moritz Bunkus
use SL::AM;

76c486e3 Sven Schöling
use strict;

6b293028 Moritz Bunkus
sub save_form {
$main::lxdebug->enter_sub();

8c7e4493 Moritz Bunkus
my @dont_dump_keys = @_;
my %not_dumped_values;

foreach my $key (@dont_dump_keys) {
$not_dumped_values{$key} = $main::form->{$key};
delete $main::form->{$key};
}

6b293028 Moritz Bunkus
my $old_form = YAML::Dump($main::form);
20515902 Moritz Bunkus
$old_form =~ s|!|!:|g;
6b293028 Moritz Bunkus
$old_form =~ s|\n|!n|g;
$old_form =~ s|\r|!r|g;

8c7e4493 Moritz Bunkus
map { $main::form->{$_} = $not_dumped_values{$_} } keys %not_dumped_values;

6b293028 Moritz Bunkus
$main::lxdebug->leave_sub();

return $old_form;
}

sub restore_form {
$main::lxdebug->enter_sub();

8c7e4493 Moritz Bunkus
my ($old_form, $no_delete, @keep_vars) = @_;
6b293028 Moritz Bunkus
8c7e4493 Moritz Bunkus
my $form = $main::form;
my %keep_vars_map = map { $_ => 1 } @keep_vars;
6b293028 Moritz Bunkus
8c7e4493 Moritz Bunkus
map { delete $form->{$_} if (!$keep_vars_map{$_}); } keys %{$form} unless ($no_delete);
6b293028 Moritz Bunkus
$old_form =~ s|!r|\r|g;
$old_form =~ s|!n|\n|g;
805bd930 Moritz Bunkus
$old_form =~ s|![!:]|!|g;
6b293028 Moritz Bunkus
my $new_form = YAML::Load($old_form);
8c7e4493 Moritz Bunkus
map { $form->{$_} = $new_form->{$_} if (!$keep_vars_map{$_}) } keys %{ $new_form };
6b293028 Moritz Bunkus
$main::lxdebug->leave_sub();
}

8c7e4493 Moritz Bunkus
sub compare_numbers {
$main::lxdebug->enter_sub();

my $a = shift;
my $a_unit = shift;
my $b = shift;
my $b_unit = shift;

$main::all_units ||= AM->retrieve_units(\%main::myconfig, $main::form);
my $units = $main::all_units;

if (!$units->{$a_unit} || !$units->{$b_unit} || ($units->{$a_unit}->{base_unit} ne $units->{$b_unit}->{base_unit})) {
$main::lxdebug->leave_sub();
return undef;
}

$a *= $units->{$a_unit}->{factor};
$b *= $units->{$b_unit}->{factor};

$main::lxdebug->leave_sub();

return $a <=> $b;
}

4f82ce00 Moritz Bunkus
sub any (&@) {
my $f = shift;
return if ! @_;
for (@_) {
return 1 if $f->();
}
return 0;
}
8c7e4493 Moritz Bunkus
f30b6f52 Sven Schöling
=item cross BLOCK ARRAY ARRAY

Evaluates BLOCK for each combination of elements in ARRAY1 and ARRAY2
and returns a new list consisting of BLOCK's return values.
The two elements are set to $a and $b.
Note that those two are aliases to the original value so changing them
will modify the input arrays.

# append each to each
@a = qw/a b c/;
@b = qw/1 2 3/;
bc40c989 Sven Schöling
@x = cross { "$a$b" } @a, @b;
f30b6f52 Sven Schöling
# returns a1, a2, a3, b1, b2, b3, c1, c2, c3

As cross expects an array but returns a list it is not directly chainable
at the moment. This will be corrected in the future.

=cut
sub cross(&\@\@) {
my $op = shift;
use vars qw/@A @B/;
local (*A, *B) = @_; # syms for caller's input arrays

# Localise $a, $b
my ($caller_a, $caller_b) = do {
my $pkg = caller();
no strict 'refs';
\*{$pkg.'::a'}, \*{$pkg.'::b'};
};

local(*$caller_a, *$caller_b);

# This map expression is also the return value.
map { my $a_index = $_;
map { my $b_index = $_;
# assign to $a, $b as refs to caller's array elements
(*$caller_a, *$caller_b) = \($A[$a_index], $B[$b_index]);
$op->(); # perform the transformation
} 0 .. $#B;
} 0 .. $#A;
}

46e1506b Moritz Bunkus
sub _ary_calc_union_intersect {
my ($a, $b) = @_;

my %count = ();

foreach my $e (@$a, @$b) { $count{$e}++ }

my @union = ();
my @isect = ();
foreach my $e (keys %count) {
push @union, $e;
push @isect, $e if $count{$e} == 2;
}

return (\@union, \@isect);
}

sub ary_union {
return @{ (_ary_calc_union_intersect @_)[0] };
}

sub ary_intersect {
return @{ (_ary_calc_union_intersect @_)[1] };
}

sub ary_diff {
my ($a, $b) = @_;
my %in_b = map { $_ => 1 } @$b;
return grep { !$in_b{$_} } @$a;
}

67223abf Sven Schöling
sub listify {
my @ary = scalar @_ > 1 ? @_ : ref $_[0] eq 'ARRAY' ? @{ $_[0] } : (@_);
return wantarray ? @ary : scalar @ary;
}
+

6b293028 Moritz Bunkus
1;