The Perl Toolchain Summit needs more sponsors. If your company depends on Perl, please support this very important event.
# Private package (not to be used outside XML::API)
package XML::API::Element;
use strict;
use warnings;
use Carp qw(croak);
use Scalar::Util qw(weaken refaddr);

our $VERSION = '0.30';

sub new {
    my $proto = shift;
    my $class = ref($proto) || $proto;
    my $self  = {
        attrs    => {},
        contents => [],
        @_
    };

    if ( $self->{comment} ) {
        $self->{comment} =~ s/--/- -/go;
    }

    weaken( $self->{parent} ) if ( exists $self->{parent} );

    bless( $self, $class );
    return $self;
}

sub parent {
    my $self = shift;
    return $self->{parent};
}

sub inline {
    my $self = shift;
    return $self->{inline};
}

sub attrs_as_string {
    my $self = shift;
    my @strings;

    foreach my $key ( sort keys %{ $self->{attrs} } ) {
        my $val = $self->{attrs}->{$key};
        if ( !defined($val) ) {
            warn "Attribute '$key' (element '$self->{element}') is undefined";
            $val = '*undef*';
        }
        push( @strings, $key . '="' . $val . '"' );
    }

    return '' unless (@strings);
    return ' ' . join( ' ', @strings );
}

sub add {
    my $self = shift;
    push( @{ $self->{contents} }, @_ );
}

sub as_string {
    my $self       = shift;
    my $indent     = shift || '';
    my $growindent = shift || '';

    if ( $self->{comment} ) {
        return $indent . '<!-- ' . $self->{comment} . ' -->';
    }

    if ( $self->{cdata} ) {
        return $indent . '<![CDATA[' . $self->{cdata} . ']]>';
    }

    if ( !@{ $self->{contents} } ) {
        return
            $indent . '<'
          . ( $self->{ns} ? $self->{ns} . ':' : '' )
          . $self->{element}
          . $self->attrs_as_string . ' />';
    }

    my $str =
        $indent . '<'
      . ( $self->{ns} ? $self->{ns} . ':' : '' )
      . $self->{element}
      . $self->attrs_as_string . '>';
    my $complex = 0;

    foreach my $c ( @{ $self->{contents} } ) {
        if ( eval { $c->isa(__PACKAGE__) and !$c->inline } ) {
            $complex = 1;
            $str .= "\n" . $c->as_string( $indent . $growindent, $growindent );
        }
        elsif ( eval { $c->isa('XML::API') } ) {    # assume it is complex?
            $str .= "\n"
              . join( "\n",
                map { $_->as_string( $indent . $growindent, $growindent ) }
                  $c->_elements );
        }
        else {
            $str .= $c if ( defined($c) );
        }
    }

    if ($complex) {
        $str .= "\n" . $indent;
    }
    $str .=
      '</' . ( $self->{ns} ? $self->{ns} . ':' : '' ) . $self->{element} . '>';
    return $str;
}

sub fast_string {
    my $self = shift;

    $self->{comment} && return '';
    $self->{cdata}   && return '<![CDATA[' . $self->{cdata} . ']]>';

    return
        '<'
      . ( $self->{ns} ? $self->{ns} . ':' : '' )
      . $self->{element}
      . $self->attrs_as_string . ' />'
      unless @{ $self->{contents} };

    return
        '<'
      . ( $self->{ns} ? $self->{ns} . ':' : '' )
      . $self->{element}
      . $self->attrs_as_string . '>'
      . join(
        '',
        map {
            eval { $_->isa(__PACKAGE__) } ? $_->fast_string
              : (
                eval { $_->isa('XML::API') }
                ? join( '', map { $_->fast_string } $_->_elements )
                : $_
              )
        } @{ $self->{contents} }
      )
      . '</'
      . ( $self->{ns} ? $self->{ns} . ':' : '' )
      . $self->{element} . '>';
}

# Private package (not to be used outside XML::API)
package XML::API::SAXHandler;
use strict;
use warnings;
use base qw(XML::SAX::Base);

sub new {
    my $proto = shift;
    my $class = ref($proto) || $proto;
    my $self  = {
        xmlapi => undef,
        @_,
    };
    bless( $self, $class );
    return $self;
}

sub start_element {
    my $self = shift;
    my $hash = shift;
    if ( $hash->{Name} eq '_xml_api_ignore' ) {
        $self->{xml_api_ignore} = 1;
        return;
    }
    $self->{xml_api_ignore} = 0;

    my $attrs = {};
    foreach my $val ( values %{ $hash->{Attributes} } ) {
        $attrs->{ $val->{Name} } = $val->{Value};
    }

    $self->{xmlapi}->_open( $hash->{Name}, $attrs );
    return;
}

