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

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

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 1.7 by wakaba, Mon Jul 21 12:56:34 2008 UTC revision 1.17 by wakaba, Fri Aug 15 08:36:41 2008 UTC
# Line 16  my $htescape = sub ($) { Line 16  my $htescape = sub ($) {
16    return $s;    return $s;
17  };  };
18    
19    my $htescape_value = sub ($) {
20      my $s = $_[0];
21      $s =~ s/&/&/g;
22      $s =~ s/</&lt;/g;
23      $s =~ s/>/&gt;/g;
24      $s =~ s/"/&quot;/g;
25      return $s;
26    };
27    
28  sub new ($) {  sub new ($) {
29    return bless {nav => [], section_rank => 1}, shift;    require WebHACC::Input;
30      return bless {nav => [], section_rank => 1,
31                    input => WebHACC::Input->new}, shift;
32  } # new  } # new
33    
34  sub input ($;$) {  sub input ($;$) {
# Line 25  sub input ($;$) { Line 36  sub input ($;$) {
36      if (defined $_[1]) {      if (defined $_[1]) {
37        $_[0]->{input} = $_[1];        $_[0]->{input} = $_[1];
38      } else {      } else {
39        delete $_[0]->{input};        $_[0]->{input} = WebHACC::Input->new;
40      }      }
41    }    }
42        
# Line 44  sub handle ($;$) { Line 55  sub handle ($;$) {
55    return $_[0]->{handle};    return $_[0]->{handle};
56  } # handle  } # handle
57    
58    sub has_error ($;$) {
59      if (@_ > 1) {
60        if (defined $_[1]) {
61          $_[0]->{has_error} = 1;
62        } else {
63          delete $_[0]->{has_error};
64        }
65      }
66      
67      return $_[0]->{has_error};
68    } # has_error
69    
70  sub set_utf8 ($) {  sub set_utf8 ($) {
71    binmode shift->{handle}, ':utf8';    binmode shift->{handle}, ':utf8';
72  } # set_utf8  } # set_utf8
# Line 73  sub url ($$%) { Line 96  sub url ($$%) {
96    
97  sub start_tag ($$%) {  sub start_tag ($$%) {
98    my ($self, $tag_name, %opt) = @_;    my ($self, $tag_name, %opt) = @_;
99    $self->html ('<' . $htescape->($tag_name)); # escape for safety    $self->html ('<' . $htescape_value->($tag_name)); # escape for safety
100    if (exists $opt{id}) {    if (exists $opt{id}) {
101      my $id = $self->input->id_prefix . $opt{id};      my $id = $self->input->id_prefix . $opt{id};
102      $self->html (' id="' . $htescape->($id) . '"');      $self->html (' id="' . $htescape_value->($id) . '"');
103      delete $opt{id};      delete $opt{id};
104    }    }
105    for (keys %opt) {    # for safety    for (keys %opt) {    # for safety
106      $self->html (' ' . $htescape->($_) . '="' . $htescape->($opt{$_}) . '"');      $self->html (' ' . $htescape_value->($_) . '="' .
107                     $htescape_value->($opt{$_}) . '"');
108    }    }
109    $self->html ('>');    $self->html ('>');
110  } # start_tag  } # start_tag
111    
112  sub end_tag ($$) {  sub end_tag ($$) {
113    shift->html ('</' . $htescape->(shift) . '>');    shift->html ('</' . $htescape_value->(shift) . '>');
114  } # end_tag  } # end_tag
115    
116  sub start_section ($%) {  sub start_section ($%) {
117    my ($self, %opt) = @_;    my ($self, %opt) = @_;
118    
119      my $class = 'section';
120    if (defined $opt{role}) {    if (defined $opt{role}) {
121      if ($opt{role} eq 'parse-errors') {      if ($opt{role} eq 'parse-errors') {
122        $opt{id} ||= 'parse-errors';        $opt{id} ||= 'parse-errors';
123        $opt{title} ||= 'Parse Errors Section';        $opt{title} ||= 'Parse Errors Section';
124        $opt{short_title} ||= 'Parse Errors';        $opt{short_title} ||= 'Parse Errors';
125          $class .= ' errors';
126        delete $opt{role};        delete $opt{role};
127      } elsif ($opt{role} eq 'structure-errors') {      } elsif ($opt{role} eq 'structure-errors') {
128        $opt{id} ||= 'document-errors';        $opt{id} ||= 'document-errors';
129        $opt{title} ||= 'Structural Errors';        $opt{title} ||= 'Structural Errors';
130        $opt{short_title} ||= 'Struct. Errors';        $opt{short_title} ||= 'Struct. Errors';
131          $class .= ' errors';
132          delete $opt{role};
133        } elsif ($opt{role} eq 'transfer-errors') {
134          $opt{id} ||= 'transfer-errors';
135          $opt{title} ||= 'Transfer Errors';
136          $opt{short_title} ||= 'Trans. Errors';
137          $class .= ' errors';
138        delete $opt{role};        delete $opt{role};
139      } elsif ($opt{role} eq 'reformatted') {      } elsif ($opt{role} eq 'reformatted') {
140        $opt{id} ||= 'document-tree';        $opt{id} ||= 'document-tree';
141        $opt{title} ||= 'Reformatted Document Source';        $opt{title} ||= 'Reformatted Document Source';
142        $opt{short_title} ||= 'Reformatted';        $opt{short_title} ||= 'Reformatted';
143          $class .= ' dump';
144        delete $opt{role}        delete $opt{role}
145      } elsif ($opt{role} eq 'tree') {      } elsif ($opt{role} eq 'tree') {
146        $opt{id} ||= 'document-tree';        $opt{id} ||= 'document-tree';
147        $opt{title} ||= 'Document Tree';        $opt{title} ||= 'Document Tree';
148        $opt{short_title} ||= 'Tree';        $opt{short_title} ||= 'Tree';
149          $class .= ' dump';
150        delete $opt{role};        delete $opt{role};
151      } elsif ($opt{role} eq 'structure') {      } elsif ($opt{role} eq 'structure') {
152        $opt{id} ||= 'document-structure';        $opt{id} ||= 'document-structure';
153        $opt{title} ||= 'Document Structure';        $opt{title} ||= 'Document Structure';
154        $opt{short_title} ||= 'Structure';        $opt{short_title} ||= 'Structure';
155          $class .= ' dump';
156          delete $opt{role};
157        } elsif ($opt{role} eq 'subdoc') {
158          $class .= ' subdoc';
159          delete $opt{role};
160        } elsif ($opt{role} eq 'source') {
161          $opt{id} ||= 'source-string';
162          $opt{title} ||= 'Document Source';
163          $opt{short_title} ||= 'Source';
164          $class .= ' source';
165          delete $opt{role};
166        } elsif ($opt{role} eq 'result') {
167          $opt{id} ||= 'result-summary';
168          $opt{title} ||= 'Result';
169          $class .= ' result';
170        delete $opt{role};        delete $opt{role};
171      }      }
172    }    }
173    
174    $self->{section_rank}++;    $self->{section_rank}++;
175    $self->html ('<div class=section');    $self->html (qq[<div class="$class"]);
176    if (defined $opt{id}) {    if (defined $opt{id}) {
177      my $id = $self->input->id_prefix . $opt{id};      my $prefix = $self->input->id_prefix;
178      $self->html (' id="' . $htescape->($id) . '"');      $opt{parent_id} ||= $prefix;
179      push @{$self->{nav}},      my $id = $prefix . $opt{id};
180          [$id => $opt{short_title} || $opt{title} => $opt{text}]      $self->html (' id="' . $htescape->($id) . '">');
181          if $self->{section_rank} == 2;      if ($self->{section_rank} == 2 or length $opt{parent_id}) {
182          my $st = $opt{short_title} || $opt{title};
183          push @{$self->{nav}},
184              [$id => $st => $opt{text}];
185          
186          $self->start_tag ('script');
187          $self->html (qq[ addSectionLink ('$id', ']);
188          $self->nl_text ($st, text => $opt{text});
189          if (defined $opt{parent_id}) {
190            $self->html (q[', '] . $opt{parent_id});
191          }
192          $self->html (q[') ]);
193          $self->end_tag ('script');
194        }
195      } else {
196        $self->html ('>');
197    }    }
198    my $section_rank = $self->{section_rank};    my $section_rank = $self->{section_rank};
199    $section_rank = 6 if $section_rank > 6;    $section_rank = 6 if $section_rank > 6;
200    $self->html ('><h' . $section_rank . '>');    $self->html ('<h' . $section_rank . '>');
201    $self->nl_text ($opt{title}, text => $opt{text});    $self->nl_text ($opt{title}, text => $opt{text});
202    $self->html ('</h' . $section_rank . '>');    $self->html ('</h' . $section_rank . '>');
203  } # start_section  } # start_section
# Line 154  sub start_error_list ($%) { Line 219  sub start_error_list ($%) {
219      } elsif ($opt{role} eq 'structure-errors') {      } elsif ($opt{role} eq 'structure-errors') {
220        $opt{id} ||= 'document-errors-list';        $opt{id} ||= 'document-errors-list';
221        delete $opt{role};        delete $opt{role};
222        } elsif ($opt{role} eq 'transfer-errors') {
223          $opt{id} ||= 'transfer-errors-list';
224          delete $opt{role};
225      }      }
226    }    }
227    
228    $self->start_tag ('dl', %opt);    $self->start_tag ('dl', %opt);
229    
230      delete $self->{has_error}; # reset
231  } # start_error_list  } # start_error_list
232    
233  sub end_error_list ($%) {  sub end_error_list ($%) {
234    my ($self, %opt) = @_;    my ($self, %opt) = @_;
235    
236      my $no_error_message = 'No error found.';
237    
238    if (defined $opt{role}) {    if (defined $opt{role}) {
239      if ($opt{role} eq 'parse-errors') {      if ($opt{role} eq 'parse-errors') {
       delete $opt{role};  
240        $self->end_tag ('dl');        $self->end_tag ('dl');
241        ## NOTE: For parse error list, the |add_source_to_parse_error_list|        ## NOTE: For parse error list, the |add_source_to_parse_error_list|
242        ## method is invoked at the end of |generate_source_string_section|,        ## method is invoked at the end of |generate_source_string_section|,
243        ## since that generation method is invoked after the error list        ## since that generation method is invoked after the error list
244        ## is generated.        ## is generated.
245          $no_error_message = 'No parse error found.';
246      } elsif ($opt{role} eq 'structure-errors') {      } elsif ($opt{role} eq 'structure-errors') {
       delete $opt{role};  
247        $self->end_tag ('dl');        $self->end_tag ('dl');
248        $self->add_source_to_parse_error_list ('document-errors-list');        $self->add_source_to_parse_error_list ('document-errors-list');
249          $no_error_message = 'No structural error found.';
250        } elsif ($opt{role} eq 'transfer-errors') {
251          $self->end_tag ('dl');
252          $no_error_message = 'No transfer error found.';
253      } else {      } else {
254        $self->end_tag ('dl');        $self->end_tag ('dl');
255      }      }
256    } else {    } else {
257      $self->end_tag ('dl');      $self->end_tag ('dl');
258    }    }
259    
260      unless ($self->{has_error}) {
261        $self->start_tag ('p', class => 'no-errors');
262        $self->nl_text ($no_error_message);
263      }
264  } # end_error_list  } # end_error_list
265    
266  sub add_source_to_parse_error_list ($$) {  sub add_source_to_parse_error_list ($$) {
267    my $self = shift;    my $self = shift;
268    
269    $self->script (q[addSourceToParseErrorList ('] . $self->input->id_prefix .    $self->script (q[addSourceToParseErrorList ('] . $self->input->id_prefix .
270                   q[', '] . shift . q[')]);                   q[', '] . shift () . q[')]);
271  } # add_source_to_parse_error_list  } # add_source_to_parse_error_list
272    
273  sub start_code_block ($) {  sub start_code_block ($) {
# Line 218  sub dt ($$;%) { Line 298  sub dt ($$;%) {
298    $self->nl_text ($content, text => $opt{text});    $self->nl_text ($content, text => $opt{text});
299  } # dt  } # dt
300    
301    sub select ($$%) {
302      my ($self, $options, %opt) = @_;
303    
304      my $selected = $opt{selected};
305      delete $opt{selected};
306    
307      $self->start_tag ('select', %opt);
308      
309      my @options = @$options;
310      while (@options) {
311        my $opt = shift @options;
312        if ($opt->{options}) {
313          $self->html ('<optgroup label="');
314          $self->nl_text ($opt->{label});
315          $self->html ('">');
316          unshift @options, @{$opt->{options}}, {end_options => 1};
317        } elsif ($opt->{end_options}) {
318          $self->end_tag ('optgroup');
319        } else {
320          $self->start_tag ('option', value => $opt->{value},
321                            ((defined $selected and $opt->{value} eq $selected)
322                                 ? (selected => '') : ()));
323          $self->nl_text (defined $opt->{label} ? $opt->{label} : $opt->{value});
324        }
325      }
326    
327      $self->end_tag ('select');
328    } # select
329    
330  sub link ($$%) {  sub link ($$%) {
331    my ($self, $content, %opt) = @_;    my ($self, $content, %opt) = @_;
332    $self->start_tag ('a', %opt, href => $opt{url});    $self->start_tag ('a', %opt, href => $opt{url});
# Line 265  my $get_node_path = sub ($) { Line 374  my $get_node_path = sub ($) {
374    return join '/', @r;    return join '/', @r;
375  }; # $get_node_path  }; # $get_node_path
376    
377    my $get_object_path = sub ($) {
378      my $node = shift;
379      my @r;
380      while (defined $node) {
381        my $ref = ref $node;
382        $ref =~ /([^:]+)$/;
383        my $rs = $1;
384        my $node_name = $node->node_name;
385        if (defined $node_name) {
386          $rs .= ' <code>' . $htescape->($node_name) . '</code>';
387        }
388        $node = undef;
389        unshift @r, $rs;
390      }
391      return join '/', @r;
392    }; # $get_object_path
393    
394  sub node_link ($$) {  sub node_link ($$) {
395    my ($self, $node) = @_;    my ($self, $node) = @_;
396    $self->xref ($get_node_path->($node), target => 'node-' . refaddr $node);    if ($node->isa ('Message::IF::Node')) {
397        $self->xref ($get_node_path->($node), target => 'node-' . refaddr $node);
398      } else {
399        $self->html ($get_object_path->($node));
400      }
401  } # node_link  } # node_link
402    
403  {  {
# Line 367  sub html_header ($) { Line 497  sub html_header ($) {
497    $self->nl_text (q[WebHACC:Title]);    $self->nl_text (q[WebHACC:Title]);
498    $self->html (q[</title>    $self->html (q[</title>
499  <link rel="stylesheet" href="../cc-style.css" type="text/css">  <link rel="stylesheet" href="../cc-style.css" type="text/css">
500    <script src="../cc-script.js"></script>
501  </head>  </head>
502  <body>  <body onclick=" return onbodyclick (event) " onload=" onbodyload () ">
503  <h1>]);  <h1>]);
504    $self->nl_text (q[WebHACC:Heading]);    $self->nl_text (q[WebHACC:Heading]);
505    $self->html ('</h1>');    $self->html (q[</h1><script> insertNavSections () </script>]);
506  } # html_header  } # html_header
507    
508    sub generate_input_section ($$) {
509      my ($out, $cgi) = @_;
510    
511      require Encode;
512      my $decode = sub ($) {
513        if (defined $_[0]) {
514          return Encode::decode ('utf-8', $_[0]);
515        } else {
516          return undef;
517        }
518      }; # $decode
519    
520      my $options = sub ($) {
521        my $context = shift;
522    
523        $out->html (q[<div class="details default"><p class=legend onclick="nextSibling.style.display = nextSibling.style.display == 'block' ? 'none' : 'block'; parentNode.className = nextSibling.style.display == 'none' ? 'details' : 'details open'">]);
524        $out->nl_text (q[Options]);
525        $out->start_tag ('div');
526    
527        if ($context eq 'url') {
528          $out->start_tag ('p');
529          $out->start_tag ('label');
530          $out->start_tag ('input', type => 'checkbox', name => 'error-page',
531                           value => 1,
532                           ($cgi->get_parameter ('error-page')
533                                ? (checked => '') : ()));
534          $out->nl_text ('Check error page');
535          $out->end_tag ('label');
536        }
537    
538        $out->start_tag ('p');
539        $out->start_tag ('label');
540        $out->nl_text (q[Content type]);
541        $out->text (': ');
542        $out->select ([
543          {value => '', label => 'As specified'},
544          {value => 'application/atom+xml'},
545          {value => 'application/xhtml+xml'},
546          {value => 'application/xml'},
547          {value => 'text/html'},
548          {value => 'text/xml'},
549          {value => 'text/css'},
550          {value => 'text/cache-manifest'},
551          {value => 'text/x-webidl'},
552        ], name => 'i', selected => scalar $cgi->get_parameter ('i'));
553        $out->end_tag ('label');
554    
555        if ($context ne 'text') {
556          $out->start_tag ('p');
557          $out->start_tag ('label');
558          $out->nl_text (q[Charset]);
559          $out->text (q[: ]);
560          $out->select ([
561            {value => '', label => 'As specified'},
562            {label => 'Japanese charsets', options => [
563              {value => 'Windows-31J'},
564              {value => 'Shift_JIS'},
565              {value => 'EUC-JP'},
566              {value => 'ISO-2022-JP'},
567            ]},
568            {label => 'European charsets', options => [
569              {value => 'Windows-1252'},
570              {value => 'ISO-8859-1'},
571              {value => 'US-ASCII'},
572            ]},
573            {label => 'Asian charsets', options => [
574              {value => 'Windows-874'},
575              {value => 'ISO-8859-11'},
576              {value => 'TIS-620'},
577            ]},
578            {label => 'Unicode charsets', options => [
579              {value => 'UTF-8'},
580              {value => 'UTF-8n'},
581            ]},
582          ], name => 'charset',
583          selected => scalar $cgi->get_parameter ('charset'));
584          $out->end_tag ('label');
585        }
586    
587        if ($context eq 'text') {
588          $out->start_tag ('p');
589          $out->start_tag ('label');
590          $out->nl_text ('Setting innerHTML');
591          $out->text (': ');
592          $out->start_tag ('input', name => 'e',
593                           value => $decode->(scalar $cgi->get_parameter ('e')));
594          $out->end_tag ('label');
595        }
596    
597        $out->html (q[</div></div>]);
598      }; # $options
599    
600      $out->start_section (id => 'input', title => 'Input');
601      $out->html (q[<script> insertNavSections ('input') </script>]);
602    
603      $out->start_section (id => 'input-url', title => 'By URL',
604                           parent_id => 'input');
605      $out->start_tag ('form', action => './#result-summary',
606                       'accept-charset' => 'utf-8',
607                       method => 'get');
608      $out->start_tag ('input', type => 'hidden', name => '_charset_');
609    
610      $out->start_tag ('p');
611      $out->start_tag ('label');
612      $out->nl_text ('URL');
613      $out->text (': ');
614      $out->start_tag ('input',
615                       name => 'uri',
616                       type => 'url',
617                       value => $decode->(scalar $cgi->get_parameter ('uri')));
618      $out->end_tag ('label');
619    
620      $out->start_tag ('p');
621      $out->start_tag ('button', type => 'submit');
622      $out->nl_text ('Check');
623      $out->end_tag ('button');
624    
625      $options->('url');
626    
627      $out->end_tag ('form');
628      $out->end_section;
629    
630      ## TODO: File upload
631    
632      $out->start_section (id => 'input-text', title => 'By direct input',
633                           parent_id => 'input');
634      $out->start_tag ('form', action => './#result-summary',
635                       'accept-charset' => 'utf-8',
636                       method => 'post');
637      $out->start_tag ('input', type => 'hidden', name => '_charset_');
638    
639      $out->start_tag ('p');
640      $out->start_tag ('label');
641      $out->nl_text ('Document source to check');
642      $out->text (': ');
643      $out->start_tag ('br');
644      $out->start_tag ('textarea',
645                       name => 's');
646      my $s = $decode->($cgi->get_parameter ('s'));
647      $out->html ($htescape_value->($s)) if defined $s;
648      $out->end_tag ('textarea');
649      $out->end_tag ('label');
650    
651      $out->start_tag ('p');
652      $out->start_tag ('button', type => 'submit',
653                       onclick => 'form.method = form.s.value.length > 512 ? "post" : "get"');
654      $out->nl_text ('Check');
655      $out->end_tag ('button');
656    
657      $options->('text');
658    
659      $out->end_tag ('form');
660      $out->end_section;
661    
662      $out->script (q[
663        if (!document.webhaccNavigated &&
664            document.getElementsByTagName ('textarea')[0].value.length > 0) {
665          showTab ('input-text');
666          document.webhaccNavigated = false;
667        }
668      ]);
669    
670      $out->end_section;
671    } # generate_input_section
672    
673  sub encode_url_component ($$) {  sub encode_url_component ($$) {
674    shift;    shift;
675    require Encode;    require Encode;

Legend:
Removed from v.1.7  
changed lines
  Added in v.1.17

admin@suikawiki.org
ViewVC Help
Powered by ViewVC 1.1.24