Mercurial > pub > Impl
view Lib/Form/Container.pm @ 166:4267a2ac3d46
Added Class::Template,
Rewritten SQL::Schema
'use parent' directive instead of 'use base'
author | wizard |
---|---|
date | Sat, 23 Apr 2011 23:12:06 +0400 (2011-04-23) |
parents | 76515373dac0 |
children |
line wrap: on
line source
package Form::Container; use strict; use Common; use Form::Filter; use parent qw(Form::Item); BEGIN { DeclareProperty Schema => ACCESS_READ; DeclareProperty Children => ACCESS_READ; } sub CTOR { my ($this,%args) = @_; $args{Schema} or die new Exception('A schema is required'); $this->SUPER::CTOR(@args{qw(Id Form Parent Attributes)}); $this->{$Schema} = $args{Schema}; } sub ResolveItem { my ($this,$ItemId) = @_; if (my $schemaChild = $this->{$Schema}->FindChild($ItemId->Name)) { if ($schemaChild->isMulti) { defined $ItemId->InstanceID or die new Exception('Instance id is required for a muti element'); if (my $child = $this->{$Children}{$ItemId->Name}[$ItemId->InstanceID]){ return $child; } else { return undef if not $this->Form->AutoCreate; return $this->{$Children}{$ItemId->Name}[$ItemId->InstanceID] = $this->Form->CreateInstance($schemaChild,$ItemId,$this); } } else { defined $ItemId->InstanceID and die new Exception('The child is a single element',$this->Id->Canonical,$ItemId->Name); if(my $child = $this->{$Children}{$ItemId->Name}) { return $child; } else { return undef if not $this->Form->AutoCreate; return $this->{$Children}{$ItemId->Name} = $this->Form->CreateInstance($schemaChild,$ItemId,$this); } } } else { die new Exception('The requested item isn\'t exists in the schema', $this->Id->Canonical,$ItemId->Name); } } sub isEmpty { my ($this) = @_; foreach my $child (values %{$this->{$Children} || {} }) { if (ref $child eq 'ARRAY') { foreach my $inst (@$child) { return 0 if not $child->isEmpty; } } else { return 0 if not $child->isEmpty; } } return 1; } =pod �������� �������� ��������� � ���� ������, ��� ��� ������ �� ������ ����������. ���� ��������� ��������� �� �������������, �� ������ ����� �������� �� ������ ��������. =cut sub GetChild { my ($this,$name) = @_; return unless exists $this->{$Children}{$name}; return( grep $_, map { UNIVERSAL::isa($_,'ARRAY') ? @{$_} : $_ } $this->{$Children}{$name} ); } =pod ��������� ������� �� ����� ��� ���� � ���� �����. ������� ������������ �� ����� � ���������� � �������� ���������� * ������� ��� ������, * ����� ��� �������, ������ ���� * ������� �������������, �� * ������� ��� ������ �������, � ����� * ��� ������� � ����������� =cut sub Validate { my ($this,$rhDisableFilters) = @_; $rhDisableFilters ||= {}; my @errors; foreach my $filter (grep {$_->SUPPORTED_CONTEXT & (Form::Filter::CTX_SINGLE) and not exists $rhDisableFilters->{$_}} map {$_->Instance} $this->{$Schema}->Filters) { my $result = $filter->Invoke($this,Form::Filter::CTX_SINGLE | Form::Filter::CTX_EXISTENT,$this->{$Schema}); if ($result->State == Form::FilterResult::STATE_SUCCESS_STOP) { return (); } elsif ($result->State == Form::FilterResult::STATE_ERROR) { push @errors,$result; } } CHILD_LOOP: foreach my $schemaChild ($this->{$Schema}->Children) { if ($schemaChild->isMulti) { my %DisableFilters; foreach my $filter (grep {$_->SUPPORTED_CONTEXT & Form::Filter::CTX_SET} map {$_->Instance} $schemaChild->Filters) { my $result = $filter->Invoke($this->{$Children}{$schemaChild->Name},Form::Filter::CTX_SET,$schemaChild,$this); if ($result->State == Form::FilterResult::STATE_ERROR) { push @errors,$result; # �� ��������� ������ ������� ������ next CHILD_LOOP; } elsif ($result->State == Form::FilterResult::STATE_SUCCESS_STOP) { # �� ��������� ������ ������� ������ next CHILD_LOOP; } elsif ($result->State == Form::FilterResult::STATE_SUCCESS_STAY) { # �� ��������� ������ ������ �� ������ ���������� $DisableFilters{$filter} = 1; } else { # STATE_SUCCESS - ��� �� } } $_ and push @errors,$_->Validate(\%DisableFilters) foreach grep !$_->isEmpty, $this->GetChild($schemaChild->Name); } else { my %DisableFilters; # ��������� �������, ������� ����� ����������� �� �������������� �������� foreach my $filter (grep { $_->SUPPORTED_CONTEXT & Form::Filter::CTX_SINGLE and not $_->SUPPORTED_CONTEXT & Form::Filter::CTX_EXISTENT} map {$_->Instance} $schemaChild->Filters) { my $result = $filter->Invoke($this->{$Children}{$schemaChild->Name},Form::Filter::CTX_SINGLE,$schemaChild,$this); if ($result->State == Form::FilterResult::STATE_ERROR) { push @errors,$result; # �� ��������� ������ ������� ������ next CHILD_LOOP; } elsif ($result->State == Form::FilterResult::STATE_SUCCESS_STOP) { # �� ��������� ������ ������� ������ next CHILD_LOOP; } else { # STATE_SUCCESS(_STAY) - ��� �� $DisableFilters{$filter} = 1; } } # ���� �������� ����������, �� ��������� ���������� ������� push @errors,$this->{$Children}{$schemaChild->Name}->Validate(\%DisableFilters) if $this->{$Children}{$schemaChild->Name}; } } return @errors; } sub Dispose { my ($this) = @_; foreach my $child (values %{ $this->{$Children} || {} }) { if (ref $child eq 'ARRAY') { foreach my $inst (@$child) { $inst->Dispose; } } else { die new IMPL::Exception("Child is null",%{ $this->{$Children} }) if not $child; $child->Dispose; } } delete @$this{$Schema,$Children}; $this->SUPER::Dispose; } 1;