sub characters {
    my $self = shift;
    my $hash = shift;
    $self->{xml_api_ignore} && return;
    $self->{xmlapi}->_add( $hash->{Data} );
    return;
}

sub end_element {
    my $self = shift;
    my $hash = shift;
    if ( $hash->{Name} eq '_xml_api_ignore' ) {
        return;
    }

    $self->{xmlapi}->_close( $hash->{Name} );
    return;
}

# ----------------------------------------------------------------------
# XML::API - Perl extension for creating XML documents
# ----------------------------------------------------------------------
package XML::API;
use strict;
use warnings;
use overload '""' => \&_as_string, 'fallback' => 1;
use Carp qw(carp croak confess);
use Scalar::Util qw(weaken refaddr);
use XML::SAX;

our $VERSION          = '0.30';
our $DEFAULT_ENCODING = 'UTF-8';
our $ENCODING         = undef;
our $Indent           = '  ';
our $AUTOLOAD;

# Not implemented yet:
#  strict   => 0|1                # Optional, defaults to 0
#By default strict checking is performed to make sure that the structure
#of the document matches the Schema. This can be turned off by setting
#'strict' to false (0 or undef).

sub new {
    my $proto = shift;
    my $class = ref($proto) || $proto;

    my $self = {
        doctype  => undef,
        encoding => undef,
        debug    => undef,
        @_,
    };

    #
    # Derived classes
    #
    if ( $class ne __PACKAGE__ ) {
        if ( $self->{doctype} ) {
            confess("Must not specify doctype when instantiating $class");
        }
    }
    elsif ( $self->{doctype} ) {
        $class = $class . '::' . uc( $self->{doctype} );
        if ( !eval "require $class;1;" ) {
            die "Could not load module '$class'";
        }
    }
    delete $self->{doctype};
    bless( $self, $class );

    $self->{encoding} = $self->{encoding} || $ENCODING || $DEFAULT_ENCODING;
    $self->{elements} = [];
    $self->{current}  = undef;
    $self->{string}   = undef;
    $self->{ids}      = {};
    $self->{langs}    = {};

    return $self;
}

sub _root_element {
    return '';
}

sub _root_attrs {
    return {};
}

sub _doctype {
    return '';
}

sub _elements {
    my $self = shift;
    return @{ $self->{elements} };
}

sub _open {
    my $self = shift;
    my $element = shift || croak '_open($element,...)';

    my $namespace = $self->{namespace};

    # reset the output string in case it has been cached
    $self->{string} = undef;

    if ( $element eq $self->_root_element ) {
        $self->{has_root_element} = 1;
    }

    my $attrs = {};
    my @content;

    my $total = scalar(@_) - 1;
    my $next;

    foreach my $i ( 0 .. $total ) {
        if ($next) {
            $next = undef;
            next;
        }

        my $arg = $_[$i];
        if ( ref($arg) eq 'HASH' ) {
            while ( my ( $key, $val ) = each %$arg ) {
                $attrs->{$key} = _escapeXML($val);
                if ( !defined($val) ) {
                    carp "attribute '$key' undefined (element '$element')";
                    $attrs->{$key} = '';
                }
            }
        }
        elsif ( defined($arg) and $arg =~ m/^-[^0-9\.]+/o ) {
            $arg =~ s/^-//o;
            $attrs->{$arg} = _escapeXML( $_[ ++$i ] );
            if ( !defined( $attrs->{$arg} ) ) {
                carp "attribute '$arg' undefined (element '$element') ";
                $attrs->{$arg} = '';
            }
            $next = 1;
            next;
        }
        else {
            push( @content, $arg );
        }
    }

    #
    # Start with the default root element attributes and add those
    # given if this is the root element
    #
    if ( $element eq $self->_root_element ) {
        my $rootattrs = $self->_root_attrs;
        while ( my ( $key, $val ) = each %$attrs ) {
            $rootattrs->{$key} = $val;
        }
        $attrs = $rootattrs;
    }

    my ( $file, $line ) = (caller)[ 1, 2 ] if ( $self->{debug} );

    if ( $self->{langnext} ) {
        $attrs->{'xml:lang'} = delete $self->{langnext};
    }
    if ( $self->{dirnext} ) {
        $attrs->{'dir'} = delete $self->{dirnext};
    }

    my $e;
    if ( $self->{current} ) {
        $e = XML::API::Element->new(
            element => $element,
            attrs   => $attrs,
            ns      => $namespace,
            parent  => $self->{current},
        );
        $self->_add($e);
    }
    else {
        $e = XML::API::Element->new(
            element => $element,
            attrs   => $attrs,
            ns      => $namespace,
        );
        push( @{ $self->{elements} }, $e );
    }

    $self->{current} = $e;
    if ( $self->{_raw} ) {
        $self->_raw(@content);
    }
    else {
        $self->_add(@content);
    }

    $self->_comment("DEBUG: '$element' (open) at $file:$line")
      if ( $self->{debug} );

    return $e;
}

