The Perl and Raku Conference 2025: Greenville, South Carolina - June 27-29 Learn more

use Moose;
=head1 NAME
HTML::FormHandler::Field::Compound - field consisting of subfields
=head1 SYNOPSIS
This field class is designed as the base (parent) class for fields with
multiple subfields. Examples are L<HTML::FormHandler::DateTime>
and L<HTML::FormHandler::Duration>.
A compound parent class requires the use of sub-fields prepended
with the parent class name plus a dot
has_field 'birthdate' => ( type => 'DateTime' );
has_field 'birthdate.year' => ( type => 'Year' );
has_field 'birthdate.month' => ( type => 'Month' );
has_field 'birthdate.day' => ( type => 'MonthDay');
If all validation is performed in the parent class so that no
validation is necessary in the child classes, then the field class
'Nested' may be used.
Error messages will be applied to both parent classes and child
classes unless the 'errors_on_parent' flag is set. (This flag is
set for the 'Nested' field class.)
The process method of this field runs the process methods on the child fields
and then builds a hash of these fields values. This hash is available for
further processing by L<HTML::FormHandler::Field/actions> and the validate method.
Example:
has_field 'date_time' => (
type => 'Compound',
actions => [ { transform => sub{ DateTime->new( $_[0] ) } } ],
);
has_field 'date_time.year' => ( type => 'Text', );
has_field 'date_time.month' => ( type => 'Text', );
has_field 'date_time.day' => ( type => 'Text', );
=head2 widget
Widget type is 'compound'
=cut
has '+widget' => ( default => 'compound' );
sub BUILD
{
my $self = shift;
$self->build_fields;
}
augment 'process' => sub {
my $self = shift;
my $input = $self->input;
# this isn't right
if( ref $input eq 'HASH' )
{
foreach my $field ( $self->fields )
{
my $field_name = substr( $field->full_name, length($self->full_name) + 1 );
# Trim values and move to "input" slot
if ( exists $input->{$field_name} )
{
$field->input( $field->trim_value( $input->{$field_name} ) )
}
elsif ( $field->has_input_without_param )
{
$field->input( $field->input_without_param );
}
}
}
$self->clear_fif;
return unless $self->has_fields;
$self->fields_validate;
my %value_hash;
for my $field ( $self->fields )
{
$value_hash{ $field->accessor } = $field->value;
}
$self->value( \%value_hash );
};
# this is a kludge. We need to factor this stuff better...
# create the 'fif' for compound fields
sub _build_fif
{
my $self = shift;
my $fif;
for my $field ($self->fields)
{
$fif->{$field->full_name} = $field->fif;
}
$self->fif($fif) if $fif;
}
__PACKAGE__->meta->make_immutable;
no Moose;
1;