]> git.deb.at Git - deb/packages.git/blob - lib/Packages/HTML.pm
Add basic l10n support.
[deb/packages.git] / lib / Packages / HTML.pm
1 package Packages::HTML;
2
3 use strict;
4 use warnings;
5
6 use Exporter;
7 use URI::Escape;
8 use HTML::Entities;
9 use Locale::gettext;
10
11 use Packages::CGI ();
12 use Packages::Search qw( read_entry_simple );
13 use Packages::Config qw( :all );
14
15 #use Packages::Util;
16 use Packages::I18N::Locale;
17 use Packages::I18N::Languages;
18 use Packages::I18N::LanguageNames;
19 #use Generated::Strings qw( gettext dgettext );
20
21 our @ISA = qw( Exporter );
22 our @EXPORT = qw( header title trailer file_changed time_stamp
23                   read_md5_hash write_md5_hash simple_menu
24                   ds_begin ds_item ds_end title marker pdesc
25                   pdeplegend pkg_list pmoreinfo print_deps print_src_deps );
26
27 our $CHANGELOG_URL = '/changelogs';
28
29 sub img {
30     my ( $root, $url, $src, $alt, %attr ) = @_; 
31     my @attr;
32
33     foreach my $a ( keys %attr ) {
34         push @attr, "$a=\"$attr{$a}\"";
35     }
36
37     return "<a href=\"$root$url\"><img src=\"$root$src\" alt=\"$alt\" @attr></a>";
38 }
39
40 sub simple_menu {
41     my $str = "";
42     foreach my $entry (@_) {
43         $str .= "[&nbsp;$entry->[0] <a title=\"$entry->[1]\" href=\"$entry->[2]\">$entry->[3]</a>&nbsp;]\n";
44     }
45     return $str;
46 }
47
48 sub title {
49     return "<h1>$_[0]</h1>\n";
50 }
51
52 sub marker {
53     return "[<strong class=\"pmarker\">$_[0]</strong>]";
54 }
55
56 sub pdesc {
57     my ( $short_desc, $long_desc ) = @_;
58     my $str = "";
59
60     $str .= "<div id=\"pdesc\">\n";
61     $str .= "<h2>$short_desc</h2>\n";
62
63     $str .= "<p>$long_desc\n";
64     $str .= "</div> <!-- end pdesc -->\n";
65
66     return $str;
67 }
68
69 sub pdeplegend {
70     my $str = "<table id=\"pdeplegend\" summary=\"legend\"><tr>\n";
71
72     foreach my $entry (@_) {
73         $str .= "<td><img src=\"$ROOT/Pics/$entry->[0].gif\" alt=\"[$entry->[0]]\" width=\"16\" height=\"16\">= $entry->[1]</td>";
74     }
75
76     $str .= "\n</tr></table>\n";
77     return $str;
78 }
79
80 sub pkg_list {
81     my ( $packages, $opts, $pkgs, $lang ) = @_;
82     my $suite = $opts->{suite}[0];
83
84     my $str = "";
85     foreach my $p ( @$pkgs ) {
86
87         my $short_desc = (read_entry_simple( $packages, $p, $opts->{h_archives}, $suite))->[-1];
88
89         if ( $short_desc ) {
90             $str .= "<dt><a href=\"$ROOT/$suite/$p\">$p</a></dt>\n".
91                     "\t<dd>$short_desc</dd>\n";
92         } else {
93             $str .= "<dt>$p</dt>\n\t<dd>"._g("Not available")."</dd>\n";
94         }
95     }
96     if ($str) {
97         $str = "<dl>$str</dl>\n";
98     }
99
100     return $str;
101 }
102
103 sub pmoreinfo {
104     my %info = @_;
105     
106     my $name = $info{name} or return;
107     my $env = $info{env} or return;
108     my $opts = $info{opts} or return;
109     my $page = $info{data} or return;
110     my $is_source = $info{is_source};
111     my $suite = $opts->{suite}[0];
112
113     my $str = "<div id=\"pmoreinfo\">";
114     $str .= sprintf( "<h2>"._g( "More Information on %s" )."</h2>",
115                      $name );
116     
117     if ($info{bugreports}) {
118         my $bug_url = $is_source ? $SRC_BUG_URL : $BUG_URL; 
119         $str .= "<p>\n".sprintf( _g( "Check for <a href=\"%s\">Bug Reports</a> about %s." )."<br>\n",
120                          $bug_url.$name, $name );
121     }
122         
123     my $source = $page->get_src( 'package' );
124     my $source_version = $page->get_src( 'version' );
125     my $src_dir = $page->get_src('directory');
126     if ($info{sourcedownload}) {
127         my $files = $page->get_src( 'files' );
128         my $path = (@{$opts->{archive}} >1) ?
129             $suite :
130             "$suite/$opts->{archive}[0]";
131         $str .= _g( "Source Package:" );
132         $str .= " <a href=\"$ROOT/$path/source/$source\">$source</a>, ".
133             _g( "Download" ).":\n";
134
135         unless (defined($files) and @$files) {
136             $str .= _g( "Not found" );
137         } else {
138             foreach( @$files ) {
139                 my ($src_file_md5, $src_file_size, $src_file_name) = split /\s/o, $_;
140                 for ($page->get_newest('archive')) {
141                     /security/o && do {
142                         $str .= "<a href=\"$env->{security}/$src_dir/$src_file_name\">["; last };
143                     /volatile/o && do {
144                         $str .= "<a href=\"$env->{volatile}/$src_dir/$src_file_name\">["; last };
145                     /backports/o && do {
146                         $str .= "<a href=\"$env->{backports}/$src_dir/$src_file_name\">["; last };
147                     /non-us/io && do {
148                         $str .= "<a href=\"$env->{nonus_site}/$src_dir/$src_file_name\">["; last };
149                     $str .= "<a href=\"$env->{us}/$src_dir/$src_file_name\">[";
150                 }
151                 if ($src_file_name =~ /dsc$/) {
152                     $str .= "dsc";
153                 } else {
154                     $str .= $src_file_name;
155                 }
156                 $str .= "]</a>\n";
157             }
158         }
159 #           $package_page .= sprintf( _g( " (These sources are for version %s)\n" ), $src_version )
160 #               if ($src_version ne $version) && !$src_version_given_in_control;
161     }
162
163     if ($info{changesandcopy}) {
164         if ( $src_dir ) {
165             (my $src_basename = $source_version) =~ s,^\d+:,,; # strip epoche
166             $src_basename = "${source}_$src_basename";
167             $src_dir =~ s,pool/updates,pool,o;
168             $src_dir =~ s,pool/non-US,pool,o;
169             $str .= "<br>".sprintf( _g( 'View the <a href="%s">Debian changelog</a>' ),
170                                     "$CHANGELOG_URL/$src_dir/$src_basename/changelog" )."<br>\n";
171             my $copyright_url = "$CHANGELOG_URL/$src_dir/$src_basename/";
172             $copyright_url .= ( $is_source ? 'copyright' : "$name.copyright" );
173
174             $str .= sprintf( _g( 'View the <a href="%s">copyright file</a>' ),
175                              $copyright_url )."</p>";
176         }
177    }
178
179     if ($info{maintainers}) {
180         my $uploaders = $page->get_src( 'uploaders' );
181         if ($uploaders && @$uploaders) {
182             foreach (@$uploaders) {
183                 $_->[0] = encode_entities( $_->[0], '&<>' );
184             }
185             my ($maint_name, $maint_mail) = @{shift @$uploaders}; 
186             unless (@$uploaders) {
187                 $str .= "<p>\n".sprintf( _g( "%s is responsible for this Debian package." ).
188                                          "\n",
189                                          "<a href=\"mailto:$maint_mail\">$maint_name</a>" 
190                                          );
191             } else {
192                 my $up_str = "<a href=\"mailto:$maint_mail\">$maint_name</a>";
193                 my @uploaders_str;
194                 foreach (@$uploaders) {
195                     push @uploaders_str, "<a href=\"mailto:$_->[1]\">$_->[0]</a>";
196                 }
197                 my $last_up = pop @uploaders_str;
198                 $up_str .= ", ".join ", ", @uploaders_str if @uploaders_str;
199                 $up_str .= sprintf( _g( " and %s are responsible for this Debian package." ), $last_up );
200                 $str .= "<p>\n$up_str ";
201             }
202         }
203
204         $str .= sprintf( _g( "See the <a href=\"%s\">developer information for %s</a>." )."</p>", $QA_URL.$source, $name ) if $source;
205     }
206
207     if ($info{search}) {
208         my $encodedname = uri_escape( $name );
209         my $search_url = $is_source ? "$ROOT/source" : $ROOT;
210         $str .= "<p>".sprintf( _g( "Search for <a href=\"%s\">other versions of %s</a>" ),
211             "$search_url/$encodedname", $name )."</p>\n";
212     }
213
214     $str .= "</div> <!-- end pmoreinfo -->\n";
215     return $str;
216 }
217
218 sub dep_item {
219     my ( $link, $name, $info, $desc ) = @_;
220     my $post_link = '';
221     if ($link) {
222         $link = "<a href=\"$link\">";
223         $post_link = '</a>';
224     } else {
225         $link = '';
226     }
227     if ($info) {
228         $info = " $info";
229     } else {
230         $info = '';
231     }
232     if ($desc) {
233         $desc = "</dt><dd>$desc</dd>";
234     } else {
235         $desc = '</dt>';
236     }
237
238     return "$link$name$post_link$info$desc";
239 } # end dep_item
240
241 sub print_deps {
242     my ( $packages, $opts, $pkg, $relations, $type) = @_;
243     my %dep_type = ('depends' => 'dep', 'recommends' => 'rec', 
244                     'suggests' => 'sug', 'build-depends' => 'adep',
245                     'build-depends-indep' => 'idep' );
246     my $res = "<ul class=\"ul$dep_type{$type}\">\n";
247     my $first = 1;
248     my $suite = $opts->{suite}[0];
249     my $one_archive = @{$opts->{archive}} > 1 ? '': $opts->{archive}[0];
250
251 #    use Data::Dumper;
252 #    debug( "print_deps called:\n".Dumper( $pkg, $relations, \$type ), 3 );
253
254     foreach my $rel (@$relations) {
255         my $is_old_pkgs = $rel->[0];
256         my @res_pkgs = ();
257
258         if ($is_old_pkgs)  {
259             $res .= "<dt>";
260         } else {
261             if ($first) {
262                 $res .= "<li>";
263                 $first = 0;
264             } else {
265                 $res .= "</dl></li>\n<li>";
266             }
267             $res .= "<dl><dt><img class=\"hidecss\" src=\"$ROOT/Pics/$dep_type{$type}.gif\" alt=\"[$dep_type{$type}]\"> ";
268         }
269
270         foreach my $rel_alt ( @$rel ) {
271             next unless ref($rel_alt);
272             my ( $p_name, $pkg_version, $arch_neg,
273                  $arch_str, $subsection, $available ) = @$rel_alt;
274
275             if ($arch_str ||= '') {
276                 if ($arch_neg) {
277                     $arch_str = " ["._g("not")." $arch_str]";
278                 } else {
279                     $arch_str = " [$arch_str]";
280                 }
281             }
282             $pkg_version = "($pkg_version)" if $pkg_version ||= '';
283             
284             my @results;
285             my %entries;
286             my $entry = $entries{$p_name} ||
287                 read_entry_simple( $packages, $p_name, $opts->{h_archives}, $suite);
288             my $short_desc = $entry->[-1];
289             my $arch = $entry->[2];
290             my $archive = $entry->[0];
291             if ( $short_desc ) {
292                 my $path = $one_archive eq $archive ? "$suite/$archive" :
293                     $suite;
294                 if ( $is_old_pkgs ) {
295                     push @res_pkgs, dep_item( "$ROOT/$path/$p_name",
296                                               $p_name, "$pkg_version$arch_str" );
297                 } elsif ($arch eq 'virtual') {
298                     my @provided_by = split /\s/, $short_desc;
299                     $short_desc = "virtual package provided by ";
300                     if (@provided_by < 10) {
301                         $short_desc .= join( ', ',map { "<a href=\"$ROOT/$path/$_\">$_</a>" } @provided_by);
302                     } else {
303                         $short_desc .= scalar(@provided_by)." packages";
304                     }
305                     push @res_pkgs, dep_item( "$ROOT/$path/$p_name",
306                                               $p_name, "$pkg_version$arch_str", $short_desc );
307                 } else {
308                     $entries{$p_name} ||= $entry;
309                     $short_desc = encode_entities( $short_desc, "<>&\"" );
310                     push @res_pkgs, dep_item( "$ROOT/$path/$p_name",
311                                               $p_name, "$pkg_version$arch_str", $short_desc );
312                 }
313             } elsif ( $is_old_pkgs ) {
314                 push @res_pkgs, dep_item( undef, $p_name, "$pkg_version$arch_str" );
315             } else {
316                 my $short_desc = _g( "Package not available" );
317                 push @res_pkgs, dep_item( undef, $p_name, "$pkg_version$arch_str", $short_desc );
318             }
319             
320         }
321         
322         $res .= "\n".join( "<dt>"._g( "or" )." ", @res_pkgs )."\n";
323     }
324     if (@$relations) {
325         $res .= "</dl></li>\n";
326         $res .= "</ul>\n";
327     } else {
328         $res = "";
329     }
330     return $res;
331 } # end print_deps
332
333 my $ds_begin = '<dl>';
334 my $ds_item_desc  = '<dt>';
335 my $ds_item = ':</dt><dd>';
336 my $ds_item_end = '</dd>';
337 my $ds_end = '</dl>';
338 #           my $ds_begin = '<table><tbody>';
339 #           my $ds_item_desc  = '<tr><td>';
340 #           my $ds_item = '</td><td>';
341 #           my $ds_item_end = '</td></tr>';
342 #           my $ds_end = '</tbody></table>';
343
344 sub ds_begin {
345     return $ds_begin;
346 }
347 sub ds_item {
348     return "$ds_item_desc$_[0]$ds_item$_[1]$ds_item_end\n";
349 }
350 sub ds_end {
351     return $ds_end;
352 }
353
354 sub header {
355     my (%params) = @_;
356
357     my $DESC_LINE;
358     if (defined $params{desc}) {
359         $DESC_LINE = "<meta name=\"Description\" content=\"$params{desc}\">";
360     }
361     else {
362         $DESC_LINE = '';
363     }
364
365     my $title_keywords = $params{title_keywords} || $params{title} || '';
366     my $title_tag = $params{title_tag} || $params{title} || '';
367     my $title_in_header = $params{page_title} || $params{title} || '';
368     my $page_title = $params{page_title} || $params{title} || '';
369     my $meta = $params{meta} || '';
370
371     my $search_in_header = '';
372     $params{print_search_field} ||= "";
373     if ($params{print_search_field} eq 'packages') {
374         my %values = %{$params{search_field_values}};
375         my %checked_searchon = ( names => "",
376                                  all => "",
377                                  sourcenames => "",
378                                  contents => "");
379         $checked_searchon{$values{searchon}} = "checked=\"checked\"";
380         $checked_searchon{names} = "checked=\"checked\""
381                 if $values{searchon} eq 'default';
382         $search_in_header = <<MENU;
383 <form method="GET" action="$SEARCH_URL">
384 <div id="hpacketsearch">
385 <input type="hidden" name="debug" value="$values{debug}">
386 <input type="hidden" name="suite" value="$values{suite}">
387 <input type="hidden" name="exact" value="$values{exact}">
388 <input type="hidden" name="arch" value="$values{arch}">
389 <input type="hidden" name="section" value="$values{section}">
390 <input type="text" size="30" name="keywords" value="$values{keywords}" id="kw">
391 <input type="submit" value="%s">
392 <span style="font-size: 60%%"><a href="$SEARCH_PAGE#search_packages">%s</a></span>
393 <br>
394 <div style="font-size: 80%%">%s
395 <input type="radio" name="searchon" value="names" id="onlynames" $checked_searchon{names}>
396 <label for="onlynames">%s</label>&nbsp;&nbsp;
397 <input type="radio" name="searchon" value="all" id="descs" $checked_searchon{all}>
398 <label for="descs">%s</label>
399 <br>
400 <input type="radio" name="searchon" value="sourcenames" id="src" $checked_searchon{sourcenames}>
401 <label for="src">%s</label>
402 <input type="radio" name="searchon" value="contents" id="conts" $checked_searchon{contents}>
403 <label for="conts">%s</label>
404 </div>
405 </div> <!-- end hpacketsearch -->
406 </form>
407 MENU
408 ;
409         $search_in_header = sprintf( $search_in_header,
410                                      _g( 'Search' ),
411                                      _g( 'Full options' ),
412                                      _g( 'Search on:'),
413                                      _g( 'Package Names' ),
414                                      _g( 'Descriptions' ),
415                                      _g( 'Source package names' ),
416                                      _g( 'Package contents' ));
417 #     } elsif ($params{print_search_field} eq 'contents') {
418 #       my %values = %{$params{search_field_values}};
419 #       my %checked_searchmode = ( searchfiles => "",
420 #                                  searchfilesanddirs => "",
421 #                                  searchword => "",
422 #                                  filelist => "", );
423 #       $checked_searchmode{$values{searchmode}} = "checked=\"checked\"";
424 #       $search_in_header = <<MENU;
425 # <form method="GET" action="$CONTENTS_SEARCH_CGI">
426 # <div id="hpacketsearch">
427 # <input type="hidden" name="debug" value="$values{debug}" />
428 # <input type="hidden" name="version" value="$values{version}" />
429 # <input type="hidden" name="arch" value="$values{arch}" />
430 # <input type="hidden" name="case" value="$values{case}" />
431 # <input type="text" size="30" name="word" id="keyword" value="$values{keyword}">&nbsp;
432 # <input type="submit" value="Search">
433 # <span style="font-size: 60%"><a href="$SEARCH_PAGE#search_contents">Full options</a></span>
434 # <br>
435 # <div style="font-size: 80%">Display:
436 # <input type=radio name="searchmode" value="searchfiles" id="searchfiles" $checked_searchmode{searchfiles}>
437 # <label for="searchfiles">files</label>
438 # <input type=radio name="searchmode" value="searchfilesanddirs" id="searchfilesanddirs" $checked_searchmode{searchfilesanddirs}>
439 # <label for="searchfilesanddirs">files &amp; directories</label>
440 # <br>
441 # <input type=radio name="searchmode" value="searchword" id="searchword" $checked_searchmode{searchword}>
442 # <label for="searchword">subword matching</label>
443 # <input type=radio name="searchmode" value="filelist" id="filelist" $checked_searchmode{filelist}>
444 # <label for="filelist">content list</label>
445 # </div>
446 # </div> <!-- end hpacketsearch -->
447 # </form>
448 # MENU
449 # ;
450     }
451
452     my $keywords = $params{keywords} || '';
453     my $KEYWORDS_LINE = "<meta name=\"Keywords\" content=\"debian, $keywords $title_keywords\">";
454     
455     my $LANG = $params{lang};
456     my $charset = get_charset($LANG);
457     my $txt = <<HEAD;
458 <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/strict.dtd">
459 <html lang="$LANG">
460 <head>
461 <title>Debian -- $title_tag</title>
462 <link rev="made" href="mailto:$WEBMASTER_MAIL">
463 <meta http-equiv="Content-Type" content="text/html; charset=$charset">
464 <meta name="Author" content="Debian Webmaster, $WEBMASTER_MAIL">
465 $KEYWORDS_LINE
466 $DESC_LINE
467 $meta
468 <link href="$ROOT/debian.css" rel="stylesheet" type="text/css" media="all">
469 <link href="$ROOT/packages.css" rel="stylesheet" type="text/css" media="all">
470 </head>
471 <body>
472 <div id="header">
473    <div id="upperheader">
474    <div id="logo">
475   <a href="$HOME/"><img src="$HOME/logos/openlogo-nd-50.png" alt="" /></a>
476 HEAD
477 ;
478
479     $txt .= img( "$HOME/", "", "Pics/debian.png", _g( "Debian Project" ),
480                  width => 179, height => 61 );
481     $txt .= <<HEADEND;
482
483 </div> <!-- end logo -->
484 HEADEND
485 ;
486
487     $txt .= <<NAVBEGIN;
488 $search_in_header
489 </div> <!-- end upperheader -->
490
491 NAVBEGIN
492 ;
493     $txt .= "<p class=\"hidecss\"><a href=\"\#inner\">" . _g("Skip Site Navigation")."</a></p>\n";
494     $txt .= "<div id=\"navbar\">\n<ul>".
495         "<li><a href=\"$HOME/intro/about\">"._g( "About&nbsp;Debian" )."</a></li>\n".
496         "<li><a href=\"$HOME/News/\">"._g( "News" )."</a></li>\n".
497         "<li><a href=\"$HOME/distrib/\">"._g( "Getting&nbsp;Debian" )."</a></li>\n".
498         "<li><a href=\"$HOME/support\">"._g( "Support" )."</a></li>\n".
499         "<li><a href=\"$HOME/devel/\">"._g( "Development" )."</a></li>\n".
500         "<li><a href=\"$HOME/sitemap\">"._g( "Site map" )."</a></li>\n".
501         "<li><a href=\"http://search.debian.org/\">"._g( "Search" )."</a></li>\n";
502     $txt .= "</ul>\n";
503     $txt .= <<ENDNAV;
504 </div> <!-- end navbar -->
505 </div> <!-- end header -->
506 ENDNAV
507 ;
508     $txt .= <<BEGINCONTENT;
509 <div id="outer">
510 <div id="inner">
511
512 BEGINCONTENT
513 ;
514     if ($params{print_title}) {
515         $txt .= "<h1>$page_title</h1>\n";
516     }
517
518     return $txt;
519 }
520
521 sub trailer {
522     my ($ROOT, $NAME, $LANG, @USED_LANGS) = @_;
523     my $txt = "</div> <!-- end inner -->\n<div id=\"footer\">\n";
524     my $langs = languages( $NAME, $LANG, @USED_LANGS );
525     my $bl_class = $langs ? ' class="bordertop"' : "";
526     $txt .=
527         $langs.
528         "\n<hr class=\"hidecss\">\n" .
529         "<p$bl_class>".
530         sprintf( _g( "Back to: <a href=\"%s/\">Debian Project homepage</a> || <a href=\"%s/\">Packages search page</a>" ), $HOME, $ROOT ).
531         "</p>\n<hr class=\"hidecss\">\n".
532         "<div id=\"fineprint\" class=\"bordertop\"><p>".
533         sprintf( _g( "To report a problem with the web site, e-mail <a href=\"mailto:%s\">%s</a>. For other contact information, see the Debian <a href=\"%s/contact\">contact page</a>." ), $CONTACT_MAIL, $CONTACT_MAIL, $HOME).
534         "</p>\n".
535         "<p>". _g( "Last Modified: " ). "LAST_MODIFIED_DATE".
536         "<br>\n".
537         sprintf( _g( "Copyright &copy; 1997-2005 <a href=\"http://www.spi-inc.org\">SPI</a>; See <a href=\"%s/license\">license terms</a>." ), "$HOME/" )."<br>\n".
538         _g( "Debian is a registered trademark of Software in the Public Interest, Inc." ).
539         "</div> <!-- end fineprint -->\n".
540         "</div> <!-- end footer -->\n".
541         "</div> <!-- end outer -->\n".
542         "</body>\n</html>\n";
543
544     return $txt;
545 }
546
547 sub languages {
548     my ( $name, $lang, @used_langs ) = @_;
549     
550     my $str = "";
551     
552     if (@used_langs) {
553         $str .= "<hr class=\"hidecss\">\n";
554         $str .= "<!--UdmComment-->\n<p>\n";
555         $str .= _g( "This page is also available in the following languages:\n" );
556         $str .= "</p><p class=\"navpara\">\n";
557         
558         my @printed_langs = ();
559         foreach (@used_langs) {
560             next if $_ eq $lang; # Never print the current language
561             unless (get_selfname($_)) { warn "missing language $_"; next } #DEBUG
562             push @printed_langs, $_;
563         }
564         return "" unless scalar @printed_langs;
565         # Sort on uppercase to work with languages which use lowercase initial
566         # letters.
567         foreach my $cur_lang (sort langcmp @printed_langs) {
568             my $tooltip = dgettext( "langs", get_language_name($cur_lang) );
569             $str .= "<a href=\"$name.$cur_lang.html\" title=\"$tooltip\" hreflang=\"$cur_lang\" lang=\"$cur_lang\" rel=\"alternate\">".get_selfname($cur_lang);
570             $str .= " (".get_transliteration($cur_lang).")" if defined get_transliteration($cur_lang);
571             $str .= "</a>\n";
572         }
573         $str .= "\n</p><p>\n";
574         $str .= sprintf( _g( "How to set <a href=\"%s\">the default document language</a>" ), $CN_HELP_URL )."</p>";
575         $str .= "\n<!--/UdmComment-->\n";
576     }
577     
578     return $str;
579 }
580
581 1;