aboutsummaryrefslogtreecommitdiffstats
path: root/lib/Reaction/UI/ViewPort/Field/Mutable/ChooseMany.pm
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