/[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.14 by wakaba, Thu Aug 14 11:58:32 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 73  sub url ($$%) { Line 84  sub url ($$%) {
84    
85  sub start_tag ($$%) {  sub start_tag ($$%) {
86    my ($self, $tag_name, %opt) = @_;    my ($self, $tag_name, %opt) = @_;
87    $self->html ('<' . $htescape->($tag_name)); # escape for safety    $self->html ('<' . $htescape_value->($tag_name)); # escape for safety
88    if (exists $opt{id}) {    if (exists $opt{id}) {
89      my $id = $self->input->id_prefix . $opt{id};      my $id = $self->input->id_prefix . $opt{id};
90      $self->html (' id="' . $htescape->($id) . '"');      $self->html (' id="' . $htescape_value->($id) . '"');
91      delete $opt{id};      delete $opt{id};
92    }    }
93    for (keys %opt) {    # for safety    for (keys %opt) {    # for safety
94      $self->html (' ' . $htescape->($_) . '="' . $htescape->($opt{$_}) . '"');      $self->html (' ' . $htescape_value->($_) . '="' .
95                     $htescape_value->($opt{$_}) . '"');
96    }    }
97    $self->html ('>');    $self->html ('>');
98  } # start_tag  } # start_tag
99    
100  sub end_tag ($$) {  sub end_tag ($$) {
101    shift->html ('</' . $htescape->(shift) . '>');    shift->html ('</' . $htescape_value->(shift) . '>');
102  } # end_tag  } # end_tag
103    
104  sub start_section ($%) {  sub start_section ($%) {
105    my ($self, %opt) = @_;    my ($self, %opt) = @_;
106    
107      my $class = 'section';
108    if (defined $opt{role}) {    if (defined $opt{role}) {
109      if ($opt{role} eq 'parse-errors') {      if ($opt{role} eq 'parse-errors') {
110        $opt{id} ||= 'parse-errors';        $opt{id} ||= 'parse-errors';
111        $opt{title} ||= 'Parse Errors Section';        $opt{title} ||= 'Parse Errors Section';
112        $opt{short_title} ||= 'Parse Errors';        $opt{short_title} ||= 'Parse Errors';
113          $class .= ' errors';
114        delete $opt{role};        delete $opt{role};
115      } elsif ($opt{role} eq 'structure-errors') {      } elsif ($opt{role} eq 'structure-errors') {
116        $opt{id} ||= 'document-errors';        $opt{id} ||= 'document-errors';
117        $opt{title} ||= 'Structural Errors';        $opt{title} ||= 'Structural Errors';
118        $opt{short_title} ||= 'Struct. Errors';        $opt{short_title} ||= 'Struct. Errors';
119          $class .= ' errors';
120        delete $opt{role};        delete $opt{role};
121      } elsif ($opt{role} eq 'reformatted') {      } elsif ($opt{role} eq 'reformatted') {
122        $opt{id} ||= 'document-tree';        $opt{id} ||= 'document-tree';
123        $opt{title} ||= 'Reformatted Document Source';        $opt{title} ||= 'Reformatted Document Source';
124        $opt{short_title} ||= 'Reformatted';        $opt{short_title} ||= 'Reformatted';
125          $class .= ' dump';
126        delete $opt{role}        delete $opt{role}
127      } elsif ($opt{role} eq 'tree') {      } elsif ($opt{role} eq 'tree') {
128        $opt{id} ||= 'document-tree';        $opt{id} ||= 'document-tree';
129        $opt{title} ||= 'Document Tree';        $opt{title} ||= 'Document Tree';
130        $opt{short_title} ||= 'Tree';        $opt{short_title} ||= 'Tree';
131          $class .= ' dump';
132        delete $opt{role};        delete $opt{role};
133      } elsif ($opt{role} eq 'structure') {      } elsif ($opt{role} eq 'structure') {
134        $opt{id} ||= 'document-structure';        $opt{id} ||= 'document-structure';
135        $opt{title} ||= 'Document Structure';        $opt{title} ||= 'Document Structure';
136        $opt{short_title} ||= 'Structure';        $opt{short_title} ||= 'Structure';
137          $class .= ' dump';
138          delete $opt{role};
139        } elsif ($opt{role} eq 'subdoc') {
140          $class .= ' subdoc';
141          delete $opt{role};
142        } elsif ($opt{role} eq 'source') {
143          $opt{id} ||= 'source-string';
144          $opt{title} ||= 'Document Source';
145          $opt{short_title} ||= 'Source';
146          $class .= ' source';
147        delete $opt{role};        delete $opt{role};
148      }      }
149    }    }
150    
151    $self->{section_rank}++;    $self->{section_rank}++;
152    $self->html ('<div class=section');    $self->html (qq[<div class="$class"]);
153    if (defined $opt{id}) {    if (defined $opt{id}) {
154      my $id = $self->input->id_prefix . $opt{id};      my $prefix = $self->input->id_prefix;
155      $self->html (' id="' . $htescape->($id) . '"');      $opt{parent_id} ||= $prefix;
156      push @{$self->{nav}},      my $id = $prefix . $opt{id};
157          [$id => $opt{short_title} || $opt{title} => $opt{text}]      $self->html (' id="' . $htescape->($id) . '">');
158          if $self->{section_rank} == 2;      if ($self->{section_rank} == 2 or length $opt{parent_id}) {
159          my $st = $opt{short_title} || $opt{title};
160          push @{$self->{nav}},
161              [$id => $st => $opt{text}];
162          
163          $self->start_tag ('script');
164          $self->html (qq[ addSectionLink ('$id', ']);
165          $self->nl_text ($st, text => $opt{text});
166          if (defined $opt{parent_id}) {
167            $self->html (q[', '] . $opt{parent_id});
168          }
169          $self->html (q[') ]);
170          $self->end_tag ('script');
171        }
172      } else {
173        $self->html ('>');
174    }    }
175    my $section_rank = $self->{section_rank};    my $section_rank = $self->{section_rank};
176    $section_rank = 6 if $section_rank > 6;    $section_rank = 6 if $section_rank > 6;
177    $self->html ('><h' . $section_rank . '>');    $self->html ('<h' . $section_rank . '>');
178    $self->nl_text ($opt{title}, text => $opt{text});    $self->nl_text ($opt{title}, text => $opt{text});
179    $self->html ('</h' . $section_rank . '>');    $self->html ('</h' . $section_rank . '>');
180  } # start_section  } # start_section
# Line 187  sub add_source_to_parse_error_list ($$) Line 229  sub add_source_to_parse_error_list ($$)
229    my $self = shift;    my $self = shift;
230    
231    $self->script (q[addSourceToParseErrorList ('] . $self->input->id_prefix .    $self->script (q[addSourceToParseErrorList ('] . $self->input->id_prefix .
232                   q[', '] . shift . q[')]);                   q[', '] . shift () . q[')]);
233  } # add_source_to_parse_error_list  } # add_source_to_parse_error_list
234    
235  sub start_code_block ($) {  sub start_code_block ($) {
# Line 218  sub dt ($$;%) { Line 260  sub dt ($$;%) {
260    $self->nl_text ($content, text => $opt{text});    $self->nl_text ($content, text => $opt{text});
261  } # dt  } # dt
262    
263    sub select ($$%) {
264      my ($self, $options, %opt) = @_;
265    
266      my $selected = $opt{selected};
267      delete $opt{selected};
268    
269      $self->start_tag ('select', %opt);
270      
271      my @options = @$options;
272      while (@options) {
273        my $opt = shift @options;
274        if ($opt->{options}) {
275          $self->html ('<optgroup label="');
276          $self->nl_text ($opt->{label});
277          $self->html ('">');
278          unshift @options, @{$opt->{options}}, {end_options => 1};
279        } elsif ($opt->{end_options}) {
280          $self->end_tag ('optgroup');
281        } else {
282          $self->start_tag ('option', value => $opt->{value},
283                            ((defined $selected and $opt->{value} eq $selected)
284                                 ? (selected => '') : ()));
285          $self->nl_text (defined $opt->{label} ? $opt->{label} : $opt->{value});
286        }
287      }
288    
289      $self->end_tag ('select');
290    } # select
291    
292  sub link ($$%) {  sub link ($$%) {
293    my ($self, $content, %opt) = @_;    my ($self, $content, %opt) = @_;
294    $self->start_tag ('a', %opt, href => $opt{url});    $self->start_tag ('a', %opt, href => $opt{url});
# Line 265  my $get_node_path = sub ($) { Line 336  my $get_node_path = sub ($) {
336    return join '/', @r;    return join '/', @r;
337  }; # $get_node_path  }; # $get_node_path
338    
339    my $get_object_path = sub ($) {
340      my $node = shift;
341      my @r;
342      while (defined $node) {
343        my $ref = ref $node;
344        $ref =~ /([^:]+)$/;
345        my $rs = $1;
346        my $node_name = $node->node_name;
347        if (defined $node_name) {
348          $rs .= ' <code>' . $htescape->($node_name) . '</code>';
349        }
350        $node = undef;
351        unshift @r, $rs;
352      }
353      return join '/', @r;
354    }; # $get_object_path
355    
356  sub node_link ($$) {  sub node_link ($$) {
357    my ($self, $node) = @_;    my ($self, $node) = @_;
358    $self->xref ($get_node_path->($node), target => 'node-' . refaddr $node);    if ($node->isa ('Message::IF::Node')) {
359        $self->xref ($get_node_path->($node), target => 'node-' . refaddr $node);
360      } else {
361        $self->html ($get_object_path->($node));
362      }
363  } # node_link  } # node_link
364    
365  {  {
# Line 367  sub html_header ($) { Line 459  sub html_header ($) {
459    $self->nl_text (q[WebHACC:Title]);    $self->nl_text (q[WebHACC:Title]);
460    $self->html (q[</title>    $self->html (q[</title>
461  <link rel="stylesheet" href="../cc-style.css" type="text/css">  <link rel="stylesheet" href="../cc-style.css" type="text/css">
462    <script src="../cc-script.js"></script>
463  </head>  </head>
464  <body>  <body onclick=" return onbodyclick (event) " onload=" onbodyload () ">
465  <h1>]);  <h1>]);
466    $self->nl_text (q[WebHACC:Heading]);    $self->nl_text (q[WebHACC:Heading]);
467    $self->html ('</h1>');    $self->html (q[</h1><script> insertNavSections () </script>]);
468  } # html_header  } # html_header
469    
470    sub generate_input_section ($$) {
471      my ($out, $cgi) = @_;
472    
473      my $options = sub ($) {
474        my $context = shift;
475    
476        $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'">]);
477        $out->nl_text (q[Options]);
478        $out->start_tag ('div');
479    
480        if ($context eq 'url') {
481          $out->start_tag ('p');
482          $out->start_tag ('label');
483          $out->start_tag ('input', type => 'checkbox', name => 'error-page',
484                           value => 1,
485                           ($cgi->get_parameter ('error-page')
486                                ? (checked => '') : ()));
487          $out->nl_text ('Check error page');
488          $out->end_tag ('label');
489        }
490    
491        $out->start_tag ('p');
492        $out->start_tag ('label');
493        $out->nl_text (q[Content type]);
494        $out->text (': ');
495        $out->select ([
496          {value => '', label => 'As specified'},
497          {value => 'application/atom+xml'},
498          {value => 'application/xhtml+xml'},
499          {value => 'application/xml'},
500          {value => 'text/html'},
501          {value => 'text/xml'},
502          {value => 'text/css'},
503          {value => 'text/cache-manifest'},
504          {value => 'text/x-webidl'},
505        ], name => 'i', selected => scalar $cgi->get_parameter ('i'));
506        $out->end_tag ('label');
507    
508        if ($context ne 'text') {
509          $out->start_tag ('p');
510          $out->start_tag ('label');
511          $out->nl_text (q[Charset]);
512          $out->text (q[: ]);
513          $out->select ([
514            {value => '', label => 'As specified'},
515            {label => 'Japanese charsets', options => [
516              {value => 'Windows-31J'},
517              {value => 'Shift_JIS'},
518              {value => 'EUC-JP'},
519              {value => 'ISO-2022-JP'},
520            ]},
521            {label => 'European charsets', options => [
522              {value => 'Windows-1252'},
523              {value => 'ISO-8859-1'},
524              {value => 'US-ASCII'},
525            ]},
526            {label => 'Asian charsets', options => [
527              {value => 'Windows-874'},
528              {value => 'ISO-8859-11'},
529              {value => 'TIS-620'},
530            ]},
531            {label => 'Unicode charsets', options => [
532              {value => 'UTF-8'},
533              {value => 'UTF-8n'},
534            ]},
535          ], name => 'charset',
536          selected => scalar $cgi->get_parameter ('charset'));
537          $out->end_tag ('label');
538        }
539    
540        if ($context eq 'text') {
541          $out->start_tag ('p');
542          $out->start_tag ('label');
543          $out->nl_text ('Setting innerHTML');
544          $out->text (': ');
545          $out->start_tag ('input', name => 'e',
546                           value => scalar $cgi->get_parameter ('e'));
547          $out->end_tag ('label');
548        }
549    
550        $out->html (q[</div></div>]);
551      }; # $options
552    
553      $out->start_section (id => 'input', title => 'Input');
554      $out->html (q[<script> insertNavSections ('input') </script>]);
555    
556      $out->start_section (id => 'input-url', title => 'By URL',
557                           parent_id => 'input');
558      $out->start_tag ('form', action => './#result-summary',
559                       'accept-charset' => 'utf-8',
560                       method => 'get');
561      $out->start_tag ('input', type => 'hidden', name => '_charset_');
562    
563      $out->start_tag ('p');
564      $out->start_tag ('label');
565      $out->nl_text ('URL');
566      $out->text (': ');
567      $out->start_tag ('input',
568                       name => 'uri',
569                       type => 'url',
570                       value => $cgi->get_parameter ('uri'));
571      $out->end_tag ('label');
572    
573      $out->start_tag ('p');
574      $out->start_tag ('button', type => 'submit');
575      $out->nl_text ('Check');
576      $out->end_tag ('button');
577    
578      $options->('url');
579    
580      $out->end_tag ('form');
581      $out->end_section;
582    
583      $out->end_tag ('fieldset');
584    
585      ## TODO: File upload
586    
587      $out->start_section (id => 'input-text', title => 'By direct input',
588                           parent_id => 'input');
589      $out->start_tag ('form', action => './#result-summary',
590                       'accept-charset' => 'utf-8',
591                       method => 'post');
592      $out->start_tag ('input', type => 'hidden', name => '_charset_');
593    
594      $out->start_tag ('p');
595      $out->start_tag ('label');
596      $out->nl_text ('Document source to check');
597      $out->text (': ');
598      $out->start_tag ('br');
599      $out->start_tag ('textarea',
600                       name => 's');
601      my $s = $cgi->get_parameter ('s');
602      $out->html ($htescape_value->($s)) if defined $s;
603      $out->end_tag ('textarea');
604      $out->end_tag ('label');
605    
606      $out->start_tag ('p');
607      $out->start_tag ('button', type => 'submit',
608                       onclick => 'form.method = form.s.value.length > 512 ? "post" : "get"');
609      $out->nl_text ('Check');
610      $out->end_tag ('button');
611    
612      $options->('text');
613    
614      $out->end_tag ('form');
615      $out->end_section;
616    
617      $out->script (q[
618        if (!document.webhaccNavigated &&
619            document.getElementsByTagName ('textarea')[0].value.length > 0) {
620          showTab ('input-text');
621          document.webhaccNavigated = false;
622        }
623      ]);
624    
625      $out->end_section;
626    } # generate_input_section
627    
628  sub encode_url_component ($$) {  sub encode_url_component ($$) {
629    shift;    shift;
630    require Encode;    require Encode;

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

admin@suikawiki.org
ViewVC Help
Powered by ViewVC 1.1.24