/[suikacvs]/test/html-webhacc/WebHACC/Output.pm
Suika

Contents of /test/html-webhacc/WebHACC/Output.pm

Parent Directory Parent Directory | Revision Log Revision Log


Revision 1.2 - (show annotations) (download)
Sun Jul 20 16:53:10 2008 UTC (16 years, 4 months ago) by wakaba
Branch: MAIN
Changes since 1.1: +16 -1 lines
++ ChangeLog	20 Jul 2008 16:48:51 -0000
2008-07-21  Wakaba  <wakaba@suika.fam.cx>

	* cc.cgi: Errors and results are now handled by WebHACC::Result.
	Decode |uri| parameter as UTF-8.  HTTP header dump and
	input error are now handled by WebHACC::Input.

++ html/WebHACC/Language/ChangeLog	20 Jul 2008 16:53:06 -0000
2008-07-21  Wakaba  <wakaba@suika.fam.cx>

	* Base.pm (_get_cc_url, _encode_url_component): Remove (now
	supported by WebHACC::Output).

	* CSS.pm, CacheManifest.pm, DOM.pm, Default.pm,
	HTML.pm, WebIDL.pm, XML.pm: Error reporting is now delegated to
	WebHACC::Result.

++ html/WebHACC/ChangeLog	20 Jul 2008 16:50:41 -0000
2008-07-21  Wakaba  <wakaba@suika.fam.cx>

	* Input.pm (generate_transfer_sections, generate_http_header_section):
	New (partially comes from cc.cgi).

	* Output.pm (link): Call |start_tag| such that attributes
	can be set.
	(link_to_webhacc): New.
	(encode_url_component): From WebHACC::Language::Base.

	* Result.pm: Support for error outputting and result table
	generation.

1 package WebHACC::Output;
2 use strict;
3 require IO::Handle;
4
5 my $htescape = sub ($) {
6 my $s = $_[0];
7 $s =~ s/&/&amp;/g;
8 $s =~ s/</&lt;/g;
9 $s =~ s/>/&gt;/g;
10 $s =~ s/"/&quot;/g;
11 $s =~ s{([\x00-\x09\x0B-\x1F\x7F-\xA0\x{FEFF}\x{FFFC}-\x{FFFF}])}{
12 sprintf '<var>U+%04X</var>', ord $1;
13 }ge;
14 return $s;
15 };
16
17 sub new ($) {
18 return bless {nav => []}, shift;
19 } # new
20
21 sub input ($;$) {
22 if (@_ > 1) {
23 if (defined $_[1]) {
24 $_[0]->{input} = $_[1];
25 } else {
26 delete $_[0]->{input};
27 }
28 }
29
30 return $_[0]->{input};
31 } # input
32
33 sub handle ($;$) {
34 if (@_ > 1) {
35 if (defined $_[1]) {
36 $_[0]->{handle} = $_[1];
37 } else {
38 delete $_[0]->{handle};
39 }
40 }
41
42 return $_[0]->{handle};
43 } # handle
44
45 sub set_utf8 ($) {
46 binmode shift->{handle}, ':utf8';
47 } # set_utf8
48
49 sub set_flush ($) {
50 shift->{handle}->autoflush (1);
51 } # set_flush
52
53 sub unset_flush ($) {
54 shift->{handle}->autoflush (0);
55 } # unset_flush
56
57 sub html ($$) {
58 shift->{handle}->print (shift);
59 } # html
60
61 sub text ($$) {
62 shift->html ($htescape->(shift));
63 } # text
64
65 sub url ($$%) {
66 my ($self, $url, %opt) = @_;
67 $self->html (q[<code class=uri>&lt;]);
68 $self->link ($url, %opt, url => $url);
69 $self->html (q[></code>]);
70 } # url
71
72 sub start_tag ($$%) {
73 my ($self, $tag_name, %opt) = @_;
74 $self->html ('<' . $htescape->($tag_name)); # escape for safety
75 if (exists $opt{id}) {
76 my $id = $self->input->id_prefix . $opt{id};
77 $self->html (' id="' . $htescape->($id) . '"');
78 delete $opt{id};
79 }
80 for (keys %opt) { # for safety
81 $self->html (' ' . $htescape->($_) . '="' . $htescape->($opt{$_}) . '"');
82 }
83 $self->html ('>');
84 } # start_tag
85
86 sub end_tag ($$) {
87 shift->html ('</' . $htescape->(shift) . '>');
88 } # end_tag
89
90 sub start_section ($%) {
91 my ($self, %opt) = @_;
92 $self->html ('<div class=section');
93 if (defined $opt{id}) {
94 my $id = $self->input->id_prefix . $opt{id};
95 $self->html (' id="' . $htescape->($id) . '"');
96 push @{$self->{nav}}, [$id => $opt{short_title} || $opt{title}]
97 unless $self->input->nested;
98 }
99 $self->html ('><h2>' . $htescape->($opt{title}) . '</h2>');
100 } # start_section
101
102 sub end_section ($) {
103 my $self = shift;
104 $self->html ('</div>');
105 $self->{handle}->flush;
106 } # end_section
107
108 sub start_code_block ($) {
109 shift->html ('<pre><code>');
110 } # start_code_block
111
112 sub end_code_block ($) {
113 shift->html ('</code></pre>');
114 } # end_code_block
115
116 sub code ($$) {
117 shift->html ('<code>' . $htescape->(shift) . '</code>');
118 } # code
119
120 sub link ($$%) {
121 my ($self, $content, %opt) = @_;
122 $self->start_tag ('a', %opt, href => $opt{url});
123 $self->text ($content);
124 $self->html ('</a>');
125 } # link
126
127 sub xref ($$%) {
128 my ($self, $content, %opt) = @_;
129 $self->html ('<a href="#' . $htescape->($self->input->id_prefix . $opt{target}) . '">');
130 $self->text ($content);
131 $self->html ('</a>');
132 } # xref
133
134 sub link_to_webhacc ($$%) {
135 my ($self, $content, %opt) = @_;
136 $opt{url} = './?uri=' . $self->encode_url_component ($opt{url});
137 $self->link ($content, %opt);
138 } # link_to_webhacc
139
140 sub nav_list ($) {
141 my $self = shift;
142 $self->html (q[<ul class="navigation" id="nav-items">]);
143 for (@{$self->{nav}}) {
144 $self->html (qq[<li><a href="@{[$htescape->($_->[0])]}">@{[$htescape->($_->[1])]}</a>]);
145 }
146 $self->html ('</ul>');
147 } # nav_list
148
149
150 sub encode_url_component ($$) {
151 shift;
152 require Encode;
153 my $s = Encode::encode ('utf8', shift);
154 $s =~ s/([^0-9A-Za-z_.~-])/sprintf '%%%02X', ord $1/ge;
155 return $s;
156 } # encode_url_component
157
158 1;

admin@suikawiki.org
ViewVC Help
Powered by ViewVC 1.1.24