blob: 227ad33b03900ec435b537a4023e6aacf190d27b (
plain) (
blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
|
package Reaction::UI::ViewPort::Field::Mutable::ChooseMany;
use Reaction::Class;
my $listify = sub{
return [] unless defined($_[0]);
return ref $_[0] eq 'ARRAY' ? $_[0] : [$_[0]];
};
use namespace::clean -except => [ qw(meta) ];
use MooseX::Types::Moose qw/ArrayRef/;
extends 'Reaction::UI::ViewPort::Field';
with 'Reaction::UI::ViewPort::Field::Role::Mutable' => { value_type => 'ArrayRef' };
with 'Reaction::UI::ViewPort::Field::Role::Choices';
#MUST BE HERE, BELOW THE 'does', OR THE TRIGGER WILL NOT HAPPEN!
#has '+value' => (isa => ArrayRef);
around value => sub {
my $orig = shift;
my $self = shift;
return $orig->($self) unless @_;
my $value = $listify->(shift);
$_ = $self->str_to_ident($_) for @$value;
my $checked = $self->attribute->check_valid_value($self->model, $value);
# i.e. fail if any of the values fail
confess "Not a valid set of values"
if (@$checked < @$value || grep { !defined($_) } @$checked);
$orig->($self, $checked);
};
around _value_string_from_value => sub {
my $orig = shift;
my $self = shift;
join(", ", (map {$self->obj_to_name($_->{value}) } @{ $self->current_value_choices }));
};
sub is_current_value {
my ($self, $check_value) = @_;
return unless $self->_model_has_value;
my @our_values = @{$self->value || []};
$check_value = $self->obj_to_str($check_value) if ref($check_value);
return grep { $self->obj_to_str($_) eq $check_value } @our_values;
};
sub current_value_choices {
my $self = shift;
my @all = grep { $self->is_current_value($_->{value}) } @{$self->value_choices};
return [ @all ];
};
sub available_value_choices {
my $self = shift;
my @all = grep { !$self->is_current_value($_->{value}) } @{$self->value_choices};
return [ @all ];
};
around handle_events => sub {
my $orig = shift;
my ($self, $events) = @_;
$events->{value} = [] if $events->{no_current_value};
my $ev_value = $listify->($events->{value});
if (delete $events->{add_all_values}) {
$events->{value} = [map {$self->obj_to_str($_)} @{$self->valid_values}];
} elsif (exists $events->{add_values} && delete $events->{do_add_values}) {
my $add = $listify->(delete $events->{add_values});
$events->{value} = [ @{$ev_value}, @$add ];
} elsif (delete $events->{remove_all_values}) {
$events->{value} = [];
}elsif (exists $events->{remove_values} && delete $events->{do_remove_values}) {
my $remove = $listify->(delete $events->{remove_values});
my %r = map { ($_ => 1) } @$remove;
$events->{value} = [ grep { !$r{$_} } @{$ev_value} ];
}
return $orig->(@_);
};
__PACKAGE__->meta->make_immutable;
1;
=head1 NAME
Reaction::UI::ViewPort::Field::Mutable::ChooseMany
=head1 DESCRIPTION
=head1 METHODS
=head2 is_current_value
=head2 current_values
=head2 available_values
=head2 available_value_names
=head1 SEE ALSO
=head2 L<Reaction::UI::ViewPort::Field>
=head1 AUTHORS
See L<Reaction::Class> for authors.
=head1 LICENSE
See L<Reaction::Class> for the license.
=cut
|