sub _add {
    my $self = shift;
    $self->{string} = undef;
    if ( !$self->{current} ) {
        croak 'Cannot use _add with no current element';
    }

    foreach my $item (@_) {
        carp "undefined input" unless ( defined($item) );
        if ( eval { $item->isa(__PACKAGE__) } ) {
            if ( refaddr($item) == refaddr($self) ) {
                croak 'Cannot _add object to itself';
            }
            if ( !$self->{current} ) {
                push( @{ $self->{elements} }, $item );
            }
            else {
                $self->{current}->add($item);
            }
            bless( $item, ref($self) );
            $item->{parent} = $self;
            weaken( $item->{parent} );

            foreach my $lang ( keys %{ $item->{langs} } ) {
                $self->{langs}->{$lang} = 1;
            }
        }
        else {
            if ( eval { $item->isa('XML::API::Element') } ) {
                $self->{current}->add($item);
            }
            elsif ( eval { $item->isa('XML::API::Cache') } ) {
                foreach my $lang ( $item->langs ) {
                    $self->{langs}->{$lang} = 1;
                }
                $self->{current}->add($item);
            }
            else {
                $self->{current}->add( _escapeXML($item) );
            }
        }
    }
}

sub _raw {
    my $self = shift;
    $self->{string} = undef;
    foreach my $item (@_) {
        carp "undefined input" unless ( defined($item) );
        if ( ref($item) and $item->isa(__PACKAGE__) ) {
            croak 'Cannot add XML::API objects as raw';
        }
        if ( $self->{current} ) {
            $self->{current}->add($item);
        }
        else {
            push( @{ $self->{elements} }, $item );
        }
    }
}

sub _close {
    my $self = shift;
    my $element = shift || croak '_close($element)';

    my ( $file, $line ) = (caller)[ 1, 2 ] if ( $self->{debug} );

    if ( !$self->{current} ) {
        carp 'attempt to close non-existent element "' . $element . '"';
        return;
    }

    if ( $element eq $self->{current}->{element} ) {
        if ( $self->{current}->parent ) {
            $self->{current} = $self->{current}->parent;
            $self->_comment("DEBUG: '$element' close at $file:$line")
              if ( $self->{debug} );
        }
        else {
            $self->{current} = undef;
        }
    }
    else {
        carp 'attempted to close element "'
          . $element
          . '" when current '
          . 'element is "'
          . $self->{current}->{element} . '"';
    }
    return;
}

sub _element {
    my $self    = shift;
    my $element = shift || croak '_element($element)';
    my $e       = $self->_open( $element, @_ );
    $self->_close($element);
    return $e;
}

#
# The implementation for element, element_open and element_close
#

sub AUTOLOAD {
    my $self    = shift;
    my $element = $AUTOLOAD;

    my ( $open, $close ) = ( 0, 0 );

    if ( $element =~ s/.*::(.+)_open$/$1/o ) {
        my $old_ns = $self->{namespace};

        if ( $element =~ s/(.+)__(.+)/$2/o ) {
            $self->{namespace} = $1;
        }

        my $e = $self->_open( $element, @_ );
        $self->{namespace} = $old_ns;
        return $e;
    }
    elsif ( $element =~ s/.*::(.+)_close$/$1/o ) {
        $element =~ s/(.+)__(.+)/$2/o;
        return $self->_close($element);
    }
    elsif ( $element =~ s/.*::(.+)_raw$/$1/o ) {
        $element =~ s/(.+)__(.+)/$2/o;
        $self->{_raw} = 1;
        $self->_open( $element, @_ );
        $self->{_raw} = 0;
        return $self->_close($element);
    }

    $element =~ s/.*:://o;
    croak 'element not defined' unless ($element);

    if ( $element =~ /^_/o ) {
        croak 'Undefined subroutine &' . ref($self) . "::$element called";
    }

    my $old_ns = $self->{namespace};

    if ( $element =~ s/(.+)__(.+)/$2/o ) {
        $self->{namespace} = $1;
    }
    my $e = $self->_open( $element, @_ );
    $self->{namespace} = $old_ns;
    $self->_close($element);
    return $e;
}

