NAME
     HTML::Transmorgify - HTML transformation compiler

SYNOPSIS
     use HTML::Transmorgify;

     my $magic = HTML::Transmorgify->new(xml_quoting => 1);

     $magic->mixin('HTML::Transmorgify::Metatags');
     $magic->mixin('HTML::Transmorgify::FormDefaults');

     my $output = $magic->process($input_text, { %options }, %variables);
     my %output_by_type = $magic->process($input_text, { %options }, %variables);

DESCRIPTION
    HTML::Transmorgify is an HTML compiler framework. It transforms HTML
    into a arrays of static text and CODE callbacks. The arrays can be
    turned into customized HTML very quickly. The compilation process is
    quick and the runtime process is even quicker. It is designed for sites
    that will dynamically generate all their content.

    By itself, HTML::Transmorgify doesn't do anything useful: it is just a
    framework. Most users will want to use one or more of the addon modules
    for HTML::Transmorgify:

    HTML::Transmorgify::Metatags
        Provides a template language processor with macros, control flow,
        and basic functions.

    HTML::Transmorgify::FormChecksum
        Adds cryptographic checksums to forms so that users cannot add
        additional elements, change or remove hidden values, or set
        dropdowns or radio buttons to values that were not provided.

    HTML::Transmorgify::FormDefault
        Overrides the default values in the form definition so that the
        end-user's prior input is remembered when re-displaying the form.

    HTML::Transmorgify::Crumbs
        Adds cookie crumbs to URLs to and forms so to protect against
        cross-site scripting attacks that direct a logged-in user back to a
        page that will do something naughty.

    HTML::Transmorgify::Images
        Changes multiple <img> tags into <div> tags with background
        attributes that point into shared composite image. This decreases
        page load time because many fewer items need to be fetched.

        It also adds "height" and "width" attributes to images that don't
        have them yet.

        It notices roll-over images and adds javascript to display them.

        As of January 2011, this addon is not yet complete.

    Additional modules can be added to the framework.

USING HTML::Transmorgify
    By itself, HTML::Transmorgify doesn't do anything useful. You need to
    blend in one of the addon modules or write your own.

    You invoke HTML::Transmorgify with:

     my $output = $magic->process($input_text, { %options }, %variables);

    or

     my %output_by_type = $magic->process($input_text, { %options }, %variables);

    In array context, the results from processing templates are returned as
    a hash, mapping the type of result to the result string. Two result
    types are always present: "text" and "script". Other result types may be
    added by add-on modules.

    Currently, all of the options in "{ %options }" are made available to
    addon modules via the $invocation_options variable. In additon the
    following items are singled out and have defined meanings:

    query_param    A reference to a CGI.pm-style query parameters hash.

    input_file     The filename from which the $input_text came from.

    input_line     A line-number offset from the beginning of the input_file
                   so that error messages can refer to the line number of
                   the error.

    The %variables are used by HTML::Transmorgify::Metatags as pre-defined
    macro values. They can be strings, hashes, arrays, or objects. If
    they're objects, they should inherit from
    HTML::Transmorgify::ObjectGlue.

