home *** CD-ROM | disk | FTP | other *** search
-
- <HTML>
- <HEAD>
- <TITLE>HTML::Stream - HTML output stream class, and some markup utilities</TITLE>
- <LINK REL="stylesheet" HREF="../../../Active.css" TYPE="text/css">
- <LINK REV="made" HREF="mailto:">
- </HEAD>
-
- <BODY>
- <TABLE BORDER=0 CELLPADDING=0 CELLSPACING=0 WIDTH=100%>
- <TR><TD CLASS=block VALIGN=MIDDLE WIDTH=100% BGCOLOR="#cccccc">
- <STRONG><P CLASS=block> HTML::Stream - HTML output stream class, and some markup utilities</P></STRONG>
- </TD></TR>
- </TABLE>
-
- <A NAME="__index__"></A>
- <!-- INDEX BEGIN -->
-
- <UL>
-
- <LI><A HREF="#name">NAME</A></LI><LI><A HREF="#supportedplatforms">SUPPORTED PLATFORMS</A></LI>
-
- <LI><A HREF="#synopsis">SYNOPSIS</A></LI>
- <LI><A HREF="#description">DESCRIPTION</A></LI>
- <LI><A HREF="#introduction (the neapolitan dessert special)">INTRODUCTION (the Neapolitan dessert special)</A></LI>
- <UL>
-
- <LI><A HREF="#function interface">Function interface</A></LI>
- <LI><A HREF="#oo interface, vanilla">OO interface, vanilla</A></LI>
- <LI><A HREF="#oo interface, chocolate">OO interface, chocolate</A></LI>
- <LI><A HREF="#oo interface, with whipped cream">OO interface, with whipped cream</A></LI>
- <LI><A HREF="#oo interface, strawberry">OO interface, strawberry</A></LI>
- </UL>
-
- <LI><A HREF="#advanced topics">ADVANCED TOPICS</A></LI>
- <UL>
-
- <LI><A HREF="#autoformatting and inserting newlines">Auto-formatting and inserting newlines</A></LI>
- <LI><A HREF="#entities">Entities</A></LI>
- <LI><A HREF="#autoescaping: changing the way text is escaped">Auto-escaping: changing the way text is escaped</A></LI>
- <LI><A HREF="#outputting html to things besides filehandles">Outputting HTML to things besides filehandles</A></LI>
- <LI><A HREF="#subclassing">Subclassing</A></LI>
- </UL>
-
- <LI><A HREF="#public interface">PUBLIC INTERFACE</A></LI>
- <UL>
-
- <LI><A HREF="#functions">Functions</A></LI>
- <LI><A HREF="#vanilla">Vanilla</A></LI>
- <LI><A HREF="#strawberry">Strawberry</A></LI>
- <LI><A HREF="#chocolate">Chocolate</A></LI>
- </UL>
-
- <LI><A HREF="#subclasses">SUBCLASSES</A></LI>
- <UL>
-
- <LI><A HREF="#html::stream::latin1">HTML::Stream::Latin1</A></LI>
- </UL>
-
- <LI><A HREF="#performance">PERFORMANCE</A></LI>
- <LI><A HREF="#why in the world did i write this">WHY IN THE WORLD DID I WRITE THIS?</A></LI>
- <LI><A HREF="#change log">CHANGE LOG</A></LI>
- <LI><A HREF="#version">VERSION</A></LI>
- <LI><A HREF="#acknowledgements">ACKNOWLEDGEMENTS</A></LI>
- <LI><A HREF="#author">AUTHOR</A></LI>
- </UL>
- <!-- INDEX END -->
-
- <HR>
- <P>
- <H1><A NAME="name">NAME</A></H1>
- <P>HTML::Stream - HTML output stream class, and some markup utilities</P>
- <P>
- <HR>
- <H1><A NAME="supportedplatforms">SUPPORTED PLATFORMS</A></H1>
- <UL>
- <LI>Linux</LI>
- <LI>Solaris</LI>
- <LI>Windows</LI>
- </UL>
- <HR>
- <H1><A NAME="synopsis">SYNOPSIS</A></H1>
- <P>Here's small sample of some of the non-OO ways you can use this module:</P>
- <PRE>
- use HTML::Stream qw(:funcs);
- </PRE>
- <PRE>
-
- print html_tag('A', HREF=>$link);
- print html_escape("<<Hello & welcome!>>");</PRE>
- <P>And some of the OO ways as well:</P>
- <PRE>
- use HTML::Stream;
- $HTML = new HTML::Stream \*STDOUT;
- </PRE>
- <PRE>
-
- # The vanilla interface...
- $HTML->tag('A', HREF=>"$href");
- $HTML->tag('IMG', SRC=>"logo.gif", ALT=>"LOGO");
- $HTML->text($copyright);
- $HTML->tag('_A');</PRE>
- <PRE>
-
- # The chocolate interface...
- $HTML -> A(HREF=>"$href");
- $HTML -> IMG(SRC=>"logo.gif", ALT=>"LOGO");
- $HTML -> t($caption);
- $HTML -> _A;</PRE>
- <PRE>
-
- # The chocolate interface, with whipped cream...
- $HTML -> A(HREF=>"$href")
- -> IMG(SRC=>"logo.gif", ALT=>"LOGO")
- -> t($caption)
- -> _A;</PRE>
- <PRE>
- # The strawberry interface...
- output $HTML [A, HREF=>"$href"],
- [IMG, SRC=>"logo.gif", ALT=>"LOGO"],
- $caption,
- [_A];</PRE>
- <P>
- <HR>
- <H1><A NAME="description">DESCRIPTION</A></H1>
- <P>The <STRONG>HTML::Stream</STRONG> module provides you with an object-oriented
- (and subclassable) way of outputting HTML. Basically, you open up
- an ``HTML stream'' on an existing filehandle, and then do all of your
- output to the HTML stream. You can intermix HTML-stream-output and
- ordinary-print-output, if you like.</P>
- <P>There's even a small built-in subclass, <STRONG>HTML::Stream::Latin1</STRONG>, which can
- handle Latin-1 input right out of the box. But all in good time...</P>
- <P>
- <HR>
- <H1><A NAME="introduction (the neapolitan dessert special)">INTRODUCTION (the Neapolitan dessert special)</A></H1>
- <P>
- <H2><A NAME="function interface">Function interface</A></H2>
- <P>Let's start out with the simple stuff.
- This module provides a collection of non-OO utility functions
- for escaping HTML text and producing HTML tags, like this:</P>
- <PRE>
- use HTML::Stream qw(:funcs); # imports functions from @EXPORT_OK
- </PRE>
- <PRE>
-
- print html_tag(A, HREF=>$url);
- print '© 1996 by', html_escape($myname), '!';
- print html_tag('/A');</PRE>
- <P>By the way: that last line could be rewritten as:</P>
- <PRE>
- print html_tag(_A);</PRE>
- <P>And if you need to get a parameter in your tag that doesn't have an
- associated value, supply the <EM>undefined</EM> value (<EM>not</EM> the empty string!):</P>
- <PRE>
- print html_tag(TD, NOWRAP=>undef, ALIGN=>'LEFT');
- </PRE>
- <PRE>
-
- <TD NOWRAP ALIGN=LEFT></PRE>
- <PRE>
-
- print html_tag(IMG, SRC=>'logo.gif', ALT=>'');</PRE>
- <PRE>
-
- <IMG SRC="logo.gif" ALT=""></PRE>
- <P>There are also some routines for reversing the process, like:</P>
- <PRE>
- $text = "This <i>isn't</i> "fun"...";
- print html_unmarkup($text);
- </PRE>
- <PRE>
-
- This isn't "fun"...</PRE>
- <PRE>
-
- print html_unescape($text);</PRE>
- <PRE>
-
- This isn't "fun"...</PRE>
- <P><EM>Yeah, yeah, yeah</EM>, I hear you cry. <EM>We've seen this stuff before.</EM>
- But wait! There's more...</P>
- <P>
- <H2><A NAME="oo interface, vanilla">OO interface, vanilla</A></H2>
- <P>Using the function interface can be tedious... so we also
- provide an <STRONG>``HTML output stream''</STRONG> class. Messages to an instance of
- that class generally tell that stream to output some HTML. Here's the
- above example, rewritten using HTML streams:</P>
- <PRE>
- use HTML::Stream;
- $HTML = new HTML::Stream \*STDOUT;
- </PRE>
- <PRE>
-
- $HTML->tag(A, HREF=>$url);
- $HTML->ent('copy');
- $HTML->text(" 1996 by $myname!");
- $HTML->tag(_A);</PRE>
- <P>As you've probably guessed:</P>
- <PRE>
- text() Outputs some text, which will be HTML-escaped.
- </PRE>
- <PRE>
-
- tag() Outputs an ordinary tag, like <A>, possibly with parameters.
- The parameters will all be HTML-escaped automatically.</PRE>
- <PRE>
-
- ent() Outputs an HTML entity, like the © or < .
- You mostly don't need to use it; you can often just put the
- Latin-1 representation of the character in the text().</PRE>
- <P>You might prefer to use <CODE>t()</CODE> and <CODE>e()</CODE> instead of <CODE>text()</CODE>
- and <A HREF="#item_ent"><CODE>ent()</CODE></A>: they're absolutely identical, and easier to type:</P>
- <PRE>
- $HTML -> tag(A, HREF=>$url);
- $HTML -> e('copy');
- $HTML -> t(" 1996 by $myname!");
- $HTML -> tag(_A);</PRE>
- <P>Now, it wouldn't be nice to give you those <CODE>text()</CODE> and <A HREF="#item_ent"><CODE>ent()</CODE></A> shortcuts
- without giving you one for <CODE>tag()</CODE>, would it? Of course not...</P>
- <P>
- <H2><A NAME="oo interface, chocolate">OO interface, chocolate</A></H2>
- <P>The known HTML tags are even given their own <STRONG>tag-methods,</STRONG> compiled on
- demand. The above code could be written even more compactly as:</P>
- <PRE>
- $HTML -> A(HREF=>$url);
- $HTML -> e('copy');
- $HTML -> t(" 1996 by $myname!");
- $HTML -> _A;</PRE>
- <P>As you've probably guessed:</P>
- <PRE>
- A(HREF=>$url) == tag(A, HREF=>$url) == <A HREF="/the/url">
- _A == tag(_A) == </A></PRE>
- <P>All of the autoloaded ``tag-methods'' use the tagname in <EM>all-uppercase</EM>.
- A <CODE>"_"</CODE> prefix on any tag-method means that an end-tag is desired.
- The <CODE>"_"</CODE> was chosen for several reasons:
- (1) it's short and easy to type,
- (2) it doesn't produce much visual clutter to look at,
- (3) <CODE>_TAG</CODE> looks a little like <CODE>/TAG</CODE> because of the straight line.</P>
- <UL>
- <LI>
- <EM>I know, I know... it looks like a private method.
- You get used to it. Really.</EM>
- <P></P></UL>
- <P>I should stress that this module will only auto-create tag methods
- for <STRONG>known</STRONG> HTML tags. So you're protected from typos like this
- (which will cause a fatal exception at run-time):</P>
- <PRE>
- $HTML -> IMGG(SRC=>$src);</PRE>
- <P>(You're not yet protected from illegal tag parameters, but it's a start,
- ain't it?)</P>
- <P>If you need to make a tag known (sorry, but this is currently a
- <EM>global</EM> operation, and not stream-specific), do this:</P>
- <PRE>
- accept_tag HTML::Stream 'MARQUEE'; # for you MSIE fans...</PRE>
- <P><STRONG>Note: there is no corresponding ``reject_tag''.</STRONG> I thought and thought
- about it, and could not convince myself that such a method would
- do anything more useful than cause other people's modules to suddenly
- stop working because some bozo function decided to reject the <CODE>FONT</CODE> tag.</P>
- <P>
- <H2><A NAME="oo interface, with whipped cream">OO interface, with whipped cream</A></H2>
- <P>In the grand tradition of C++, output method chaining is supported
- in both the Vanilla Interface and the Chocolate Interface.
- So you can (and probably should) write the above code as:</P>
- <PRE>
- $HTML -> A(HREF=>$url)
- -> e('copy') -> t(" 1996 by $myname!")
- -> _A;</PRE>
- <P><EM>But wait! Neapolitan ice cream has one more flavor...</EM></P>
- <P>
- <H2><A NAME="oo interface, strawberry">OO interface, strawberry</A></H2>
- <P>I was jealous of the compact syntax of HTML::AsSubs, but I didn't
- want to worry about clogging the namespace with a lot of functions
- like p(), a(), etc. (especially when markup-functions like <A HREF="../../../lib/Pod/perlfunc.html#item_tr"><CODE>tr()</CODE></A> conflict
- with existing Perl functions). So I came up with this:</P>
- <PRE>
- output $HTML [A, HREF=>$url], "Here's my $caption", [_A];</PRE>
- <P>Conceptually, arrayrefs are sent to <CODE>html_tag()</CODE>, and strings to
- <CODE>html_escape()</CODE>.</P>
- <P>
- <HR>
- <H1><A NAME="advanced topics">ADVANCED TOPICS</A></H1>
- <P>
- <H2><A NAME="autoformatting and inserting newlines">Auto-formatting and inserting newlines</A></H2>
- <P><EM>Auto-formatting</EM> is the name I give to the Chocolate Interface feature
- whereby newlines (and maybe, in the future, other things)
- are inserted before or after the tags you output in order to make
- your HTML more readable. So, by default, this:</P>
- <PRE>
- $HTML -> HTML
- -> HEAD
- -> TITLE -> t("Hello!") -> _TITLE
- -> _HEAD
- -> BODY(BGCOLOR=>'#808080');</PRE>
- <P>Actually produces this:</P>
- <PRE>
- <HTML><HTML>
- <HEAD>
- <TITLE>Hello!</TITLE>
- </HEAD>
- <BODY BGCOLOR="#808080"></PRE>
- <P><STRONG>To turn off autoformatting altogether</STRONG> on a given HTML::Stream object,
- use the <CODE>auto_format()</CODE> method:</P>
- <PRE>
- $HTML->auto_format(0); # stop autoformatting!</PRE>
- <P><STRONG>To change whether a newline is automatically output</STRONG> before/after the
- begin/end form of a tag at a <STRONG>global</STRONG> level, use <CODE>set_tag()</CODE>:</P>
- <PRE>
- HTML::Stream->set_tag('B', Newlines=>15); # 15 means "\n<B>\n \n</B>\n"
- HTML::Stream->set_tag('I', Newlines=>7); # 7 means "\n<I>\n \n</I> "</PRE>
- <P><STRONG>To change whether a newline is automatically output</STRONG> before/after the
- begin/end form of a tag <STRONG>for a given stream</STRONG> level, give the stream
- its own private ``tag info'' table, and then use <CODE>set_tag()</CODE>:</P>
- <PRE>
- $HTML->private_tags;
- $HTML->set_tag('B', Newlines=>0); # won't affect anyone else!</PRE>
- <P><STRONG>To output newlines explicitly</STRONG>, just use the special <CODE>nl</CODE> method
- in the Chocolate Interface:</P>
- <PRE>
- $HTML->nl; # one newline
- $HTML->nl(6); # six newlines</PRE>
- <P>I am sometimes asked, ``why don't you put more newlines in automatically?''
- Well, mostly because...</P>
- <UL>
- <LI>
- Sometimes you'll be outputting stuff inside a <CODE>PRE</CODE> environment.
- <P></P>
- <LI>
- Sometimes you really do want to jam things (like images, or table
- cell delimiters and the things they contain) right up against each other.
- <P></P></UL>
- <P>So I've stuck to outputting newlines in places where it's most likely
- to be harmless.</P>
- <P>
- <H2><A NAME="entities">Entities</A></H2>
- <P>As shown above, You can use the <A HREF="#item_ent"><CODE>ent()</CODE></A> (or <CODE>e()</CODE>) method to output
- an entity:</P>
- <PRE>
- $HTML->t('Copyright ')->e('copy')->t(' 1996 by Me!');</PRE>
- <P>But this can be a pain, particularly for generating output with
- non-ASCII characters:</P>
- <PRE>
- $HTML -> t('Copyright ')
- -> e('copy')
- -> t(' 1996 by Fran') -> e('ccedil') -> t('ois, Inc.!');</PRE>
- <P>Granted, Europeans can always type the 8-bit characters directly in
- their Perl code, and just have this:</P>
- <PRE>
- $HTML -> t("Copyright \251 1996 by Fran\347ois, Inc.!');</PRE>
- <P>But folks without 8-bit text editors can find this kind of output
- cumbersome to generate. Sooooooooo...</P>
- <P>
- <H2><A NAME="autoescaping: changing the way text is escaped">Auto-escaping: changing the way text is escaped</A></H2>
- <P><EM>Auto-escaping</EM> is the name I give to the act of taking an ``unsafe''
- string (one with ``>'', ``&'', etc.), and magically outputting ``safe'' HTML.</P>
- <P>The default ``auto-escape'' behavior of an HTML stream can be a drag if
- you've got a lot character entities that you want to output, or if
- you're using the Latin-1 character set, or some other input encoding.
- Fortunately, you can use the <CODE>auto_escape()</CODE> method to change the
- way a particular HTML::Stream works at any time.</P>
- <P>First, here's a couple of special invocations:</P>
- <PRE>
- $HTML->auto_escape('ALL'); # Default; escapes [<>"&] and 8-bit chars.
- $HTML->auto_escape('LATIN_1'); # Like ALL, but uses Latin-1 entities
- # instead of decimal equivalents.
- $HTML->auto_escape('NON_ENT'); # Like ALL, but leaves "&" alone.</PRE>
- <P>You can also install your own auto-escape function (note
- that you might very well want to install it for just a little bit
- only, and then de-install it):</P>
- <PRE>
- sub my_auto_escape {
- my $text = shift;
- HTML::Entities::encode($text); # start with default
- $text =~ s/\(c\)/©/ig; # (C) becomes copyright
- $text =~ s/\\,(c)/\&$1cedil;/ig; # \,c becomes a cedilla
- $text;
- }
- </PRE>
- <PRE>
-
- # Start using my auto-escape:
- my $old_esc = $HTML->auto_escape(\&my_auto_escape);</PRE>
- <PRE>
-
- # Output some stuff:
- $HTML-> IMG(SRC=>'logo.gif', ALT=>'Fran\,cois, Inc');
- output $HTML 'Copyright (C) 1996 by Fran\,cois, Inc.!';</PRE>
- <PRE>
-
- # Stop using my auto-escape:
- $HTML->auto_escape($old_esc);</PRE>
- <P>If you find yourself in a situation where you're doing this a lot,
- a better way is to create a <STRONG>subclass</STRONG> of HTML::Stream which installs
- your custom function when constructed. For an example, see the
- <STRONG>HTML::Stream::Latin1</STRONG> subclass in this module.</P>
- <P>
- <H2><A NAME="outputting html to things besides filehandles">Outputting HTML to things besides filehandles</A></H2>
- <P>As of Revision 1.21, you no longer need to supply <CODE>new()</CODE> with a
- filehandle: <EM>any object that responds to a print() method will do</EM>.
- Of course, this includes <STRONG>blessed</STRONG> FileHandles, and IO::Handles.</P>
- <P>If you supply a GLOB reference (like <CODE>\*STDOUT</CODE>) or a string (like
- <CODE>"Module::FH"</CODE>), HTML::Stream will automatically create an invisible
- object for talking to that filehandle (I don't dare bless it into a
- FileHandle, since the underlying descriptor would get closed when
- the HTML::Stream is destroyed, and you might not want that).</P>
- <P>You say you want to print to a string? For kicks and giggles, try this:</P>
- <PRE>
- package StringHandle;
- sub new {
- my $self = '';
- bless \$self, shift;
- }
- sub print {
- my $self = shift;
- $$self .= join('', @_);
- }
- </PRE>
- <PRE>
-
- package main;
- use HTML::Stream;</PRE>
- <PRE>
-
- my $SH = new StringHandle;
- my $HTML = new HTML::Stream $SH;
- $HTML -> H1 -> t("Hello & <<welcome>>!") -> _H1;
- print "PRINTED STRING: ", $$SH, "\n";</PRE>
- <P>
- <H2><A NAME="subclassing">Subclassing</A></H2>
- <P>This is where you can make your application-specific HTML-generating code
- <EM>much</EM> easier to look at. Consider this:</P>
- <PRE>
- package MY::HTML;
- @ISA = qw(HTML::Stream);
- </PRE>
- <PRE>
-
- sub Aside {
- $_[0] -> FONT(SIZE=>-1) -> I;
- }
- sub _Aside {
- $_[0] -> _I -> _FONT;
- }</PRE>
- <P>Now, you can do this:</P>
- <PRE>
- my $HTML = new MY::HTML \*STDOUT;
- </PRE>
- <PRE>
-
- $HTML -> Aside
- -> t("Don't drink the milk, it's spoiled... pass it on...")
- -> _Aside;</PRE>
- <P>If you're defining these markup-like, chocolate-interface-style functions,
- I recommend using mixed case with a leading capital. You probably
- shouldn't use all-uppercase, since that's what this module uses for
- real HTML tags.</P>
- <P>
- <HR>
- <H1><A NAME="public interface">PUBLIC INTERFACE</A></H1>
- <P>
- <H2><A NAME="functions">Functions</A></H2>
- <DL>
- <DT><STRONG><A NAME="item_html_escape_TEXT">html_escape TEXT</A></STRONG><BR>
- <DD>
- Given a TEXT string, turn the text into valid HTML by escaping ``unsafe''
- characters. Currently, the ``unsafe'' characters are 8-bit characters plus:
- <PRE>
- < > = &</PRE>
- <P><STRONG>Note:</STRONG> provided for convenience and backwards-compatibility only.
- You may want to use the more-powerful <STRONG>HTML::Entities::encode</STRONG>
- function instead.</P>
- <P></P>
- <DT><STRONG><A NAME="item_html_tag_TAG_%5B%2C_PARAM%3D%3EVALUE%2C_%2E%2E%2E%">html_tag TAG [, PARAM=>VALUE, ...]</A></STRONG><BR>
- <DD>
- Return the text for a given TAG, possibly with parameters.
- As an efficiency hack, only the values are HTML-escaped currently:
- it is assumed that the tag and parameters will already be safe.
- <P>For convenience and readability, you can say <CODE>_A</CODE> instead of <CODE>"/A"</CODE>
- for the first tag, if you're into barewords.</P>
- <P></P>
- <DT><STRONG><A NAME="item_html_unescape_TEXT">html_unescape TEXT</A></STRONG><BR>
- <DD>
- Remove angle-tag markup, and convert the standard ampersand-escapes
- (<CODE>lt</CODE>, <CODE>gt</CODE>, <CODE>amp</CODE>, <CODE>quot</CODE>, and <CODE>#ddd</CODE>) into ASCII characters.
- <P><STRONG>Note:</STRONG> provided for convenience and backwards-compatibility only.
- You may want to use the more-powerful <STRONG>HTML::Entities::decode</STRONG>
- function instead: unlike this function, it can collapse entities
- like <CODE>copy</CODE> and <CODE>ccedil</CODE> into their Latin-1 byte values.</P>
- <P></P>
- <DT><STRONG><A NAME="item_html_unmarkup_TEXT">html_unmarkup TEXT</A></STRONG><BR>
- <DD>
- Remove angle-tag markup from TEXT, but do not convert ampersand-escapes.
- Cheesy, but theoretically useful if you want to, say, incorporate
- externally-provided HTML into a page you're generating, and are worried
- that the HTML might contain undesirable markup.
- <P></P></DL>
- <P>
- <H2><A NAME="vanilla">Vanilla</A></H2>
- <DL>
- <DT><STRONG><A NAME="item_new_%5BPRINTABLE%5D">new [PRINTABLE]</A></STRONG><BR>
- <DD>
- <EM>Class method.</EM>
- Create a new HTML output stream.
- <P>The PRINTABLE may be a FileHandle, a glob reference, or any object
- that responds to a <A HREF="../../../lib/Pod/perlfunc.html#item_print"><CODE>print()</CODE></A> message.
- If no PRINTABLE is given, does a <A HREF="../../../lib/Pod/perlfunc.html#item_select"><CODE>select()</CODE></A> and uses that.</P>
- <P></P>
- <DT><STRONG><A NAME="item_auto_escape_%5BNAME%7CSUBREF%5D">auto_escape [NAME|SUBREF]</A></STRONG><BR>
- <DD>
- <EM>Instance method.</EM>
- Set the auto-escape function for this HTML stream.
- <P>If the argument is a subroutine reference SUBREF, then that subroutine
- will be used. Declare such subroutines like this:</P>
- <PRE>
- sub my_escape {
- my $text = shift; # it's passed in the first argument
- ...
- $text;
- }</PRE>
- <P>If a textual NAME is given, then one of the appropriate built-in
- functions is used. Possible values are:</P>
- <DL>
- <DT><STRONG><A NAME="item_ALL">ALL</A></STRONG><BR>
- <DD>
- Default for HTML::Stream objects. This escapes angle brackets,
- ampersands, double-quotes, and 8-bit characters. 8-bit characters
- are escaped using decimal entity codes (like <CODE>#123</CODE>).
- <P></P>
- <DT><STRONG><A NAME="item_LATIN_1">LATIN_1</A></STRONG><BR>
- <DD>
- Like <CODE>"ALL"</CODE>, but uses Latin-1 entity names (like <CODE>ccedil</CODE>) instead of
- decimal entity codes to escape characters. This makes the HTML more readable
- but it is currently not advised, as ``older'' browsers (like Netscape 2.0)
- do not recognize many of the ISO-8859-1 entity names (like <CODE>deg</CODE>).
- <P><STRONG>Warning:</STRONG> If you specify this option, you'll find that it attempts
- to ``require'' <STRONG>HTML::Entities</STRONG> at run time. That's because I didn't want
- to <EM>force</EM> you to have that module just to use the rest of HTML::Stream.
- To pick up problems at compile time, you are advised to say:</P>
- <PRE>
- use HTML::Stream;
- use HTML::Entities;</PRE>
- <P>in your source code.</P>
- <P></P>
- <DT><STRONG><A NAME="item_NON_ENT">NON_ENT</A></STRONG><BR>
- <DD>
- Like <CODE>"ALL"</CODE>, except that ampersands (&) are <EM>not</EM> escaped.
- This allows you to use &-entities in your text strings, while having
- everything else safely escaped:
- <PRE>
- output $HTML "If A is an acute angle, then A > 90°";</PRE>
- <P></P></DL>
- <P>Returns the previously-installed function, in the manner of <A HREF="../../../lib/Pod/perlfunc.html#item_select"><CODE>select()</CODE></A>.
- No arguments just returns the currently-installed function.</P>
- <DT><STRONG><A NAME="item_auto_format_ONOFF">auto_format ONOFF</A></STRONG><BR>
- <DD>
- <EM>Instance method.</EM>
- Set the auto-formatting characteristics for this HTML stream.
- Currently, all you can do is supply a single defined boolean
- argument, which turns auto-formatting ON (1) or OFF (0).
- The self object is returned.
- <P>Please use no other values; they are reserved for future use.</P>
- <P></P>
- <DT><STRONG><A NAME="item_comment">comment COMMENT</A></STRONG><BR>
- <DD>
- <EM>Instance method.</EM>
- Output an HTML comment.
- As of 1.29, a newline is automatically appended.
- <P></P>
- <DT><STRONG><A NAME="item_ent">ent ENTITY</A></STRONG><BR>
- <DD>
- <EM>Instance method.</EM>
- Output an HTML entity. For example, here's how you'd output a
- non-breaking space:
- <PRE>
- $html->ent('nbsp');</PRE>
- <P>You may abbreviate this method name as <CODE>e</CODE>:</P>
- <PRE>
- $html->e('nbsp');</PRE>
- <P><STRONG>Warning:</STRONG> this function assumes that the entity argument is legal.</P>
- <P></P>
- <DT><STRONG><A NAME="item_io">io</A></STRONG><BR>
- <DD>
- Return the underlying output handle for this HTML stream.
- All you can depend upon is that it is some kind of object
- which responds to a <A HREF="../../../lib/Pod/perlfunc.html#item_print"><CODE>print()</CODE></A> message:
- <PRE>
- $HTML->io->print("This is not auto-escaped or nuthin!");</PRE>
- <P></P>
- <DT><STRONG><A NAME="item_nl_%5BCOUNT%5D">nl [COUNT]</A></STRONG><BR>
- <DD>
- <EM>Instance method.</EM>
- Output COUNT newlines. If undefined, COUNT defaults to 1.
- <P></P>
- <DT><STRONG><A NAME="item_tag_TAGNAME_%5B%2C_PARAM%3D%3EVALUE%2C_%2E%2E%2E%5">tag TAGNAME [, PARAM=>VALUE, ...]</A></STRONG><BR>
- <DD>
- <EM>Instance method.</EM>
- Output a tag. Returns the self object, to allow method chaining.
- You can say <CODE>_A</CODE> instead of <CODE>"/A"</CODE>, if you're into barewords.
- <P></P>
- <DT><STRONG><A NAME="item_text_TEXT%2E%2E%2E">text TEXT...</A></STRONG><BR>
- <DD>
- <EM>Instance method.</EM>
- Output some text. You may abbreviate this method name as <CODE>t</CODE>:
- <PRE>
- $html->t('Hi there, ', $yournamehere, '!');</PRE>
- <P>Returns the self object, to allow method chaining.</P>
- <P></P>
- <DT><STRONG><A NAME="item_text_nbsp_TEXT%2E%2E%2E">text_nbsp TEXT...</A></STRONG><BR>
- <DD>
- <EM>Instance method.</EM>
- Output some text, but with all spaces output as non-breaking-space
- characters:
- <PRE>
- $html->t("To list your home directory, type: ")
- ->text_nbsp("ls -l ~yourname.")</PRE>
- <P>Returns the self object, to allow method chaining.</P>
- <P></P></DL>
- <P>
- <H2><A NAME="strawberry">Strawberry</A></H2>
- <DL>
- <DT><STRONG><A NAME="item_output_ITEM%2C%2E%2E%2E%2CITEM">output ITEM,...,ITEM</A></STRONG><BR>
- <DD>
- <EM>Instance method.</EM>
- Go through the items. If an item is an arrayref, treat it like
- the array argument to <CODE>html_tag()</CODE> and output the result. If an item
- is a text string, escape the text and output the result. Like this:
- <PRE>
- output $HTML [A, HREF=>$url], "Here's my $caption!", [_A];</PRE>
- <P></P></DL>
- <P>
- <H2><A NAME="chocolate">Chocolate</A></H2>
- <DL>
- <DT><STRONG><A NAME="item_accept_tag_TAG">accept_tag TAG</A></STRONG><BR>
- <DD>
- <EM>Class method.</EM>
- Declares that the tag is to be accepted as valid HTML (if it isn't already).
- For example, this...
- <PRE>
- # Make sure methods MARQUEE and _MARQUEE are compiled on demand:
- HTML::Stream->accept_tag('MARQUEE');</PRE>
- <P>...gives the Chocolate Interface permission to create (via AUTOLOAD)
- definitions for the MARQUEE and _MARQUEE methods, so you can then say:</P>
- <PRE>
- $HTML -> MARQUEE -> t("Hi!") -> _MARQUEE;</PRE>
- <P>If you want to set the default attribute of the tag as well, you can
- do so via the <CODE>set_tag()</CODE> method instead; it will effectively do an
- <CODE>accept_tag()</CODE> as well.</P>
- <PRE>
- # Make sure methods MARQUEE and _MARQUEE are compiled on demand,
- # *and*, set the characteristics of that tag.
- HTML::Stream->set_tag('MARQUEE', Newlines=>9);</PRE>
- <P></P>
- <DT><STRONG><A NAME="item_private_tags">private_tags</A></STRONG><BR>
- <DD>
- <EM>Instance method.</EM>
- Normally, HTML streams use a reference to a global table of tag
- information to determine how to do such things as auto-formatting,
- and modifications made to that table by <CODE>set_tag</CODE> will
- affect everyone.
- <P>However, if you want an HTML stream to have a private copy of that
- table to munge with, just send it this message after creating it.
- Like this:</P>
- <PRE>
- my $HTML = new HTML::Stream \*STDOUT;
- $HTML->private_tags;</PRE>
- <P>Then, you can say stuff like:</P>
- <PRE>
- $HTML->set_tag('PRE', Newlines=>0);
- $HTML->set_tag('BLINK', Newlines=>9);</PRE>
- <P>And it won't affect anyone else's <EM>auto-formatting</EM> (although they will
- possibly be able to use the BLINK tag method without a fatal
- exception <CODE>:-(</CODE> ).</P>
- <P>Returns the self object.</P>
- <P></P>
- <DT><STRONG><A NAME="item_set_tag_TAG%2C_%5BTAGINFO%2E%2E%2E%5D">set_tag TAG, [TAGINFO...]</A></STRONG><BR>
- <DD>
- <EM>Class/instance method.</EM>
- Accept the given TAG in the Chocolate Interface, and (if TAGINFO
- is given) alter its characteristics when being output.
- <UL>
- <LI>
- <STRONG>If invoked as a class method,</STRONG> this alters the ``master tag table'',
- and allows a new tag to be supported via an autoloaded method:
- <PRE>
- HTML::Stream->set_tag('MARQUEE', Newlines=>9);</PRE>
- <P>Once you do this, <EM>all</EM> HTML streams you open from then on
- will allow that tag to be output in the chocolate interface.</P>
- <P></P>
- <LI>
- <STRONG>If invoked as an instance method,</STRONG> this alters the ``tag table'' referenced
- by that HTML stream, usually for the purpose of affecting things like
- the auto-formatting on that HTML stream.
- <P><STRONG>Warning:</STRONG> by default, an HTML stream just references the ``master tag table''
- (this makes <CODE>new()</CODE> more efficient), so <EM>by default, the
- instance method will behave exactly like the class method.</EM></P>
- <PRE>
- my $HTML = new HTML::Stream \*STDOUT;
- $HTML->set_tag('BLINK', Newlines=>0); # changes it for others!</PRE>
- <P>If you want to diddle with <EM>one</EM> stream's auto-formatting <EM>only,</EM>
- you'll need to give that stream its own <EM>private</EM> tag table. Like this:</P>
- <PRE>
- my $HTML = new HTML::Stream \*STDOUT;
- $HTML->private_tags;
- $HTML->set_tag('BLINK', Newlines=>0); # doesn't affect other streams</PRE>
- <P><STRONG>Note:</STRONG> this will still force an default entry for BLINK in the <EM>master</EM>
- tag table: otherwise, we'd never know that it was legal to AUTOLOAD a
- BLINK method. However, it will only alter the <EM>characteristics</EM> of the
- BLINK tag (like auto-formatting) in the <EM>object's</EM> tag table.</P>
- <P></P></UL>
- <P>The TAGINFO, if given, is a set of key=>value pairs with the following
- possible keys:</P>
- <DL>
- <DT><STRONG><A NAME="item_Newlines">Newlines</A></STRONG><BR>
- <DD>
- Assumed to be a number which encodes how newlines are to be output
- before/after a tag. The value is the logical OR (or sum) of a set of flags:
- <PRE>
- 0x01 newline before <TAG> .<TAG>. .</TAG>.
- 0x02 newline after <TAG> | | | |
- 0x04 newline before </TAG> 1 2 4 8
- 0x08 newline after </TAG></PRE>
- <P>Hence, to output BLINK environments which are preceded/followed by newlines:</P>
- <PRE>
- set_tag HTML::Stream 'BLINK', Newlines=>9;</PRE>
- <P></P></DL>
- <P>Returns the self object on success.</P>
- <DT><STRONG><A NAME="item_tags">tags</A></STRONG><BR>
- <DD>
- <EM>Class/instance method.</EM>
- Returns an unsorted list of all tags in the class/instance tag table
- (see <CODE>set_tag</CODE> for class/instance method differences).
- <P></P></DL>
- <P>
- <HR>
- <H1><A NAME="subclasses">SUBCLASSES</A></H1>
- <P>
- <H2><A NAME="html::stream::latin1">HTML::Stream::Latin1</A></H2>
- <P>A small, public package for outputting Latin-1 markup. Its
- default auto-escape function is <A HREF="#item_LATIN_1"><CODE>LATIN_1</CODE></A>, which tries to output
- the mnemonic entity markup (e.g., <CODE>ç</CODE>) for ISO-8859-1 characters.</P>
- <P>So using HTML::Stream::Latin1 like this:</P>
- <PRE>
- use HTML::Stream;
- </PRE>
- <PRE>
-
- $HTML = new HTML::Stream::Latin1 \*STDOUT;
- output $HTML "\253A right angle is 90\260, \277No?\273\n";</PRE>
- <P>Prints this:</P>
- <PRE>
- «A right angle is 90°, ¿No?»</PRE>
- <P>Instead of what HTML::Stream would print, which is this:</P>
- <PRE>
- «A right angle is 90°, ¿No?»</PRE>
- <P><STRONG>Warning:</STRONG> a lot of Latin-1 HTML markup is not recognized by older
- browsers (e.g., Netscape 2.0). Consider using HTML::Stream; it will output
- the decimal entities which currently seem to be more ``portable''.</P>
- <P><STRONG>Note:</STRONG> using this class ``requires'' that you have HTML::Entities.</P>
- <P>
- <HR>
- <H1><A NAME="performance">PERFORMANCE</A></H1>
- <P>Slower than I'd like. Both the <CODE>output()</CODE> method and the various ``tag''
- methods seem to run about 5 times slower than the old
- just-hardcode-the-darn stuff approach. That is, in general, this:</P>
- <PRE>
- ### Approach #1...
- tag $HTML 'A', HREF=>"$href";
- tag $HTML 'IMG', SRC=>"logo.gif", ALT=>"LOGO";
- text $HTML $caption;
- tag $HTML '_A';
- text $HTML $a_lot_of_text;</PRE>
- <P>And this:</P>
- <PRE>
- ### Approach #2...
- output $HTML [A, HREF=>"$href"],
- [IMG, SRC=>"logo.gif", ALT=>"LOGO"],
- $caption,
- [_A];
- output $HTML $a_lot_of_text;</PRE>
- <P>And this:</P>
- <PRE>
- ### Approach #3...
- $HTML -> A(HREF=>"$href")
- -> IMG(SRC=>"logo.gif", ALT=>"LOGO")
- -> t($caption)
- -> _A
- -> t($a_lot_of_text);</PRE>
- <P>Each run about 5x slower than this:</P>
- <PRE>
- ### Approach #4...
- print '<A HREF="', html_escape($href), '>',
- '<IMG SRC="logo.gif" ALT="LOGO">',
- html_escape($caption),
- '</A>';
- print html_escape($a_lot_of_text);</PRE>
- <P>Of course, I'd much rather use any of first three <EM>(especially #3)</EM>
- if I had to get something done right in a hurry. Or did you not notice
- the typo in approach #4? <CODE>;-)</CODE></P>
- <P>(BTW, thanks to Benchmark:: for allowing me to... er... benchmark stuff.)</P>
- <P>
- <HR>
- <H1><A NAME="why in the world did i write this">WHY IN THE WORLD DID I WRITE THIS?</A></H1>
- <P>I was just mucking about with different ways of generating large
- HTML documents, seeing which ways I liked the most/least.</P>
- <P>
- <HR>
- <H1><A NAME="change log">CHANGE LOG</A></H1>
- <DL>
- <DT><STRONG><A NAME="item_Version_1%2E45">Version 1.45</A></STRONG><BR>
- <DD>
- Cleanup for Perl 5.005: removed duplicate typeglob assignments.
- <P></P>
- <DT><STRONG><A NAME="item_Version_1%2E44">Version 1.44</A></STRONG><BR>
- <DD>
- Win95 install (5.004) now works.
- Added SYNOPSIS to POD.
- <P></P>
- <DT><STRONG><A NAME="item_Version_1%2E41">Version 1.41</A></STRONG><BR>
- <DD>
- Removed $& for efficiency.
- <EM>Thanks, Andreas!</EM>
- <P>Added support for OPTION, and default now puts newlines after SELECT
- and /SELECT. Also altered ``TELEM'' syntax to put newline after end-tags
- of list element tags (like /OPTION, /LI, etc.). In theory, this change
- could produce undesireable results for folks who embed lists inside of PRE
- environments... however, that kind of stuff was done in the days before
- TABLEs; also, you can always turn it off if you really need to.
- <EM>Thanks to John D Groenveld for these patches.</EM></P>
- <P>Added text_nbsp().
- <EM>Thanks to John D Groenveld for the patch.</EM>
- This method may also be invoked as <CODE>nbsp_text()</CODE> as in the original patch,
- but that's sort of a private tip-of-the-hat to the patch author, and the
- synonym may go away in the future.</P>
- <P></P>
- <DT><STRONG><A NAME="item_Version_1%2E37">Version 1.37</A></STRONG><BR>
- <DD>
- No real change; just trying to make CPAN.pm happier.
- <P></P>
- <DT><STRONG><A NAME="item_Version_1%2E32">Version 1.32</A></STRONG><BR>
- <DD>
- <STRONG>NEW TOOL for generating Perl code which uses HTML::Stream!</STRONG>
- Check your toolkit for <STRONG>html2perlstream</STRONG>.
- <P>Added built-in support for escaping 8-bit characters.</P>
- <P>Added <A HREF="#item_LATIN_1"><CODE>LATIN_1</CODE></A> auto-escape, which uses HTML::Entities to generate
- mnemonic entities. This is now the default method for HTML::Stream::Latin1.</P>
- <P>Added <CODE>auto_format(),</CODE>
- so you can now turn auto-formatting off/on.</P>
- <P>Added <A HREF="#item_private_tags"><CODE>private_tags()</CODE></A>,
- so it is now possible for HTML streams to each have their own ``private''
- copy of the %Tags table, for use by <CODE>set_tag()</CODE>.</P>
- <P>Added <CODE>set_tag()</CODE>. The tags tables may now be modified dynamically so
- as to change how formatting is done on-the-fly. This will hopefully not
- compromise the efficiency of the chocolate interface (until now,
- the formatting was compiled into the method itself), and <EM>will</EM> add
- greater flexibility for more-complex programs.</P>
- <P>Added POD documentation for all subroutines in the public interface.</P>
- <P></P>
- <DT><STRONG><A NAME="item_Version_1%2E29">Version 1.29</A></STRONG><BR>
- <DD>
- Added terminating newline to comment().
- <EM>Thanks to John D Groenveld for the suggestion and the patch.</EM>
- <P></P>
- <DT><STRONG><A NAME="item_Version_1%2E27">Version 1.27</A></STRONG><BR>
- <DD>
- Added built-in HTML::Stream::Latin1, which does a very simple encoding
- of all characters above ASCII 127.
- <P>Fixed bug in accept_tag(), where 'my' variable was shadowing argument.
- <EM>Thanks to John D Groenveld for the bug report and the patch.</EM></P>
- <P></P>
- <DT><STRONG><A NAME="item_Version_1%2E26">Version 1.26</A></STRONG><BR>
- <DD>
- Start of history.
- <P></P></DL>
- <P>
- <HR>
- <H1><A NAME="version">VERSION</A></H1>
- <P>$Revision: 1.45 $</P>
- <P>
- <HR>
- <H1><A NAME="acknowledgements">ACKNOWLEDGEMENTS</A></H1>
- <P>Warmest thanks to...</P>
- <PRE>
- John Buckman For suggesting that I write an "html2perlstream",
- and inspiring me to look at supporting Latin-1.
- Tony Cebzanov For suggesting that I write an "html2perlstream"
- John D Groenveld Bug reports, patches, and suggestions
- B. K. Oxley (binkley) For suggesting the support of "writing to strings"
- which became the "printable" interface.</PRE>
- <P>
- <HR>
- <H1><A NAME="author">AUTHOR</A></H1>
- <P>Eryq, <EM><A HREF="mailto:eryq@zeegee.com">eryq@zeegee.com</A></EM>.
- President, Zero G Inc. (<EM><A HREF="http://www.zeegee.com">http://www.zeegee.com</A></EM>).</P>
- <P>Enjoy.</P>
- <TABLE BORDER=0 CELLPADDING=0 CELLSPACING=0 WIDTH=100%>
- <TR><TD CLASS=block VALIGN=MIDDLE WIDTH=100% BGCOLOR="#cccccc">
- <STRONG><P CLASS=block> HTML::Stream - HTML output stream class, and some markup utilities</P></STRONG>
- </TD></TR>
- </TABLE>
-
- </BODY>
-
- </HTML>
-