sub _ast {
    my $self = shift;

    foreach my $i ( 1 .. int( scalar(@_) / 2 ) ) {
        my ( $e, $val ) = splice( @_, 0, 2 );

        if ( !ref($val) ) {
            $self->_element( $e, $val );
            next;
        }

        my $attr     = {};
        my @contents = ();

        if ( ref($val) and ref($val) eq 'ARRAY' ) {
            my @val = @$val;

            foreach my $i ( 1 .. int( scalar(@val) / 2 ) ) {
                my ( $arg, $arg2 ) = splice( @val, 0, 2 );

                if ( $arg =~ s/^-(.+)/$1/o ) {
                    $attr->{$arg} = $arg2;
                }
                elsif ( ref($arg2) and ref($arg2) eq 'ARRAY' ) {
                    push( @contents, [ $arg, $arg2 ] );
                }
                else {
                    push( @contents, { $arg => $arg2 } );
                }
            }

            push( @contents, @val ) if (@val);
        }
        else {
            push( @contents, $val );
        }
        $self->_open( $e, $attr );

        foreach my $c (@contents) {
            if ( ref($c) and ref($c) eq 'ARRAY' ) {
                $self->_ast(@$c);
            }
            elsif ( ref($c) and ref($c) eq 'HASH' ) {
                my ( $k, $v ) = each %$c;
                $self->_open($k);
                $self->_add($v);
                $self->_close($k);
            }
            else {
                $self->_add($c);
            }
        }

        $self->_close($e);
    }

    return;
}

sub _comment {
    my $self = shift;

    # FIXME: should escape?
    $self->_raw( XML::API::Element->new( comment => join( '', @_ ) ) );
    return;
}

sub _cdata {
    my $self = shift;
    $self->_raw( XML::API::Element->new( cdata => join( '', @_ ) ) );
    return;
}

sub _css {
    my $self    = shift;
    my $content = shift;
    if ( $content =~ /\n/s ) {
        $self->_raw( '/*<![CDATA[*/' . "\n" . $content . '/*]]>*/' );
    }
    else {
        $self->_raw( '/*<![CDATA[*/ ' . $content . ' /*]]>*/' );
    }
    return;
}

sub _javascript {
    my $self = shift;
    $self->script_open( -type => 'text/javascript' );
    $self->_raw( '// -------- JavaScript Begin -------- <![CDATA[' . "\n" );
    $self->_raw(@_);
    $self->_raw('// --------- JavaScript End --------- ]]>');
    $self->script_close;
    return;
}

sub _parse {
    my $self    = shift;
    my $current = $self->{current};

    foreach (@_) {
        next unless ( defined($_) and $_ ne '' );
        local $XML::SAX::ParserPackage = 'XML::LibXML::SAX';
        my $parser =
          XML::SAX::ParserFactory->parser(
            Handler => XML::API::SAXHandler->new( xmlapi => $self ), );

        # remove leading and trailing space, otherwise SAX barfs at us.
        ( my $t = $_ ) =~ s/(^\s+)|(\s+$)//go;

        # escape '&' as well
        $t =~ s/\&(\w+\;)/__AMP__$1/go;

        # escape '&' in urls
        $t =~ s/\&(\w+=)/__AMP__amp;$1/go;
        $parser->parse_string(
            '<_xml_api_ignore>' . $t . '</_xml_api_ignore>' );
    }

    # always make sure that we finish where we started
    $self->{current} = $current;
}

sub _parse_chunk {
    my $self    = shift;
    my $current = $self->{current};

    foreach (@_) {
        next unless ( defined($_) and $_ ne '' );
        local $XML::SAX::ParserPackage = 'XML::LibXML::SAX';
        my $parser =
          XML::SAX::ParserFactory->parser(
            Handler => XML::API::SAXHandler->new( xmlapi => $self ), );

        # remove leading and trailing space, otherwise SAX barfs at us.
        ( my $t = $_ ) =~ s/(^\s+)|(\s+$)//go;

        # escape '&' as well
        $t =~ s/\&(\w+\;)/__AMP__$1/go;

        # escape '&' in urls
        $t =~ s/\&(\w+=)/__AMP__amp;$1/go;

        # escape '&' on their own.
        $t =~ s/(\W)\&(\W)/$1__AMP__amp;$2/go;
        $parser->parse_chunk($t);
    }

    # always make sure that we finish where we started
    $self->{current} = $current;
}

sub _attrs {
    my $self = shift;

    if (@_) {
        my $attrs = shift;
        if ( !$attrs or ref($attrs) ne 'HASH' ) {
            croak 'usage: _attrs($hashref)';
        }
        $self->{current}->{attrs} = $attrs;
    }
    return $self->{current}->{attrs};
}

sub _encoding {
    my $self = shift;
    if (@_) {
        $self->{encoding} = shift;
    }
    return $self->{encoding};
}

sub _set_lang {
    my $self = shift;
    my $lang = shift || croak 'usage: set_lang($lang)';
    my $dir  = shift;

    if ( $self->{has_root_element} and !$self->_lang ) {
        $self->{elements}->[0]->{attrs}->{'xml:lang'} = $lang;
        if ($dir) {
            $self->{elements}->[0]->{attrs}->{'dir'} = $dir;
        }
    }
    else {
        $self->{langnext} = $lang;
        $self->{dirnext} = $dir if ($dir);
    }
    $self->{langs}->{$lang} = 1;

    return;
}

