The Perl Toolchain Summit needs more sponsors. If your company depends on Perl, please support this very important event.
package VoiceXML::Client::Document;



use strict;

use Data::Dumper;
use base qw(VoiceXML::Client::Item);


use vars qw{
		$VERSION
};

$VERSION = $VoiceXML::Client::Item::VERSION;


=head1 COPYRIGHT AND LICENSE

	
	Copyright (C) 2007,2008 by Pat Deegan.
	All rights reserved
	http://voicexml.psychogenic.com

This library is released under the terms of the GNU GPL version 3, making it available only for 
free programs ("free" here being used in the sense of the GPL, see http://www.gnu.org for more details). 
Anyone wishing to use this library within a proprietary or otherwise non-GPLed program MUST contact psychogenic.com to 
acquire a distinct license for their application.  This approach encourages the use of free software 
while allowing for proprietary solutions that support further development.


This file is part of VoiceXML::Client.

 
 
    VoiceXML::Client is free software: you can redistribute it and/or modify
    it under the terms of the GNU General Public License as published by
    the Free Software Foundation, either version 3 of the License, or
    (at your option) any later version.

    VoiceXML::Client is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    GNU General Public License for more details.

    You should have received a copy of the GNU General Public License
    along with VoiceXML::Client.  If not, see <http://www.gnu.org/licenses/>.


=cut


=head2 new 

=cut
sub new {
	my $class = shift;
	my $docname = shift;
	my $runtime = shift;
	
	
	
	my $self = {};
        
        bless $self, ref $class || $class;
	
	
	$self->{'type'} = ref $class || $class;
	$self->{'_runtime'} = $runtime;
	$self->{'_context'} = $runtime->create_context();
	$self->{'docname'} = $docname;
	
	$self->{'forms'} = [];
	$self->{'formindices'} = {};
	
	$self->{'variables'} = {};
	
	$self->{'nextform'} = 0;
	$self->{'nextdocument'} = 0;
	$self->{'currentitem'} = 0;
	
	
	return $self;
}

sub init {
	my $self = shift;
	
	$self->{'items'} = [];
	
	return 1;
}


=head2 addItem ITEM

Appends ITEM to the list of items for the document.  Items may be any VXML::Item::XXX subclass, while
ITEM itself may be a single ITEM object reference or an ARRAY ref of VXML::Item::XXX objects (each will
be appended in order).

Returns the number of appended items.

=cut

sub addItem {
	my $self = shift;
	my $item = shift;
	
	
	my $count = 0;
	if (ref $item eq 'ARRAY')
	{
		foreach my $singleItem (@{$item})
		{
			$count += $self->addItem($singleItem);
		}
	} else {
	
		push @{$self->{'items'}}, $item;
		$count = 1;
	}
	
	return $count;
	
}

sub nextFormId {
	my $self = shift;
	my $setTo = shift; # optional
	
	if (defined $setTo && exists $self->{'formindices'}->{$setTo})
	{
		$self->{'nextform'} = $setTo;
		my $nextFormIdx = $self->getFormItemsIndex($setTo);
		
		$self->{'currentitem'} = $nextFormIdx;
		$self->{'items'}->[$nextFormIdx]->reset();
		
		return $setTo;
		
	}
	
	my $idx = 0;
	if ($self->{'nextform'})
	{
		my $nextFormName = $self->{'nextform'};
		
		if (exists $self->{'formindices'}->{$nextFormName}
			&& ( exists $self->{'forms'}->[$self->{'formindices'}->{$nextFormName}] )
		)
		{
			$idx = $self->{'formindices'}->{$nextFormName};
			$self->{'variables'}->{$self->{'forms'}->[$idx]->{'guard'}} = undef;
		}
	}
	
	for (my $i = $idx; $i < scalar @{$self->{'forms'}}; $i++)
	{
	
		if ($self->{'variables'}->{$self->{'forms'}->[$i]->{'guard'}})
		{
			VoiceXML::Client::Util::log_msg("nextFormId() Skipping form " . $self->{'forms'}->[$i]->{'name'} 
					. ' because guard condition is set') if ($VoiceXML::Client::Debug > 1);
		} else {
			VoiceXML::Client::Util::log_msg("nextFormId() returning form id " . $self->{'forms'}->[$i]->{'name'} )
				if ($VoiceXML::Client::Debug);
			return $self->{'forms'}->[$i]->{'name'} ;
		}
	}
	
	return undef;
}

sub getForm {
	my $self = shift;
	my $id = shift ;
	
	my $idx = $self->getFormItemsIndex($id);
	
	
	return undef unless (defined $idx);

	return $self->{'items'}->[$idx];
	
}

sub getFormItemsIndex {
	my $self = shift;
	my $id = shift ;
	
	return undef unless defined ($id);

	for (my $i=0; $i < scalar @{$self->{'items'}}; $i++)
	{
		my $itm = $self->{'items'}->[$i];
		
		if ($itm->{'type'} eq 'VoiceXML::Client::Item::Form'
				&& $itm->{'id'} eq $id)
		{
			VoiceXML::Client::Util::log_msg("Found form $id att idx $i")
				if ($VoiceXML::Client::Debug > 1);
			
			return $i ;
		}
				
	}
	
	return undef;
}
	


sub getNextForm {
	my $self = shift;
	
	my $id = $self->nextFormId();
	
	
	return $self->getForm($id);
	
}

sub getNextItem {
	my $self = shift;
	
	return $self->getNextForm() if ($self->{'nextform'});
	
	
	$self->{'currentitem'} ||= 0;
	
	my $oldChildSetting = $self->{'currentitem'} ;
	my $i = $oldChildSetting;
	while ($i < scalar @{$self->{'items'}})
	{
	
		$self->{'currentitem'} = $i;
		
		my $cType = $self->{'items'}->[$i]->getType();
		if ($cType ne 'VoiceXML::Client::Item::Form')
		{
			# not a form, just return it in sequence
			return $self->{'items'}->[$i];
		
		} else 
		{
			# it IS a form... handle guard vars...
			my $formName = $self->{'items'}->[$i]->id();
			
			if (exists $self->{'formindices'}->{$formName})
			{
				my $formIdx = $self->{'formindices'}->{$formName};
				
				if (! $self->{'variables'}->{$self->{'forms'}->[$formIdx]->{'guard'}})
				{
					# not visited yet... go for it.
					return $self->{'items'}->[$i];
				}
			}
		}
		
		
		$i++;
		
	}
	
	$self->{'currentitem'} = $oldChildSetting;
	
	return undef;
}

sub currentItemPosition {
	my $self = shift;
	my $setTo = shift; # optional
	
	if (defined $setTo && $setTo =~ m/^\d+$/ && $setTo < scalar @{$self->{'items'}})
	{
		$self->{'currentitem'} = $setTo;
	}
	
	return $self->{'currentitem'};
}

sub proceedToNextItem {
	my $self = shift;
	
	return $self->{'currentitem'}++;
	
}			


sub registerForm {
	my $self = shift;
	my $formName = shift;
	my $guard = shift;
	
	push @{$self->{'forms'}}, {
				'name'	=> $formName,
				'guard'	=> $guard};
				
	$self->{'formindices'}->{$formName} = $#{$self->{'forms'}};
	
				
	$self->registerVariable($formName);
}

sub registerVariable {
	my $self = shift;
	my $varName = shift || return;
	my $val = shift;
	
	if (defined $val)
	{
		$self->{'variables'}->{$varName} = $val;
	} else {
		
		$self->{'variables'}->{$varName}  = undef;
	}
	
}

sub globalVar {
	my $self = shift;
	my $varName = shift || return;
	my $val = shift; #optional
	
	
	if (defined $val)
	{
		$self->{'variables'}->{$varName} = $val;
		
		VoiceXML::Client::Util::log_msg("SETTING $varName to '$val'") if ($VoiceXML::Client::Debug > 1);
	}
	
	$self->{'variables'}->{$varName} = undef unless (exists $self->{'variables'}->{$varName});
		
	
	return $self->{'variables'}->{$varName};
}

sub clearGlobalVar {
	my $self = shift;
	my $varName = shift || return;
	
	if (exists $self->{'variables'}->{$varName})
	{
		$self->{'variables'}->{$varName} = undef;
	}
}
	
sub pushPositionStack {
	my $self  = shift;

	unshift @{$self->{'itempositionstack'}}, $self->{'currentitem'} ;
	
	
}

sub latestPositionInStack {
	my $self  = shift;
	
	
	if (scalar @{$self->{'itempositionstack'}})
	{
		return $self->{'itempositionstack'}->[0];
	}
	
	return undef;
	
}

sub popPositionStack {
	my $self = shift;
	
	
	if (scalar @{$self->{'itempositionstack'}})
	{
		return shift @{$self->{'itempositionstack'}};
	}

	return undef;
}

sub execute {
	my $self = shift;
	my $handle = shift || return undef;
	my $params = shift || {};
	my $itemToExec = shift ||  $self->getNextItem();
	
	unless ($itemToExec)
	{
		warn "No form or other item to execute";
		return undef;
	}
	
	my $retVal = $VoiceXML::Client::Flow::Directive{'CONTINUE'};
	
	do {
		
		$self->{'nextform'} = undef; # make certain we start with a clean slate
		
		$retVal = $itemToExec->execute($handle, $params);
	
		$self->proceedToNextItem() if ($retVal != $VoiceXML::Client::Flow::Directive{'JUMP'});
		$itemToExec = $self->getNextItem();
		
		if ($retVal == $VoiceXML::Client::Flow::Directive{'SUBRETURN'})
		{
			my $latestPosInStack = $self->latestPositionInStack();
			if (defined $latestPosInStack)
			{
				$self->popPositionStack();
				$self->{'currentitem'} = $latestPosInStack;
				$itemToExec = $self->{'items'}->[$latestPosInStack];
				$retVal = $VoiceXML::Client::Flow::Directive{'CONTINUE'};
			}
		}
		
				
		
	} while ($itemToExec &&
			($retVal == $VoiceXML::Client::Flow::Directive{'CONTINUE'}
				||
			$retVal == $VoiceXML::Client::Flow::Directive{'JUMP'}));
	
	
	if ($retVal == $VoiceXML::Client::Flow::Directive{'CONTINUE'})
	{
	
		
		warn "Made it through the entire VXML document but instructed to continue--aborting";
		return $VoiceXML::Client::Flow::Directive{'ABORT'};
	}

	return $retVal if ($retVal == $VoiceXML::Client::Flow::Directive{'ABORT'}
				|| $retVal == $VoiceXML::Client::Flow::Directive{'DONE'});
	
	if ($retVal == $VoiceXML::Client::Flow::Directive{'NEXTDOC'})
	{
		# need to jump somewhere... 
		my $nextDoc = $self->nextDocument();
		warn "Told to go to next doc, but nextDocument not set" unless ($nextDoc);
		
		VoiceXML::Client::Util::log_msg("GOING TO $nextDoc") if ($VoiceXML::Client::Debug);
		
	}
	
	return $retVal;
	
}
		
sub nextDocument {
	my $self = shift;
	my $setTo = shift; # optional
	
	if (defined $setTo)
	{
		$self->{'nextdocument'} = $setTo;
	}
	
	return 	$self->{'nextdocument'};
}
	

sub isaVXMLDocument {
	my $self = shift;
	
	return 1;
}

1;