WRITING ADDON MODULES
    HTML::Transmorgify compiles HTML. It compiles HTML into an array of
    callbacks and literals. It then executes the compiled array. Compilation
    output from an input string is cached for reuse. Files are compiled
    separately.

    When you write an addon module, you register to be called back for
    particular HTML tags. When you are called back, you need to generate
    strings to be appended to the the final document or callbacks to be
    called at runtime.

    The runtime program is an array: @$HTML::Transmorgify::rbuf. There is a
    little function for appending to that array: "rbuf()".

    Addon modules generally start the same:

     package HTML::Transmorgify::MODULE_NAME;

     use HTML::Transmorgify qw(continue_compile capture_compile run queue_intercept rbuf postbuf);

     our @ISA = qw(HTML::Transmorgify Exporter);

     sub add_tags
     {
            my ($self, $tobj) = @_;
            $self->intercept_exclusive($tobj, __PACKAGE__, A_NUMBER, %EXCLUSIVE_TAGS)
            $self->intercept_shared($tobj, __PACKAGE__, A_NUMBER, %SHARED_TAGS);
     }

    Where "A_NUMBER" is a priority number for choosing which callbacks get
    called first (lower is earlier); %EXCLUSIVE_TAGS maps HTML tags to
    callbacks that compile that HTML tag; and %SHARED_TAGS maps HTML tags to
    callbacks that participate in compiling that HTML tag.

    Generally you want an exclusive tag when the tag isn't actually part of
    HTML like when you are creating a template language like what is done in
    HTML::Transmorgify::Metatags.

    For situations where you are making an adjustment to regular HTML tags,
    you usually want a shared tag. For example,
    HTML::Transmorgify::FormChecksum and HTML::Transmorgify::FormDefault
    both modify forms and attach callbacks to the <input> tags.

    Shared tags must be handled carefully so that the multiple modules that
    are acting on a tag do not get in each other's way. Not everything can
    be handled by with a shared tag For example, the <foreach> tag defined
    in HTML::Transmorgify::Metatags is an exclusive tag.

    All callbacks are invoked with the following arguments:

    $attr   A HTML::Transmorgify::Attributes object representing the a tag
            and its attributes.

    $closed Closed if this is an end-tag or self-contained tag like <hr />.

  Shared Tags
    The return value from a shared tag callback is ignored unless the return
    value is a CODE ref. The callbacks are called at compile time. Any CODE
    refs returned by the callback will be invoked at runtime BEFORE the tag
    is interpolated into the results.

    Shared tags are always interpolated into the results.

    The usual form of a shared tag is something like:

     $SHARED_TAGS{img} = \&img_tag;
     sub img_tag {
        my ($attr, $closed) = @_;
        return 1 unless $attr->raw('alt');
        $attr->eval_at_runtime(1);
        rbuf(sub {
            my $text = $attr->get('src');
            $text =~ s{.*/}{};
            $text =~ s/([a-z])([A-Z])/$1 $2/g;
            $text =~ s/_/ /g;
            $attr->set(alt => $text);
        });
        return 1;
     }

    This example transformation will try to turn the image file name into an
    <alt> tag.

    It uses "rbuf()" to add a callback that will run just before the <img>
    tag is added to the output stream.

    It calls

        $attr->eval_at_runtime(1);

    to make sure that the <img> tag is evaluated at runtime rather than
    added as a literal string at compile time.

    It returns a value of 1 to indicate that the <img> tag should be
    included in the output stream.

  Exclusive Tags
    The return value from an exclusive tag indicates whether the tag should
    be included in the resulting text or not. A true value will cause the
    tag to be inclucd; a false value will not.

  APIs for Callback Writers
    rbuf()
        The tiny rbuf function pushes its arguments on the
        "HTML::Transmorgify::rbuf" array. This array represents the output
        from compiling the HTML. It can conntain the following elements:

        strings        Plain scalars will be interpolated into the final
                       output unchanged.

        CODE refs      Code references will be called. For them to add to
                       the final output, they need to append to the strings
                       in @$HTML::Transmorgify::result. The first of these,
                       $$HTML::Transmorgify::result[0] is used for regular
                       results. If additional types of output are needed,
                       the array indexes should be allocated by calling
                       "allocate_result_type".

    postbuf()
        Pushes its arguments onto @HTML::Transmorgify::post_intercept_push.
        This is used during shared tag callback processing. The contents of
        @HTML::Transmorgify::post_intercept_push is pushed onto
        "HTML::Transmorgify::rbuf" after the rest of the shared tag callback
        processing is done.

    allocate_result_type()
        Alocates another "key" for the results array. The predefined keys
        are: "text" and "script".

        Keys can be looked up in %HTML::Transmorgify::result_index.

    boolean()
        The boolean function returns false if it's argument is: "false",
        "no", "off", 0, or undefined. Otherwise it returns true.

    run($buf, $results)
        The run function "executes" the compiled HTML in @$buf. Literal
        strings will be added to $$results[0], code references will be
        invoked.

        If $results is not specified, it will default to
        $HTML::Transmorgify::results.

    eat_cr()
        *Exclusive tag callbacks only.* Advance the input parsing position
        "pos($$HTML::Transmorgify::textref)" past a newline.

    $rbuf = compile($cacheline, $textref)
        Compiles $$textref and returns the results. Results are are cached
        so compiling the same string ($$textref) in the same context will
        result in a cached result. Do not modify the returned array (@$rbuf)
        as since it can be handed out again from another call to compile.

        The $cacheline argument specifies the caching context for the input
        string. For compiling things in the normal context where all
        currently active transformations can apply, use the context
        $HTML::Transmorgify::modules.

    ($rbuf, $deferred) = capture_compile($tag, $starting_attr, $opts, %tags)
        *Exclusive tag callbacks only.* Continues to compile the current
        $HTML::Transmorgify::textref (as passed to compile()) until a
        closing "</$tag>" is reached. For better error messages, the
        HTML::Transmorgify::Attributes for the opening tag ($starting_attr)
        is passed in.

        The callbacks for multiple tags may be overridden with the %tags
        argument. Since these may be shared callbacks, only the callbacks
        for the module doing the overridding should be modified. That should
        be specified as: $$opts{tag_package}.

        The call to capture_returns after the input pointer has moved past
        "</$tag>". No "</$tag>". is pushed into the compile output buffer
        "HTML::Transmorgify::rbuf" and the callbacks requested for "</$tag>"
        have not been called. To invoke them, simply run:

         $deferred->doit();

        This will also push a "</$tag>" into the output stream.

    continue_compile($tag, $starting_attr, $opts, %tags)
        *Exclusive tag callbacks only.* continue_compile() is very much like
        "capture_compile()" except that the output from the compilation
        process is appended to the existing "HTML::Transmorgify::rbuf" array
        and there is no return value from the function.

        The purpose of continue_compile() is to allow the normal compilation
        process to continue for a while. The function returns after
        "</$tag>" has been rpocessed. Tag callbacks may be temporarily
        overridden for the duration of the call.

        If $tag is undef, the compilation will continue until the end of
        $$textref is reached or until a "HTML::Transmorgify::CloseTag"
        callback is invoked.

    local($HTML::Transmorgify::dispatch{"/$tag"}) =
    HTML::Transmorgify::ClosedTag->new($HTML::Transmorgify::dispatch{"/$tag"
    });
        This will override the callbacks for "</$tag>" so that
        continue_compile or capture_compile will stop when "</$tag>" is
        reached. This is normally done automatically but this invocation is
        needed if stopping for more than one tag.

    queue_capture($coderef)
        *Shared tag callbacks only.* Shared tag callbacks cannot use
        capture_compile() and continue_compile(): they can use
        queue_capture() instead. The function that handles calling all the
        shared tag callbacks will do a capture_compile(), looking for the
        close tag to match the current tag if any of the shared tag
        callbacks uses queue_capture().

        After the $coderefs are invoked (they receive the $rbuf from
        capture_compile() as their only argument) the temporary $rbuf
        compile buffer will be appended to the main compile buffer. After
        that, the deferred callbacks from the close tag will be invoked.

    queue_intercept($tag_pkg, %tags)
        *Shared tag callbacks only.* Shared tag callbacks cannot use
        capture_compile() and continue_compile(): they can use
        queue_capture() instead. The function that handles calling all the
        shared tag callbacks will do a continue_compile(), looking for the
        close tag to match the current tag if any of the shared tag
        callbacks uses queue_intercept(). If queue_capture is also used by a
        tag callback, then capture_compile() will be used instead.

        For the duration of the continue_compile() (or capture_compile())
        tag callbacks will be overridden by %tags. $tag_pkg should be set to
        the name of the invoking package (__PACKAGE__).

NAMING
    Calvin & Hobbs used to transmogrify not transmorgify things. This module
    will be renamed to fix my spelling mistake soon. Other API changes may
    still be made. Please contact the author if you want to be included in
    dicussions of this module's future.

SEE ALSO
    Other modules that do similar things: HTML::Seamstreess

AUTHOR
    David Muir Sharnoff <muir at idiom dot org>

    Development of this module is hosted on github:
    <http://github.com/muir/HTML--Transmorgify>.

    Some parts of this module are Copyright (C) Google, Inc.

LICENSE
    Your choice of LGPL or Artistic License.