sub _lang {
    my $self = shift;

    if ( $self->{current} ) {
        return $self->{current}->{attrs}->{'xml:lang'}
          if ( exists( $self->{current}->{attrs}->{'xml:lang'} ) );

        my $e = $self->{current};
        while ( $e = $e->{parent} ) {
            return $e->{attrs}->{'xml:lang'}
              if ( exists( $e->{attrs}->{'xml:lang'} ) );
        }
    }
    return $self->{langnext} if ( $self->{langnext} );
    return $self->{parent}->_lang if ( $self->{parent} );
    return;
}

sub _langs {
    my $self = shift;
    return keys %{ $self->{langs} };
}

sub _dir {
    my $self = shift;

    if ( $self->{current} ) {
        return $self->{current}->{attrs}->{'dir'}
          if ( exists( $self->{current}->{attrs}->{'dir'} ) );

        my $e = $self->{current};
        while ( $e = $e->{parent} ) {
            return $e->{attrs}->{'dir'}
              if ( exists( $e->{attrs}->{'dir'} ) );
        }
    }
    return $self->{dirnext} if ( $self->{dirnext} );
    return $self->{parent}->_dir if ( $self->{parent} );
    return;
}

sub _ns {
    my $self = shift;
    if (@_) {
        $self->{namespace} = shift;
    }
    return $self->{namespace};
}

sub _debug {
    my $self = shift;
    if (@_) {
        $self->{debug} = shift;
    }
    return $self->{debug};
}

sub _current {
    my $self = shift;
    return $self->{current};
}

sub _set_id {
    my $self = shift;
    my $id   = shift;

    if ( !defined($id) or $id eq '' ) {
        carp '_set_id called without a valid id';
        return;
    }
    if ( defined( $self->{ids}->{$id} ) ) {
        carp 'id ' . $id . ' already defined - overwriting';
    }
    $self->{ids}->{$id} = $self->{current};
}

sub _goto {
    my $self = shift;

    if (@_) {
        my $id = shift;
        if ( !defined $id ) {
            $self->{current} = undef;
            return;
        }
        if ( $id->isa('XML::API::Element') ) {
            $self->{current} = $id;
        }
        elsif ( defined( $self->{ids}->{$id} ) ) {
            $self->{current} = $self->{ids}->{$id};
        }
        else {
            carp "Nonexistent ID given to _goto: '$id'. ",
              '(Known IDs: ', join( ',', keys( %{ $self->{ids} } ) ), ')';
            $self->{current} = undef;
        }
    }
    return $self->{current};
}

sub _as_string {
    my $self = shift;
    my $file = shift;

    my $string = '';

    if ( ref($self) eq __PACKAGE__ or $self->{has_root_element} ) {
        $string = qq{<?xml version="1.0" encoding="$self->{encoding}" ?>\n};
        $string .= $self->_doctype . "\n" if ( $self->_doctype );
    }

    $string .= join(
        "\n",
        map {
            $_->isa(__PACKAGE__)
              ? join( "\n", map { $_->as_string } $_->_elements )
              : $_->as_string( '', '  ' )
        } @{ $self->{elements} }
    );

    if ($file) {
        open( FH, '>' . $file ) || die "open '$file': $!";
        if ( $self->_encoding eq 'UTF-8' ) {
            binmode( FH, ':utf8' );
        }
        print FH $string;
        close(FH);
        return;
    }
    return $string;
}

sub _fast_string {
    my $self = shift;
    my $file = shift;

    my $string = '';

    if ( ref($self) eq __PACKAGE__ or $self->{has_root_element} ) {
        $string = qq{<?xml version="1.0" encoding="$self->{encoding}" ?>};
        $string .= $self->_doctype if ( $self->_doctype );
    }

    $string .=
      join( "\n", map { $_->fast_string( '', '  ' ) } @{ $self->{elements} } );

    if ($file) {
        open( FH, '>' . $file ) || die "open '$file': $!";
        if ( $self->_encoding eq 'UTF-8' ) {
            binmode( FH, ':utf8' );
        }
        print FH $string;
        close(FH);
        return;
    }
    return $string;
}

sub _escapeXML {
    my $data = $_[0];
    return '' unless ( defined($data) );
    if ( $data =~ /[\&\<\>\"(__AMP__)\']/o ) {
        $data =~ s/\&(?!\w+\;)/\&amp\;/go;
        $data =~ s/\</\&lt\;/go;
        $data =~ s/\>/\&gt\;/go;
        $data =~ s/\"/\&quot\;/go;
        $data =~ s/__AMP__/\&/go;
        $data =~ s/\'/\&apos\;/go;
    }
    return $data;
}

#
# We must specify the DESTROY function explicitly otherwise our AUTOLOAD
# function gets called at object death.
#
DESTROY { }

1;
__END__

=head1 NAME

XML::API - Perl extension for writing XML

=head1 VERSION

0.30 (2016-04-11)

=head1 SYNOPSIS

  use XML::API;
  my $x = XML::API->new(doctype => 'xhtml');
  $x->_comment('My --First-- XML::API document');
  
  $x->html_open();
  $x->head_open();
  $x->title('Test Page');
  $x->head_close();
  $x->body_open();
  $x->div_open(-id => 'content');
  $x->p(-class => 'test', 'Some <<odd>> input');
  $x->ns__p(-class => 'test', '& some other &stuff;');
  $x->div_close();
  $x->body_close();
  $x->html_close();

  print $x;

Will produce this nice output:

  <?xml version="1.0" encoding="UTF-8" ?>
  <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict...>
  <!-- My - -First- - XML::API document -->
  <html xmlns="http://www.w3.org/1999/xhtml">
    <head>
      <title>Test Page</title>
    </head>
    <body>
      <div id="content">
        <p class="test">Some &lt;&lt;odd&gt;&gt; input</p>
        <ns:p class="test">&amp; some other &stuff;</ns:p>
      </div>
    </body>
  </html>

=head1 DESCRIPTION

B<XML::API> is a class for creating XML documents using object method
calls. This class is meant for generating XML programatically and not
for reading or parsing it.

A document author calls the desired methods (representing elements) to
create an XML tree in memory which can then be rendered or saved as
desired. The advantage of having the in-memory tree is that you can be
very flexible about when different parts of the document are created
and the final output is always nicely rendered.

=head1 TUTORIAL

The first step is to create an object. The 'doctype' attribute is
mandatory. Known values (ie - distributed with XML::API) are 'xhtml'
and 'rss'. The encoding is not mandatory and will default to 'UTF-8'.

  use XML::API;
  my $x = XML::API->new(doctype => 'xhtml', encoding => 'UTF-8');

$x is the only object we need for our entire XHTML document. It starts
out empty so we want to open up the html element:

  $x->html_open;

Because we have called a *_open() function the 'current' or
'containing' element is now 'html'. All further elements will be added
inside the 'html' element. So lets add head and title elements and the
title content ('Document Title') to our object:

  $x->head_open;
  $x->title('Document Title');

The 'title()' method on its own (ie not 'title_open()') indicates that
we are specifiying the entire title element. Further method calls will
continue to place elements inside the 'head' element until we specifiy
we want to move on by calling the _close method:

  $x->head_close();

This sets the current element back to 'html'.

So, basic elements seem relatively easy. How do we create elements with
attributes? When either the element() or element_open() methods are
called with a hashref argument the keys and values of the hashref
become the attributes:

  $x->body_open({id => 'bodyid'}, 'Content', 'more content');

or if you want, you can also use CGI-style attributes which I prefer
because it takes less typing:

  $x->body_open(-id => 'bodyid', 'Content', 'more content');

By the way, both the element() and element_open() methods take
arbitrary numbers of content arguments as shown above. However if you
don't want to specify the content of the element at the time you open
it up you can use the _add() utility method later on:

  $x->div_open();
  $x->_add('Content added after the _open');

The final thing is to close out the elements and render the docment.

  $x->div_close();
  $x->body_close();
  print $x->_as_string();

Because we are not adding any more elements or content it is not
strictly necessary to close out all elements, but consider it good
practice.

You can add XML::API object to other objects, which lets you create for
instance the head and body parts separately, and just bring them all
together just before printing:

  my $h = XML::API::XHTML->new();
  $h->head_open
  ...
  my $x = XML::API::XHTML->new();
  $x->html_open;
  $x->_add($h);
  $x->html_close;
  print $x;

Note that it is also possible to call the XML::API::<doctype> class
directly.

=head1 CLASS SUBROUTINES

=head2 new

Create a new XML::API based object. The object is initialized as empty
(ie contains no elements). Takes the following optional arguments:

  doctype  => '(xhtml|rss|WIX2)'
  encoding => 'xxx'
  debug    => 1|0

If a valid (ie known to XML::API) doctype is given then an object of
class XML::API::DOCTYPE will be returned instead. This method will die
if doctype is unknown. You can also call XML::API::DOCTYPE->new()
directly.

For the effects of the encoding and debug parameters see the
documentation for '_encoding' and '_debug' below.

=head1 CONTENT

=head2 $x->element_open(-attribute => $value, {attr2 => 'val2'}, $content)

Add a new element to the 'current' element, and set the current element
to be the element just created. Returns a reference (private data type)
to the new element which can be used in the _goto function below.

Ie given that $x currently represents:

  <html>  <---- 'current' element
          <---- future elements/content goes here
  </html>

then $x->head_open(-attribute => $value) means the tree is now:

  <html>
    <head attribute="$value">  <---- 'current' element
                               <---- future elements/content goes here
    </head>
  </html>


=head2 $x->_open('element', -attribute => $value, {attr2 => 'val2'}, $content)

The generic/underlying implementation of $x->element_open. Useful if
your element names are not suitable as Perl method calls, or are
otherwise funny (eg starting with '_').

=head2 $x->_add($content)

Add $content to the 'current' element. If there is no current element
then this method will croak.

If $content is a scalar (ie plain text or numbers) then the characters
'<&">' will be XML-escaped.  If $content is another XML::API object the
elements of that object will be added to content tree.

This method will also croak if you attempt to add $x to itself or if $x
is an empty XML::API object.

=head2 $x->_raw($content)

Adds unescaped content to the 'current' element. You need to be careful
of characters that mean something in XML such as '<','&' and '>'. This
method will die if $content is an XML::API derivative or if $x does not
have a current element.

=head2 $x->element_close( )

This does not actually modify the tree but simply tells the object that
future elements will be added to the parent of the current element. Ie
given that $x currently represents:

  <div>
    <p>  <---- 'current' element
      $content
           <---- future elements/content goes here
    </p>
  </div>

then $x->p_close() means the tree is now:

  <div>    <---- 'current' element
    <p>
      $content
    </p>
           <---- future elements go here
  </div>

If you try to call a _close() method that doesn't match the current
element a warning will be issued and the call will fail.

=head2 $x->_close('element')

The generic/underlying implementation of $x->element_close. Useful if
your element names are not suitable as Perl method calls, or are
otherwise funny (eg starting with '_').

=head2 $x->element(-attribute => $value, {attr2 => 'val2'}, $content)

Add a new element to the 'current' element but keep the 'current'
element the same. Returns a reference (private data type) to the new
element which can be used in the _goto function below.

This is effectively the same as the following:

    $x->element_open(-attribute => $value, -attr2=>'val2');
    $x->_add($content);
    $x->element_close;

If $content is not given (or never added with the _add method) for an
element then it will be rendered as empty. Ie, $x->br() produces:

    <br />

=head2 $x->_element('element',...)

The generic implementation of $x->element. Useful if your element names
are not suitable as Perl method calls, or are otherwise funny (eg
starting with '_').

=head2 $x->element_raw('raw content',...)

Adds unescaped content inside an element named 'element'. This is a
shortcut for the case where you find yourself doing the following:

    $x->element_open();
    $x->_raw($content);
    $x->element_close();

=head2 $x->ns__element_open(...)

Same as $x->element_open but prefixed with an XML namespace. Equivalent
to the following.

  $x->_ns('ns');
  $x->element_open(...);
  ...
  $x->element_close;
  $x->_ns(undef);

=head2 $x->ns__element(...)

Same as $x->element but prefixed with an XML namespace. Equivalent to
the following.

  $x->_ns('ns');
  $x->element(...);
  $x->_ns(undef);

=head2 $x->_comment($comment)

Add an XML comment to $x. Is almost the same as this:

    $x->_raw("\n<!--");
    $x->_raw($content);
    $x->_raw('-->')

Except that indentation is correct. Any occurences of '--' in $content
will be replaced with '- -'.

=head2 $x->_cdata($content)

A shortcut for $x->_raw("\n<![CDATA[", $content, " ]]>");

=head2 $x->_css($content )

Adds $content inside a pair of CDATA tags which are encapsulated inside
CSS comments. Similar to:

 $x->_raw('/*<![CDATA[*/ '. $content .' /*]]>*/');

=head2 $x->_javascript($script )

A shortcut for adding $script inside a pair of <script
type="text/javascript"> elements and a _CDATA tag.

=head2 $x->_parse(@content)

Adds content to the current element, but will parse it for xml elements
and add them as method calls. Regardless of $content (missing end tags
etc) the current element will remain the same. Relies on XML::SAX to do
the parsing using the "parse_string" method. In this case XML::SAX
requires that the content is a complete xml document.

=head2 $x->_parse_chunk(@content)

Adds content to the current element, but will parse it for xml elements
and add them as method calls. Regardless of $content (missing end tags
etc) the current element will remain the same. Relies on XML::SAX to do
the parsing, but using the "parse_chunk" method. This method is
suitable for parsing xml fragments which are not necessarily complete.

=head2 $x->_ast(@content)

Sometimes you may want to just build some kind of abstract syntax tree
structure and just feed it to XML::API without having to make all the
method calls yourself. This method lets you do just that.

The following input:

  p => [
      label => 'Body',
      textarea => [
          -rows  => 10,
          -cols  => 50,
          -name  => 'body',
          'the body',
      ],
  ],

results in the following xml:

  <p>
    <label>Body</label>
    <textarea cols="50" name="body" rows="10">the body</textarea>
  </p>


=head2 $x->_attrs( )

Allows you to get/set the attributes of the current element. Accepts
and returns and hashref.

=head1 META DATA

=head2 $x->_encoding($value)

Set the encoding definition produced in the xml declaration. Returns
the current value if called without an argument. This is an alternative
to defining the encoding in the call to 'new'.

The XML encoding definition for objects is determined by the following,
in this order:

  * the last call to _encoding
  * the encoding parameter given at object creation
  * $XML::API::ENCODING, set by your script before calling new
  * UTF-8, the package default

If you _add one object to another with different encodings the
top-level object's definition will be used.



=head2 $x->_set_lang($lang, [$dir])

Add an 'xml:lang' attribute, and an optional text direction $dir to the
next element to be created. In terms of output created this means that:

  $x->_set_lang('de', 'ltr');
  $x->p('Was sagst du?');

is equivalent to:

  $x->p(-xml:lang => 'de', -dir => 'ltr', 'Was sagst du?');

with the added difference that _set_lang keeps track of each call and
the list of languages set can be retrieved using the _langs method
below.

The first time _set_lang is called the xml:lang attribute will be added
to the root element instead of the next one, unless $x is a generic XML
document. Without a XML::API::<class> object we don't know if we have
the root element or not.

=head2 $x->_lang

Returns the language of the B<current element>. Note that this is not
always the same as the last value given to _set_lang, but depends on
what the current element is. Returns 'undef' if the document has no
'xml:lang' at all.

=head2 $x->_langs

Returns the list of the languages that have been specified by
_set_lang.

=head2 $x->_dir

Returns the text direction of the B<current element>. Note that this is
not always the same as the last value given to _set_lang, but depends
on what the current element is. Returns 'undef' if the document has no
direction specified.

=head2 $x->_ns($namespace)

Sets the XML namespace for future elements added with element() or
element_open(). Use 'undef' for no namespace. If you just want a
single-shot namespace you can also call element methods with a
namespace postfixed by '__' (two underscores).

=head2 $x->_debug(1|0)

Turn on|off debugging from this point onwards. Debugging appears as xml
comments in the rendered XML output.


=head2 $x->_current( )

Returns a reference (private data type) to the current element. Can be
used in the _goto method to get back to the current element in the
future.


=head2 $x->_set_id($id)

Set an identifier for the current element. You can then use the value
of $id in the _goto() method.



=head2 $x->_goto($id)

Change the 'current' element. $id is a value which has been previously
used in the _set_id() method, or the return value of a _current() call.

This is useful if you create the basic structure of your document, but
then later want to go back and modify it or fill in the details.



=head1 OUTPUT

=head2 $x->_as_string($file)

Returns the xml-rendered version of the object. If $x has the root
element for the doctype, or if $x is a pure XML::API object then the
string is prefixed by the XML declaration (with the encoding as defined
in the '_encoding' method documentation):

  <?xml version="1.0" encoding="UTF-8" ?>

The optional $file argument will cause the string to be written out to
disk instead of being returned.

=head2 $x->_fast_string($file)

Returns the rendered version of the XML document without newlines or
indentation.

The optional $file argument will cause the string to be written out to
disk instead of being returned.

=head1 OVERLOADING

See the source code of XML::API::XHTML for how to create a new doctype.

These are methods which may return interesting values if the
XML::API::<class> module has overloaded them.

=head2 _doctype

Returns the XML DOCTYPE declaration

=head2 _root_element

Returns the root element

=head2 _root_attrs

Returns a hashref containing default key/value attributes for the root
element

=head2 _content_type

Returns a string suitable for including in a HTTP 'Content-Type'
header.


=head1 COMPATABILITY

Since version 0.10 a call to new() does not automatically add the root
element to the object. If it did so you wouldn't be able to add one
object to another.

Version 0.13 made the doctype parameter to new() optional, so that
generic (ie no DOCTYPE declaration) XML documents can be created.

Version 0.15 removed the pointless _print method.

=head1 SEE ALSO

L<XML::Generator> and L<XML::Writer> are other xml producing modules.

If you are thinking of using B<XML::API> in a CGI program check out
L<NCGI>.

=head1 AUTHOR

Mark Lawrence E<lt>nomad@null.netE<gt>

A small request: if you use this module I would appreciate hearing
about it.

=head1 COPYRIGHT

Copyright (C) 2004-2016 Mark Lawrence <nomad@null.net>

=head1 LICENSE

This program 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 2 of the License, or (at your
option) any later version.

=cut

# vim: set tabstop=4 expandtab: