1 # bsta_lib.pm is generated from bsta_lib.1.pm
5 # Copyright (C) 2016, 2017, 2019, 2020, 2022, 2023, 2024 Balthasar Szczepański
7 # This program is free software: you can redistribute it and/or modify
8 # it under the terms of the GNU Affero General Public License as
9 # published by the Free Software Foundation, either version 3 of the
10 # License, or (at your option) any later version.
12 # This program is distributed in the hope that it will be useful,
13 # but WITHOUT ANY WARRANTY; without even the implied warranty of
14 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 # GNU Affero General Public License for more details.
17 # You should have received a copy of the GNU Affero General Public License
18 # along with this program. If not, see <http://www.gnu.org/licenses/>.
22 # TODO: BB & INFO indent
30 use Encode ('encode', 'decode');
32 use vars qw($VERSION @ISA @EXPORT @EXPORT_OK %EXPORT_TAGS);
34 ###PERL_EXPORT_VERSION: our $VERSION = 'x.x.x';
35 our @ISA = qw(Exporter);
38 'STATE', 'TEXT_MODE', 'INTF_STATE', 'CHAT_STATE', 'CHAT_ACTION',
39 'get_remote_addr', 'get_id', 'get_frame', 'get_password',
40 'print_html_start', 'print_html_end',
41 'print_html_head_start', 'print_html_head_end',
42 'print_html_body_start', 'print_html_body_end',
43 'print_viewer_page', 'print_goto',
44 'write_index', 'write_static_viewer_page', 'write_static_goto',
45 'get_frame_file', 'get_page_file',
46 'read_frame_data', 'write_frame_data', 'read_default', 'read_noaccess',
47 'read_state', 'write_state',
48 'read_words_list', 'write_words_list', 'read_words', 'write_words',
49 'read_story', 'write_story',
50 'read_goto', 'write_goto',
51 'read_chat', 'write_chat',
52 'read_settings', 'read_attachment', 'read_coincidence',
54 'eval_bb', 'bb_to_bbcode', 'bb_to_html'
57 ###PERL_LIB: use lib /botm/lib/bsta
60 'url_query_decode', 'url_query_encode',
61 'url_decode', 'url_encode',
62 'html_entity_encode_dec',
65 'read_data_file', 'write_data_file',
67 'copy_encoded', 'open_encoded', '_x_encoded',
68 'http_header_line', 'http_status',
69 'http_header_status', 'http_header_allow', 'http_header_location',
73 ###PERL_PATH_SEPARATOR: PATH_SEPARATOR = /
75 ###PERL_CGI_PATH: CGI_PATH = /bsta/
76 ###PERL_CGI_ATTACH_PATH: CGI_ATTACH_PATH = /bsta/a
77 ###PERL_CGI_2WORDS_PATH: CGI_2WORDS_PATH = /bsta/2words
78 ###PERL_CGI_BBCODE_PATH: CGI_BBCODE_PATH = /bsta/b
79 ###PERL_DATA_CHAT_PATH: DATA_CHAT_PATH = /botm/data/bsta/chat
80 ###PERL_CGI_COIN_PATH: CGI_COIN_PATH = /bsta/coin
81 ###PERL_CGI_CSS_PATH: CGI_CSS_PATH = /bsta/bsta.css
82 ###PERL_CGI_FRAME_PATH: CGI_FRAME_PATH = /bsta/f
83 ###PERL_CGI_GOTO_PATH: CGI_GOTO_PATH = /bsta/g
84 ###PERL_CGI_INFO_PATH: CGI_INFO_PATH = /bsta/i
85 ###PERL_CGI_LIST_PATH: CGI_LIST_PATH = /bsta/goto.htm
86 ###PERL_CGI_LOGO_PATH: CGI_LOGO_PATH = /bsta/botmlogo.png
87 ###PERL_CGI_TIMER_PATH: CGI_TIMER_PATH = /bsta/timer.js
88 ###PERL_CGI_VIEWER_PATH: CGI_VIEWER_PATH = /bsta/v
89 ###PERL_CGI_WORDS_PATH: CGI_WORDS_PATH = /bsta/w
91 ###PERL_DATA_PATH: DATA_PATH = /botm/data/bsta/
92 ###PERL_DATA_ATTACH_PATH: DATA_ATTACH_PATH = /botm/data/bsta/a
93 ###PERL_DATA_COIN_PATH: DATA_COIN_PATH = /botm/data/bsta/coincidence
94 ###PERL_DATA_DEFAULT_PATH: DATA_DEFAULT_PATH = /botm/data/bsta/default
95 ###PERL_DATA_LIST_PATH: DATA_LIST_PATH = /botm/data/bsta/list
96 ###PERL_DATA_NOACCESS_PATH: DATA_NOACCESS_PATH = /botm/data/bsta/noaccess
97 ###PERL_DATA_SETTINGS_PATH: DATA_SETTINGS_PATH = /botm/data/bsta/settings
98 ###PERL_DATA_STATE_PATH: DATA_STATE_PATH = /botm/data/bsta/state
99 ###PERL_DATA_STORY_PATH: DATA_STORY_PATH = /botm/data/bsta/story
100 ###PERL_DATA_WORDS_PATH: DATA_WORDS_PATH = /botm/data/bsta/words/
102 ###PERL_WWW_PATH: WWW_PATH = /botm/www/
103 ###PERL_WWW_GOTO_PATH: WWW_GOTO_PATH = /botm/www/1190/bsta/goto.htm
104 ###PERL_WWW_INDEX_PATH: WWW_INDEX_PATH = /botm/www/1190/bsta/index.htm
106 ###PERL_SCHEME: SCHEME = http
107 ###PERL_WEBSITE: WEBSITE = 1190.bicyclesonthemoon.info
108 ###PERL_WEBSITE_NAME: WEBSITE_NAME = Bicycles on the Moon
109 ###PERL_FAVICON_PATH: FAVICON_PATH = /img/favicon.png
111 ###PERL_COIN_DATE: COIN_DATE = 13-Nov-2016 22:15
112 ###PERL_INTF_DATE: INTF_DATE = 28-Sep-2016 20:34
114 ###PERL_STORY_CREDITS: STORY_CREDITS = "BSTA" by Balthasar Szczepański
115 ###PERL_INTF_CREDITS: INTF_CREDITS = Online interface © Balthasar Szczepański; AGPL 3 license
116 ###PERL_SOURCE_URL: SOURCE_URL = http://bicyclesonthemoon.info/git-projects/?p=ott/bsta
118 ###PERL_COMMENT_PAGE_LENGTH:COMMENT_PAGE_LENGTH= 16
120 use constant STATE => {
126 use constant INTF_STATE => {
139 use constant TEXT_MODE => {
145 use constant CHAT_STATE => {
150 use constant CHAT_ACTION => {
158 use constant tags_bbcode => {
165 'quote' => '[quote]',
166 'quote=' => '[quote="',
168 '/quote' => '[/quote]',
169 'ni' => '[color=#0057AF]',
171 'br' => '[color=#BB6622]',
173 'po' => '[color=#FF8800]',
184 '/list' => '[/list]',
188 '/?' => '[/unknown!]',
190 use constant tags_html => {
193 'fq' => '<div class="fq">',
195 'tq' => '<div class="tq">',
197 'quote' => '<div class="opomba"><div class="opomba_text">',
198 'quote=' => '<div class="opomba"><div class="opomba_info"><b>',
199 'quote/='=> '</b> wrote:</div><div class="opomba_text">',
200 '/quote' => '</div></div>',
201 'ni' => '<span class="ni">',
203 'br' => '<span class="br">',
205 'po' => '<span class="po">',
207 'url' => '<a href="#">',#think: how to add selfincluding?
208 'url=' => '<a href="',
214 'list=' => '<ol style="list-style-type: ',
215 'list=1' => 'decimal',
216 'list=A' => 'upper-alpha',
217 'list=a' => 'lower-alpha',
218 'list=I' => 'upper-roman',
219 'list=i' => 'lower-roman',
226 '/?' => '[/unknown!]',
231 # function to obtain address of remote agent
232 sub get_remote_addr {
233 if ($ENV{'HTTP_X_FORWARDED_FOR'} =~ /^.+$/) {
236 elsif ($ENV{'REMOTE_ADDR'} =~ /^.+$/) {
244 # functions to get ID/number etc.
246 (my $cgi, my $default, my $cgi_name) = @_;
247 if ($default eq '') {
250 if ($cgi_name eq '') {
254 if ($cgi->{$cgi_name} =~ /^.+$/) {
257 elsif ($ENV{'PATH_INFO'} =~ /^\/(.+)$/) {
261 return int($default);
265 # function to obtain frame number
267 (my $cgi, my $default) = @_;
268 return get_id($cgi, $default, 'f');
271 # function to obtain password
275 if ($cgi->{'p'} =~ /^.+$/) {
285 # different & simpler implementation than in post library
289 #analyse bbcode text to build tag tree
290 #TODO make [/*] optional!
292 (my $bb, my $printdebug) = @_;
305 $bbtree{"_.name" } = "ht";
306 $bbtree{"_.value" } = '';
307 $bbtree{"_.type" } = "tag";
308 $bbtree{"_.count" } = 0;
309 $bbtree{"_.closed"} = 0;
310 $debug .= debug($printdebug,
312 "<!--GENERATING BBCODE TREE:\n".
313 '[_]automatic tag: [ht]'."\n"
317 my $new_ind = $ind.'.'.$bbtree{$ind.'.count'};
319 if($bb =~ m/\[(\/?)([A-Za-z]+|\*)(=([^\[\]]*))?\]/g) {
326 if ($tag_value =~ /^"(.*)"$/) {
330 if ($pre_text ne '') {
331 $debug .= debug($printdebug, "[$new_ind]text: $pre_text\n");
332 $bbtree{$new_ind.'.type' } = 'text';
333 $bbtree{$new_ind.'.value'} = $pre_text;
334 $bbtree{ $ind.'.count'}+= 1;
335 $new_ind = $ind.'.'.$bbtree{$ind.'.count'};
338 if($tag_name =~ /^(fq|tq|quote|br|ni|po|url|i|list|\*)$/) {
339 if ($tag_end ne '') {
341 ($tag_name ne $bbtree{$ind.'.name'}) ||
344 $debug .= debug($printdebug, "[$new_ind]text: $tag\n");
345 $bbtree{$new_ind.'.type' } = 'text';
346 $bbtree{$new_ind.'.value'} = $tag;
347 $bbtree{ $ind.'.count'}+= 1;
348 # $new_ind = $ind.'.'.$bbtree{$ind.'.count'};
351 $debug .= debug($printdebug, "[$new_ind]tag: $tag\n");
352 $bbtree{$new_ind.'.type' } = 'tag';
353 $bbtree{$new_ind.'.name' } = '/'.$tag_name;
354 $bbtree{$new_ind.'.value' } = $tag_value;
355 $bbtree{ $ind.'.count' }+= 1;
356 $bbtree{ $ind.'.closed'} = 1;
358 $ind =~ s/\.[0-9]+$//;
359 # $new_ind = $ind.'.'.$bbtree{$ind.'.count'};
364 $debug .= debug($printdebug, "[$new_ind]tag: $tag\n");
365 $bbtree{$new_ind.'.type' } = 'tag';
366 $bbtree{$new_ind.'.name' } = $tag_name;
367 $bbtree{$new_ind.'.value' } = $tag_value;
368 $bbtree{$new_ind.'.count' } = 0;
369 $bbtree{$new_ind.'.closed'} = 0;
370 $bbtree{ $ind.'.count' }+= 1;
373 # $new_ind = $ind.'.'.$bbtree{$ind.'.count'};
377 $debug .= debug($printdebug, "[$new_ind]text: $tag\n");
378 $bbtree{$new_ind.'.type' } = 'text';
379 $bbtree{$new_ind.'.value'} = $tag;
380 $bbtree{ $ind.'.count'}+= 1;
381 # $new_ind = $ind.'.'.$bbtree{$ind.'.count'};
385 $debug .= debug($printdebug, "[$new_ind]text: $bb\n");
386 $bbtree{$new_ind.'.type' } = 'text';
387 $bbtree{$new_ind.'.value'} = $bb;
388 $bbtree{ $ind.'.count'}+= 1;
389 # $new_ind = $ind.'.'.$bbtree{$ind.'.count'};
393 my $final_ind = '_.'.$bbtree{"_.count"};
394 $debug .= debug($printdebug, "[$final_ind]automatic tag: [/ht]\n -->\n");
395 $bbtree{$final_ind.'.type' } = "tag";
396 $bbtree{$final_ind.'.name' } = '/ht';
397 $bbtree{ '_.count' }+= 1;
398 $bbtree{ '_.closed'} = 1;
400 return ($debug, %bbtree);
403 #convert tag tree to final text
405 (my $printdebug, my $debug, my $lang, my $bbtree) = @_;
410 my $tags = ($lang eq 'html') ? tags_html : tags_bbcode;
411 my $escape = ($lang eq 'html');
413 # $debug .= debug($printdebug, "\n****\n");
414 # foreach my $iiii (keys %tags) {
415 # $debug .= debug($printdebug, $iiii.'='.$tags->{$iiii}."\n");
417 # $debug .= debug($printdebug, "****\n");
422 $debug .= debug($printdebug, "\n<!--PROCESSING BBCODE TREE:\n");
424 while ($level >= 0) {
426 $debug .= debug($printdebug, "[$level:$ind:".int($bbtree->{$ind.'.count'})."]");
428 if ($bbtree->{$ind.'.type'} eq 'text') {
429 my $text = $bbtree->{$ind.'.value'};
430 $debug .= debug($printdebug, "text: ".$text);
431 $out .= $escape ? html_encode_line($text) : $text;
436 elsif ($bbtree->{$ind.'.type'} eq 'tag') {
437 my $name = $bbtree->{$ind.'.name'};
439 if ($name =~ /^\//) {
440 $debug .= debug($printdebug, "tag: [$name]");
442 $indd =~ s/\.([0-9]+)$//;
443 if (exists($tags->{$name.'='}) && ($bbtree->{$indd.'.value'} ne '')) {
444 $out .= $tags->{$name.'='};
446 elsif (exists($tags->{$name})) {
447 $out .= $tags->{$name};
450 $out .= $tags->{'/?'};
451 $debug .= debug($printdebug, "[unknown!]");
454 $ind =~ s/\.([0-9]+)$//;
456 $debug .= debug($printdebug, "[<]");
467 my $value = $bbtree->{$ind.'.value'};
468 if($bbtree->{$ind.'.closed'} ne '') {
469 $debug .= debug($printdebug, "tag: [$name]");
471 if (exists($tags->{$name.'='}) && ($value ne '')) {
472 if (exists($tags->{$name.'='.$value})) {
475 $tags->{$name.'='.$value} .
481 ($escape ? html_entity_encode_dec($value, 1) : $value) .
485 elsif (exists($tags->{$name})) {
486 $out .= $tags->{$name};
489 $out .= $out.$tags->{'?'};
490 $debug .= debug($printdebug, "[unknown!]");
494 $debug .= debug($printdebug, "unclosed tag: [$name]");
495 my $text = $name . (($value ne '') ? ('='.$value) : '');
496 $out .= '['.($escape ? html_encode_line($text) : $text).']';
498 if ($bbtree->{$ind.'.count'} > 0) {
501 $debug .= debug($printdebug, "[>]");
510 $debug .= debug($printdebug, "unknown thing: ".$bbtree->{$ind.'.type'});
511 #should not occur with a correct bbtree
512 #unless unimplemented
513 $ind =~ s/\.([0-9]+)$//;
515 $debug .= debug($printdebug, "[<ui]");
524 if ($goto_next ne '') {
526 $ind =~ s/\.([0-9]+)$//;
528 if (($i < $bbtree->{$ind.'.count'}) and ($1 ne '')){
535 # should not occur with a correct bbtree
536 $debug .= debug($printdebug, "[<$goto_next]");
539 } while ($level >= 0);}
542 $debug .= debug($printdebug, "[>$level:$ind]\n");
545 $debug .= debug($printdebug, "-->\n");
546 return ($debug, $out);
551 (my $bb, my $printdebug) = @_;
556 ($debug, %bbtree) = bbtree($bb, $printdebug);
557 ($debug, $ht) = convtree ($printdebug, $debug, 'html', \%bbtree);
564 (my $bb, my $printdebug) = @_;
569 ($debug, %bbtree) = bbtree($bb, $printdebug);
570 ($debug, $ht) = convtree ($printdebug, $debug, 'bb', \%bbtree);
576 (my $bb, my $full_url, my $password) = @_;
581 my $base_url = $full_url ?
582 {'scheme' => SCHEME(), 'host' => WEBSITE()} :
585 while ($bb =~ m/###([^#;]*);/g) {
590 if ($value =~ /^att&([0-9]+)$/) {
593 {'path' => CGI_ATTACH_PATH()},
597 elsif ($value =~ /^vw&([0-9]+)$/) {
600 {'path' => CGI_VIEWER_PATH()},
604 elsif ($value =~ /^fr&([0-9]+)$/) {
607 {'path' => CGI_FRAME_PATH()},
614 if (($value ne '') && ($password ne '')) {
617 {'query' => {'p' => $password}}
620 $bb = $before . $value . $after;
626 sub html_encode_line {
627 (my $text, my $non_ascii, my $all) = @_;
631 $text =~ s/\r\n/\n/gs;
634 while ($text ne '') {
635 $ind = index($text, "\n");
637 $html .= html_entity_encode_dec(substr($text, 0, $ind), $non_ascii, $all)."<br>\n";
638 $text = substr($text, $ind+1);
642 $html .= html_entity_encode_dec($text, 1);
650 (my $print, my $text) = @_;
660 sub print_html_start {
662 print $fh '<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "">'."\n";
663 print $fh '<html lang="en">'."\n";
668 print $fh '</html>'."\n";
671 sub print_html_head_start {
673 print $fh ' <head>'."\n";
674 print $fh ' <meta http-equiv="Content-type" content="text/html; charset=UTF-8">'."\n";
675 print $fh ' <link rel="icon" type="image/png" href="'.html_entity_encode_dec(FAVICON_PATH(),1).'">'."\n";
676 print $fh ' <link rel="stylesheet" href="'.html_entity_encode_dec(CGI_CSS_PATH(),1).'">'."\n";
679 sub print_html_head_end {
681 print $fh ' </head>'."\n";
684 sub print_html_body_start {
686 print $fh ' <body>'."\n";
687 print $fh ' <a href="/"><img id="botmlogo" src="'.html_entity_encode_dec(CGI_LOGO_PATH(),1).'" alt="'.html_entity_encode_dec(WEBSITE(),1).'"></a>'."\n";
688 print $fh ' <div id="all">'."\n";
691 sub print_html_body_end {
692 (my $fh, my $hide_credits) = @_;
693 print $fh ' </div>'."\n";
694 unless ($hide_credits) {
695 print $fh ' <p>'."\n";
696 print $fh ' '.html_entity_encode_dec(STORY_CREDITS(),1).'<br>'."\n";
697 print $fh ' '.html_entity_encode_dec(INTF_CREDITS(),1).'<br>'."\n";
698 print $fh ' <a href="'.html_entity_encode_dec(SOURCE_URL(),1).'" class="cz">source code</a>'."\n";
699 print $fh ' </p>'."\n";
701 print $fh ' <a href="/" class="cz">'.html_entity_encode_dec(WEBSITE(),1).'</a>'."\n";
702 print $fh ' </body>'."\n";
705 sub print_html_data {
706 (my $fh, my $data) = @_;
708 foreach my $key (keys %$data) {
709 unless ($key eq 'content') {
710 my $val = $data->{$key};
711 $val =~ s/(\r)?\n/\n /gs; # does the space make sense in HTML anyway?
712 print $fh html_encode_line("$key: $val\n", 1);
715 print $fh html_encode_line("\n".$data->{'content'});
732 $last_frame = int($state->{'last'});
733 $ong_state = int($state->{'state'});
734 $password_query = url_query_encode({'p', $settings->{'password'}});
736 my $_title = html_entity_encode_dec($settings->{'story'}, 1);
737 my $_website_name = html_entity_encode_dec(WEBSITE_NAME() , 1);
738 my $_base_url = html_entity_encode_dec(CGI_PATH() , 1);
742 unless (seek($fh, 0, 0)) {
743 #don't actually fail here
747 unless (open_encoded($fh, ">:encoding(UTF-8)", $file)) {
752 print_html_start($fh);
753 print_html_head_start($fh);
755 print $fh ' <title>GOTO • '.$_title.' • '.$_website_name.'</title>'."\n";
757 print_html_head_end($fh);
758 print_html_body_start($fh);
760 print $fh ' <div id="inst" class="ins">'."\n";
762 print $fh ' <div id="title">'."\n";
763 print $fh ' <h1 id="titletext">'.$_title.'</h1>'."\n";
764 print $fh ' </div>'."\n";
766 print $fh ' </div>'."\n";
767 print $fh ' <div id="insb" class="ins">'."\n";
769 print $fh ' <div id="chat">'."\n";
771 for (my $frame = 0; ; $frame += 1) {
774 ($frame <= $last_frame) &&
775 ($ong_state >= STATE->{'waiting'})
788 $ongtime = $goto_list->{'ongtime-'.$frame};
789 $title = $goto_list->{'title-' .$frame};
790 if (($ongtime eq '') && ($title eq '')) {
791 my %frame_data = read_frame_data($frame);
792 $ongtime = $frame_data{'ongtime'};
793 $title = $frame_data{'title'};
794 unless (keys %frame_data) {
799 if ($ongtime ne '') {
800 @time_tab = gmtime($ongtime);
801 $time_text = sprintf(
802 '%02d.%02d.%02d %02d:%02d',
811 $time_text = (($frame <= $last_frame) && ($ong_state >= STATE->{'waiting'})) ?
812 'EE.EE.EE EE:EE' : '--.--.-- --:--';
817 $timer_color = (($frame > $last_frame) || ($ong_state < STATE->{'waiting'})) ?
819 (($frame == $last_frame) && ($ong_state < STATE->{'ready'})) ?
822 $frame_text = sprintf('%03d',$frame);
823 $viewer_url = merge_url(
824 {'path' => CGI_VIEWER_PATH()},
826 ); # TODO: consider static here?
828 $viewer_url = merge_url($viewer_url, {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
831 my $_viewer_url = html_entity_encode_dec($viewer_url, 1);
832 my $_title = html_entity_encode_dec($title , 1);
834 print $fh ' <span class="'.$timer_color.'">'.$frame_text.'</span> '.$time_text.' <a href="'.$_viewer_url.'">'.$_title.'</a><br>'."\n";
836 print $fh ' </div>'."\n";
838 print $fh ' <div id="underlinks">'."\n";
839 print $fh ' <a href="'.$_base_url.'">BSTA</a>'."\n";
840 print $fh ' </div>'."\n";
842 print $fh ' </div>'."\n";
844 print_html_body_end($fh, $ong_state == STATE->{'inactive'});
847 unless (ref($file)) {
851 truncate ($fh , tell($fh));
857 sub print_viewer_page {
870 my $launch = $context->{'launch'};
871 my $access = $context->{'access'};
872 my $password_ok = $context->{'password_ok'};
873 my $static = $context->{'static'};
875 my $goto = int($context->{'goto'});
876 my $frame = int($context->{'frame'});
877 my $text_mode = int($context->{'text_mode'});
878 my $timer_unlocked = int($context->{'timer_unlocked'});
879 my $timer = int($context->{'timer'});
880 # my $words_page = int($context->{'words_page'});
882 my $prev_frame = $frame - 1;
883 my $next_frame = $frame + 1;
885 my $story = $settings->{'story'};
886 my $title = $frame_data->{'title'};
887 my $command = ($frame_data->{'command'} ne '') ?
888 $frame_data->{'command'} :
889 $next_frame_data->{'title'};
891 my $last_frame = int($state->{'last'});
892 my $ong_state = int($state->{'state'});
894 my $width = int($frame_data->{'width'});
895 my $height = int($frame_data->{'height'});
896 my $frame_type = $frame_data->{'frametype'};
898 my $timer_color_h = (($timer_unlocked >= 1) || ($ong_state >= STATE->{'ready'})) ? 'br' : 'ni';
899 my $timer_color_m = (($timer_unlocked >= 2) || ($ong_state >= STATE->{'ready'})) ? 'br' : 'ni';
900 my $timer_color_s = (($timer_unlocked >= 3) || ($ong_state >= STATE->{'ready'})) ? 'br' : 'ni';
907 (($timer >= 0) && ($frame == 0))
909 $timer_s = sprintf('%02d', $timer % 60);
910 $timer_h = int($timer / 60);
911 $timer_m = sprintf('%02d', $timer_h % 60);
912 $timer_h = sprintf('%02d', $timer_h / 60);
914 elsif (($timer >= -15) && ($ong_state >= STATE->{'ready'})) {
925 my $words_posts = int($words_data->{'posts'});
926 my $words_link_text = 'Words'.(($words_posts > 0) ? "[$words_posts]" : '');
928 my $prev_available = (($frame > 0) && $access);
929 my $next_available = ($launch || $password_ok || ($next_frame <= $last_frame));
930 my $prefetch_prev = (
932 ($prev_frame < $last_frame) || ( # avoid unseen trigger!
933 ($prev_frame <= $last_frame) &&
934 ($ong_state >= STATE->{'ready'})
937 my $prefetch_next = (
939 ($next_frame < $last_frame) || ( # avoid unseen trigger!
940 ($next_frame <= $last_frame) &&
941 ($ong_state >= STATE->{'ready'})
948 ($frame == $last_frame) && (
949 ($ong_state == STATE->{'waiting'}) ||
950 ($ong_state == STATE->{'ready'})
958 ($frame < $last_frame) || (
959 ($ong_state >= STATE->{'ready'}) &&
960 $context->{'show_command'}
963 my $show_command_link = ($next_available || (!$access));
964 my $show_command_cursor = ((!$next_available) || ($command eq ''));
965 my $show_words = ($password_ok || ($access && !$launch));
967 my $frame_indirect = !(
969 ($frame <= $last_frame) &&
970 ($ong_state > STATE->{'inactive'})
973 my $prevframe_indirect = !($prev_frame <= $last_frame);
974 my $nextframe_indirect = !($next_frame <= $last_frame);
978 my $base_url = CGI_PATH();
979 my $timer_url = CGI_TIMER_PATH();
980 my $viewer_full_url = merge_url(
981 {'scheme' => SCHEME(), 'host' => WEBSITE()},
982 {'path' => CGI_VIEWER_PATH()},
985 my $viewer_url = merge_url(
986 {'path' => CGI_VIEWER_PATH()},
989 my $viewer_0_url = merge_url(
990 {'path' => CGI_VIEWER_PATH()},
993 my $viewer_prev_url = merge_url(
994 {'path' => CGI_VIEWER_PATH()},
995 {'path' => $prev_frame}
997 my $viewer_next_url = merge_url(
998 {'path' => CGI_VIEWER_PATH()},
999 {'path' => $next_frame}
1001 my $viewer_last_url = merge_url(
1002 {'path' => CGI_VIEWER_PATH()},
1003 {'path' => ($static ? -1 : $last_frame)}
1005 my $goto_url = ($goto) ?
1008 {'path' => $viewer_url},
1014 'fragment' => 'goto'
1018 unless ($password_ok) {
1020 $viewer_0_url = $base_url;
1021 if ($prev_frame == 0) {
1022 $viewer_prev_url = $viewer_0_url;
1025 $page_file = get_page_file($prev_frame, $prev_frame_data, $settings);
1026 if (_x_encoded('-f',
1027 join_path(PATH_SEPARATOR(), WWW_PATH() , $page_file)
1029 $viewer_prev_url = merge_url(
1030 {'path' => $base_url},
1031 {'path' => $page_file}
1035 if ($next_frame < $last_frame) {
1036 $page_file = get_page_file($next_frame, $next_frame_data, $settings);
1037 if (_x_encoded('-f',
1038 join_path(PATH_SEPARATOR(), WWW_PATH() , $page_file)
1040 $viewer_next_url = merge_url(
1041 {'path' => $base_url},
1042 {'path' => $page_file}
1048 (_x_encoded('-f',WWW_GOTO_PATH()))
1050 $goto_url = CGI_LIST_PATH();
1053 my $bbcode_url = ($text_mode == TEXT_MODE->{'bb'}) ?
1055 {'path' => CGI_BBCODE_PATH()},
1062 'b' => TEXT_MODE->{'bb'}
1067 my $info_url = ($text_mode == TEXT_MODE->{'info'}) ?
1069 {'path' => CGI_INFO_PATH()},
1076 'b' => TEXT_MODE->{'info'}
1081 my $words_url = merge_url (
1085 'b' => TEXT_MODE->{'words'}
1094 my $frame_normal_url;
1096 $frame_file = get_frame_file($frame, $frame_data, $settings);
1097 $frame_normal_url = merge_url(
1098 {'path' => CGI_PATH()},
1099 {'path' => $frame_file}
1101 $frame_url = $frame_indirect ?
1103 {'path' => CGI_FRAME_PATH()},
1107 $frame_full_url = merge_url(
1108 {'scheme' => SCHEME(), 'host' => WEBSITE()},
1109 {'path' => $frame_normal_url}
1111 if ($prevframe_indirect) {
1112 $frame_prev_url = merge_url(
1113 {'path' => CGI_FRAME_PATH()},
1114 {'path' => $prev_frame}
1118 $frame_prev_url = merge_url(
1119 {'path' => CGI_PATH()},
1120 {'path' => get_frame_file($prev_frame, $prev_frame_data, $settings)}
1123 if ($nextframe_indirect) {
1124 $frame_next_url = merge_url(
1125 {'path' => CGI_FRAME_PATH()},
1126 {'path' => $next_frame}
1130 $frame_next_url = merge_url(
1131 {'path' => CGI_PATH()},
1132 {'path' => get_frame_file($next_frame, $next_frame_data, $settings)}
1137 $password_query = url_query_encode({'p', $settings->{'password'}});
1138 $goto_url = merge_url($goto_url , {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
1139 $info_url = merge_url($info_url , {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
1140 $words_url = merge_url($words_url , {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
1141 $bbcode_url = merge_url($bbcode_url , {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
1142 $viewer_url = merge_url($viewer_url , {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
1143 $viewer_0_url = merge_url($viewer_0_url , {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
1144 $viewer_prev_url = merge_url($viewer_prev_url, {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
1145 $viewer_next_url = merge_url($viewer_next_url, {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
1146 $viewer_last_url = merge_url($viewer_last_url, {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
1147 if ($frame_indirect) {
1148 $frame_url = merge_url($frame_url , {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
1150 if ($prevframe_indirect) {
1151 $frame_prev_url= merge_url($frame_prev_url , {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
1153 if ($nextframe_indirect) {
1154 $frame_next_url= merge_url($frame_next_url , {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
1157 my $_password = $password_ok ? html_entity_encode_dec($settings->{'password'}, 1) : '';
1158 my $_action_url = html_entity_encode_dec(CGI_VIEWER_PATH(), 1);
1159 my $_base_url = html_entity_encode_dec($base_url , 1);
1160 my $_goto_url = html_entity_encode_dec($goto_url , 1);
1161 my $_info_url = html_entity_encode_dec($info_url , 1);
1162 my $_words_url = html_entity_encode_dec($words_url , 1);
1163 my $_bbcode_url = html_entity_encode_dec($bbcode_url , 1);
1164 my $_timer_url = html_entity_encode_dec($timer_url , 1);
1165 my $_viewer_full_url = html_entity_encode_dec($viewer_full_url , 1);
1166 my $_viewer_url = html_entity_encode_dec($viewer_url , 1);
1167 my $_viewer_0_url = html_entity_encode_dec($viewer_0_url , 1);
1168 my $_viewer_prev_url = html_entity_encode_dec($viewer_prev_url , 1);
1169 my $_viewer_next_url = html_entity_encode_dec($viewer_next_url , 1);
1170 my $_viewer_last_url = html_entity_encode_dec($viewer_last_url , 1);
1171 my $_frame_url = html_entity_encode_dec($frame_url , 1);
1172 my $_frame_prev_url = html_entity_encode_dec($frame_prev_url , 1);
1173 my $_frame_next_url = html_entity_encode_dec($frame_next_url , 1);
1174 my $_frame_full_url = html_entity_encode_dec($frame_full_url , 1);
1176 my $_story = html_entity_encode_dec($story , 1);
1177 my $_title = html_entity_encode_dec($title , 1);
1178 my $_command = html_entity_encode_dec($command , 1);
1179 my $_frame_type = html_entity_encode_dec($frame_type, 1);
1181 my $_website_name = html_entity_encode_dec(WEBSITE_NAME(), 1);
1183 if ($text_mode == TEXT_MODE->{'info'}) {
1184 if ($show_command) {
1185 $frame_data->{'command'} = $command;
1188 $frame_data->{'frame'} = $frame_file;
1190 if ($frame_data->{'page'} eq '') {
1191 unless (($access) && ($frame < $last_frame)) {
1192 $frame_data->{'page'} = '';
1195 $frame_data->{'page'} = get_page_file($frame, $frame_data, $settings);
1200 # everything determined, now start generating
1204 unless (seek($fh, 0, 0)) {
1205 #don't actually fail here
1209 unless (open_encoded($fh, ">:encoding(UTF-8)", $file)) {
1214 print_html_start($fh);
1215 print_html_head_start($fh);
1217 print $fh ' <title>'.$_title;
1218 if ($story ne $title) {
1219 print $fh ' • '.$_story;
1221 print $fh ' • '.$_website_name.'</title>'."\n";
1222 print $fh ' <link rel="index" href="'.$_goto_url.'">'."\n";
1223 print $fh ' <link rel="start" href="'.$_viewer_0_url.'">'."\n";
1224 if ($prev_available) {
1225 print $fh ' <link rel="prev" href="'.$_viewer_prev_url.'">'."\n";
1226 if ($prefetch_prev) {
1227 print $fh ' <link rel="prefetch" href="'.$_viewer_prev_url.'">'."\n";
1228 print $fh ' <link rel="prefetch" href="'.$_frame_prev_url.'">'."\n";
1231 if ($next_available) {
1232 print $fh ' <link rel="next" href="'.$_viewer_next_url.'">'."\n";
1233 if ($prefetch_next) {
1234 print $fh ' <link rel="prefetch" href="'.$_viewer_next_url.'">'."\n";
1235 print $fh ' <link rel="prefetch" href="'.$_frame_next_url.'">'."\n";
1239 print $fh ' <script src="'.$_timer_url.'"></script>'."\n";
1242 print_html_head_end($fh);
1243 print_html_body_start($fh);
1245 print $fh ' <div id="inst" class="ins">'."\n";
1247 print $fh ' <div id="title">'."\n";
1248 print $fh ' <h1 id="titletext">'.$_title.'</h1>'."\n";
1249 print $fh ' </div>'."\n";
1251 print $fh ' </div>'."\n";
1252 print $fh ' <div id="framespace">'."\n";
1254 print $fh ' <img src="'.$_frame_url.'" id="frame" class="'.$_frame_type.'" alt="'.$frame.'" title="'.$_title.'" width="'.$width.'" height="'.$height.'">'."\n";
1256 print $fh ' </div>'."\n";
1257 print $fh ' <div id="insb" class="ins">'."\n";
1259 if ($text_mode == TEXT_MODE->{'info'}) {
1260 print $fh ' <div id="chat">'."\n";
1262 print_html_data($fh, $frame_data);
1264 print $fh ' </div>'."\n";
1266 elsif ($text_mode == TEXT_MODE->{'bb'}) {
1267 print $fh ' <div id="chat">'."\n";
1269 print $fh '[quote][center][size=200]'.$_title.'[/size]<br>'."\n";
1270 print $fh '[url='.$_viewer_full_url.'][img]'.$_frame_full_url.'[/img][/url][/center]<br>'."\n";
1271 print $fh html_encode_line(
1274 $frame_data->{'content'},
1279 print $fh '[/quote]'."\n";
1281 print $fh ' </div>'."\n";
1283 elsif ($frame_data->{'content'} ne '') {
1284 print $fh ' <div id="undertext">'."\n";
1285 print $fh bb_to_html(
1287 $frame_data->{'content'},
1289 $password_ok ? $settings->{'password'} : ''
1292 print $fh ' </div>'."\n";
1295 print $fh ' <div id="command">'."\n";
1298 print $fh ' <span id="timer">';
1299 print $fh '[<span id="ongh" class="hv '.$timer_color_h.'">'.$timer_h.'</span>';
1300 print $fh ':<span id="ongm" class="hv '.$timer_color_m.'">'.$timer_m.'</span>';
1301 print $fh ':<span id="ongs" class="hv '.$timer_color_s.'">'.$timer_s.'</span>]';
1302 print $fh '</span><br>'."\n";
1305 if ($show_command_link) {
1306 print $fh '<a href="'.($access ? $_viewer_next_url : $_viewer_last_url).'">';
1308 if ($show_command) {
1309 print $fh $_command;
1311 if ($show_command_cursor) {
1312 print $fh '<span class="inp">_</span>';
1314 if ($show_command_link) {
1318 print $fh " </div>\n";
1320 print $fh ' <div id="underlinks">'."\n ";
1322 unless (($frame == 0) && $static) {
1323 print $fh '<a href="'.$_base_url.'">Once again</a> | ';
1325 if ($prev_available) {
1326 print $fh '<a href="'.$_viewer_prev_url.'">Before</a> | ';
1328 unless ($frame == $last_frame) {
1329 print $fh '<a href="'.$_viewer_last_url.'">Now</a> | ';
1331 print $fh '<a href="'.$_goto_url.'">GOTO</a>'."\n";
1332 print $fh ' <span style="float: right;">'."\n ";
1334 ($text_mode == TEXT_MODE->{'normal'})
1338 print $fh '<a href="'.$_words_url.'">'.$words_link_text.'</a> | ';
1342 print $fh '<a href="'.$_viewer_url.'">Without</a> | ';
1344 print $fh '<a href="'.$_info_url.'">Info</a> | ';
1345 print $fh '<a href="'.$_bbcode_url.'">BB</a>';
1346 print $fh "\n </span>\n";
1348 print $fh " </div>\n";
1351 print $fh ' <div class="underlinks" id="goto">'."\n";
1352 print $fh ' <form class="goto" method="get" action="'.$_action_url.'">'."\n";
1353 print $fh ' GO TO:'."\n";
1354 print $fh ' <input class="intx" type="number" size="4" name="f"'.(
1356 ('value="'.$frame.'"') :
1359 print $fh ' <input class="inbt" type="submit" value="GO">'."\n";
1361 print $fh ' <input type="hidden" name="p" value="'.$_password.'">'."\n";
1363 print $fh ' <input type="hidden" name="g" value="2">'."\n";
1364 print $fh ' </form>'."\n";
1365 print $fh " </div>\n";
1368 print $fh " </div>\n";
1370 if (($text_mode == TEXT_MODE->{'words'}) && $show_words) {
1371 print_comments($fh, $context, $settings, $words_data);
1374 print_html_body_end($fh, $ong_state == STATE->{'inactive'});
1375 print_html_end($fh);
1378 unless (ref($file)) {
1382 truncate ($fh , tell($fh));
1388 sub print_comments {
1389 (my $fh, my $context, my $settings, my $words_data) = @_;
1391 my $password_ok = $context->{'password_ok'};
1392 my $frame = int($context->{'frame'});
1393 my $page = int($context->{'words_page'});
1394 my $post_count = int($words_data->{'posts'});
1395 my $id_start = $page * COMMENT_PAGE_LENGTH();
1396 my $id_stop = $id_start + COMMENT_PAGE_LENGTH();
1397 my $older = ($page > 0) ? ($page-1) : '';
1400 if ($id_stop >= $post_count) {
1401 $id_stop = $post_count;
1408 my $words_url = merge_url(
1409 {'path' => CGI_VIEWER_PATH()},
1412 'query' => {'b' => TEXT_MODE->{'words'}},
1415 my $older_url = merge_url(
1418 'query' => {'i' => $page-1},
1419 'fragment' => 'insw',
1423 my $newer_url = merge_url(
1426 'query' => {'i' => $page+1},
1427 'fragment' => 'insw',
1433 $password_query = url_query_encode({'p', $settings->{'password'}});
1434 $older_url = merge_url($older_url, {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
1435 $newer_url = merge_url($older_url, {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
1438 my $_password = $password_ok ? html_entity_encode_dec($settings->{'password'}, 1) : '';
1439 my $_post_url = html_entity_encode_dec(CGI_WORDS_PATH(), 1);
1440 my $_older_url = html_entity_encode_dec($older_url, 1);
1441 my $_newer_url = html_entity_encode_dec($newer_url, 1);
1443 if (($older ne '') || ($newer ne '')) {
1444 $links .= ' <div class="underlinks">'."\n";
1447 $links .= '<a href="'.$_older_url.'">Older</a>'
1449 if (($older ne '') && ($newer ne '')) {
1453 $links .= '<a href="'.$_newer_url.'">Newer</a>';
1456 $links .= ' </div>'."\n";
1459 print $fh ' <div class="space"></div>'."\n";
1460 print $fh ' <div id="insw" class="ins">'."\n";
1462 print $fh ' <div class="title" id="wordstitle">'."\n";
1463 print $fh ' <h1 class="titletext" id="wordstitletext">Words</h1>'."\n";
1464 print $fh ' </div>'."\n";
1470 print $fh ' <div class="undertext" id="words">'."\n";
1472 if ($post_count > 0) {
1473 for (my $i=$id_start; $i<$id_stop; ++$i) {
1474 my $ID = $words_data->{'content'}->[$i];
1475 my %post_data = read_words($ID);
1477 my $post_time = int($post_data{'posttime'});
1478 my $edit_time = int($post_data{'edittime'});
1483 if ($post_time != 0) {
1484 my @time_tab = gmtime($post_time);
1485 $post_time_text = sprintf(
1486 '%04d.%02d.%02d %02d:%02d:%02d UTC',
1495 if (($edit_time !=0) && ($edit_time != $post_time)) {
1496 my @time_tab = gmtime($edit_time);
1497 $edit_time_text = sprintf(
1498 '%04d.%02d.%02d %02d:%02d UTC',
1506 my $quote_url = merge_url(
1507 {'path' => CGI_WORDS_PATH()},
1515 my $edit_url = merge_url(
1516 {'path' => CGI_WORDS_PATH()},
1521 'key' => $post_data{'key'},
1522 'username' => $post_data{'name'},
1526 my $remove_url = merge_url(
1527 {'path' => CGI_WORDS_PATH()},
1532 'key' => $post_data{'key'},
1533 'username' => $post_data{'name'},
1538 $quote_url = merge_url($quote_url , {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
1539 $edit_url = merge_url($edit_url , {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
1540 $remove_url = merge_url($remove_url, {'query' => $password_query, 'append_query' => 1, 'preserve_fragment' => 1});
1543 my $_ID = html_entity_encode_dec($ID, 1);
1544 my $_name = html_entity_encode_dec($post_data{'name'}, 1);
1545 my $_quote_url = html_entity_encode_dec($quote_url, 1);
1546 my $_edit_url = html_entity_encode_dec($edit_url, 1);
1547 my $_remove_url = html_entity_encode_dec($remove_url, 1);
1549 print $fh ' <div id="'.$_ID.'"class="opomba">'."\n";
1550 print $fh ' <div class="opomba_info">'."\n";
1551 print $fh ' <a href="#'.$_ID.'" class="bi hu">'.$i.': '.$_name;
1552 if ($post_time_text ne '') {
1553 print $fh ' • '.$post_time_text;
1555 if ($edit_time_text ne '') {
1556 print $fh ' • '.$edit_time_text;
1558 print $fh '</a>'."\n";
1559 print $fh ' <div class="pr">'."\n";
1560 print $fh ' <a href="'.$_quote_url.'" class="bi hu">quote</a> | <a href="'.$_edit_url.'" class="bi hu">edit</a> | <a href="'.$_remove_url.'" class="bi hu">remove</a>'."\n";
1561 print $fh ' </div>'."\n";
1562 print $fh ' </div>'."\n";
1563 print $fh ' <div class="opomba_text">'."\n";
1564 print $fh bb_to_html(
1566 $post_data{'content'},
1568 $password_ok ? $settings->{'password'} : ''
1571 print $fh ' </div>'."\n";
1572 print $fh ' </div>'."\n";
1573 print $fh ' <br>'."\n";
1577 print $fh ' <form method="post" action="'.$_post_url.'">'."\n";
1578 print $fh ' <b>Your words:</b>'."\n";
1579 print $fh ' <textarea class="inta" name="words" rows="4"></textarea>'."\n";
1580 print $fh ' <table cellpadding="0" cellspacing="0" border="0"><tr>'."\n";
1581 print $fh ' <td><b>Your name: </b></td>'."\n";
1582 print $fh ' <td><input class="intx" type="text" name="username" value=""></td>'."\n";
1583 print $fh ' <td></td>'."\n";
1584 print $fh ' </tr><tr>'."\n";
1585 print $fh ' <td><b>Optional password: </b></td>'."\n";
1586 print $fh ' <td><input class="intx" type="password" name="password" value=""></td>'."\n";
1587 print $fh ' <td>(if you want to edit later)</td>'."\n";
1588 print $fh ' </tr><tr>'."\n";
1589 print $fh ' <td><b>Leave this empty: </b></td>'."\n";
1590 print $fh ' <td><input class="intx" type="text" name="password2" value=""></td>'."\n";
1591 print $fh ' <td>'."\n";
1592 print $fh ' <input class="inbt" type="submit" name="post" value="Send">'."\n";
1593 print $fh ' <input class="inbt" type="submit" name="preview" value="Preview">'."\n";
1594 print $fh ' </td>'."\n";
1595 print $fh ' </tr></table>'."\n";
1596 print $fh ' <input type="hidden" name="f" value="'.$frame.'">'."\n";
1598 print $fh ' <input type="hidden" name="p" value="'.$_password.'">'."\n";
1600 print $fh ' </form>'."\n";
1601 print $fh ' </div>'."\n";
1607 print $fh ' </div>'."\n";
1620 my $ong_state = int($state->{'state'});
1622 unless (open_encoded($fh, ">:encoding(UTF-8)", WWW_INDEX_PATH())) {
1626 # normal running story
1627 if ($ong_state > STATE->{'inactive'}) {
1628 my %default = read_default();
1629 my %frame_data = read_frame_data(0, \%default);
1630 my %next_frame_data= read_frame_data(1, \%default);
1631 my %words_data = read_words_list(
1636 $r = print_viewer_page(
1643 'timer_unlocked' => 3, # not relevant
1644 'timer' => 0, # not relevant
1646 'show_command' => 1,
1647 'text_mode' => TEXT_MODE->{'normal'},
1648 'words_page' => 0, # not relevant
1659 # no conditions met, pretend a normal Apache2 index
1660 elsif ($pass != 1) {
1661 my $index_of = CGI_PATH;
1662 $index_of =~ s/\/$//g;
1664 my $_index_of = html_entity_encode_dec($index_of , 1);
1665 my $_2words_date = html_entity_encode_dec(INTF_DATE(), 1);
1666 my $_coin_date = html_entity_encode_dec(COIN_DATE(), 1);
1667 my $_website = html_entity_encode_dec(WEBSITE() , 1);
1669 print_html_start ($fh);
1670 print $fh ' <head>'."\n";
1671 print $fh ' <meta http-equiv="Content-type" content="text/html; charset=UTF-8">'."\n";
1672 print $fh ' <title>Index of '.$_index_of.'</title>'."\n";
1673 print $fh ' </head>'."\n";
1674 print $fh ' <body>'."\n";
1675 print $fh ' <h1>Index of '.$_index_of.'</h1>'."\n";
1676 print $fh ' <table>'."\n";
1677 print $fh ' <tr>'."\n";
1678 print $fh ' <th><img src="/icons/blank.gif" alt="[ICO]"></th>'."\n";
1679 print $fh ' <th><a href="?C=N;O=D">Name</a></th>'."\n";
1680 print $fh ' <th><a href="?C=M;O=A">Last modified</a></th>'."\n";
1681 print $fh ' <th><a href="?C=S;O=A">Size</a></th>'."\n";
1682 print $fh ' <th><a href="?C=D;O=A">Description</a></th>'."\n";
1683 print $fh ' </tr><tr>'."\n";
1684 print $fh ' <th colspan="5"><hr></th>'."\n";
1685 print $fh ' </tr><tr>'."\n";
1686 print $fh ' <td valign="top"><img src="/icons/back.gif" alt="[DIR]"></td>'."\n";
1687 print $fh ' <td><a href="/">Parent Directory</a></td>'."\n";
1688 print $fh ' <td> </td>'."\n";
1689 print $fh ' <td align="right"> - </td>'."\n";
1690 print $fh ' <td> </td>'."\n";
1691 print $fh ' </tr><tr>'."\n";
1692 print $fh ' <td valign="top"><img src="/icons/folder.gif" alt="[DIR]"></td>'."\n";
1693 print $fh ' <td><a href="2words/">2words/</a></td>'."\n";
1694 print $fh ' <td align="right">'.$_2words_date.' </td>'."\n";
1695 print $fh ' <td align="right"> - </td><td> </td>'."\n";
1696 print $fh ' </tr><tr>'."\n";
1697 print $fh ' <td valign="top"><img src="/icons/folder.gif" alt="[DIR]"></td>'."\n";
1698 print $fh ' <td><a href="coin/">coin/</a></td>'."\n";
1699 print $fh ' <td align="right">'.$_coin_date.' </td>'."\n";
1700 print $fh ' <td align="right"> - </td><td> Coincidence </td>'."\n";
1701 print $fh ' </tr><tr>'."\n";
1702 print $fh ' <th colspan="5"><hr></th>'."\n";
1703 print $fh ' </tr>'."\n";
1704 print $fh ' </table>'."\n";
1705 print $fh ' <address>Apache/2.2.22 (Debian) Server at '.$_website.' Port 80</address>'."\n";
1706 print $fh ' </body>'."\n";
1707 print_html_end ($fh);
1711 my %default = read_default();
1712 my %frame_data = read_frame_data(0, \%default);
1713 my %next_frame_data= read_frame_data(1, \%default);
1714 my %coin_data = read_coincidence();
1716 if (($mode == INTF_STATE->{'>'}) && $pause) {
1717 $r = print_viewer_page(
1724 'timer_unlocked' => 3,
1727 'show_command' => 1,
1728 'text_mode' => TEXT_MODE->{'normal'},
1729 'words_page' => 0, # not relevant
1737 {'posts' => 0} # words_data
1742 my $index_of = CGI_PATH;
1743 $index_of =~ s/\/$//g;
1747 my $show_parent_dir = 0;
1749 my $show_folders = 0;
1751 my $timer_color = 'ni';
1752 if ($mode == INTF_STATE->{'>'}) {
1753 $title = $settings->{'story'}; # $frame_data{'title'} ?
1754 $frame_file = 'intf-tr.gif';
1758 elsif ($mode == INTF_STATE->{'<<'}) {
1759 $title = 'Index of';
1760 $frame_file = 'intf-ll.gif';
1761 $show_parent_dir = 1;
1764 $timer_color = 'br';
1766 elsif ($mode == INTF_STATE->{'>>'}) {
1767 $title = 'Index of';
1768 $frame_file = 'intf-pp.gif';
1769 $show_parent_dir = 1;
1775 $title = 'Index of '.$index_of;
1776 $frame_file = 'intf-kw.gif';
1777 $show_parent_dir = 1;
1780 my $frame_url = merge_url(
1781 {'path' => CGI_PATH()},
1782 {'path' => $frame_file}
1784 my $coin_server = $coin_data{'server'};
1786 my $_title = html_entity_encode_dec($title , 1);
1787 my $_website_name = html_entity_encode_dec(WEBSITE_NAME() , 1);
1788 my $_frame_url = html_entity_encode_dec($frame_url , 1);
1789 my $_undertext = html_entity_encode_dec($undertext , 1);
1790 my $_2words_date = html_entity_encode_dec(INTF_DATE() , 1);
1791 my $_coin_date = html_entity_encode_dec(COIN_DATE() , 1);
1792 my $_coin_server = html_entity_encode_dec($coin_server , 1);
1793 my $_2words_url = html_entity_encode_dec(CGI_2WORDS_PATH(), 1);
1794 my $_coin_url = html_entity_encode_dec(CGI_COIN_PATH() , 1);
1796 print_html_start($fh);
1797 print_html_head_start($fh);
1799 print $fh ' <title>'.$_title.' • '.$_website_name.'</title>'."\n";
1801 print_html_head_end($fh);
1802 print_html_body_start($fh);
1804 print $fh ' <div id="inst" class="ins">'."\n";
1806 print $fh ' <div id="title">'."\n";
1807 print $fh ' <h1 id="titletext">'.$_title.'</h1>'."\n";
1808 print $fh ' </div>'."\n";
1810 print $fh ' </div>'."\n";
1811 print $fh ' <div id="framespace">'."\n";
1813 print $fh ' <img src="'.$_frame_url.'" id="frame" alt="0">'."\n"; # title="'.$_title.'"
1815 print $fh ' </div>'."\n";
1816 print $fh ' <div id="insb" class="ins">'."\n";
1818 print $fh ' <div id="undertext">'."\n";
1820 if ($show_parent_dir) {
1821 print $fh ' <img src="/icons/back.gif" alt="[DIR]"> <a href="..">Parent Directory</a><br>'."\n";
1823 if ($show_folders) {
1824 print $fh ' <img src="/icons/folder.gif" alt="[DIR]"> <a href="'.$_2words_url.'">2words/</a> '.$_2words_date.' - <br>'."\n";
1825 print $fh ' <img src="/icons/folder.gif" alt="[DIR]"> <a href="'.$_coin_url.'">coin/</a> '.$_coin_date.' - '.$_coin_server."\n";
1828 print $fh ' <img src="/icons/folder.gif" alt="[DIR]"> <a href="'.$_2words_url.'">yyyyb/</a>'."\n";
1830 if ($undertext ne '') {
1831 print $fh ' '.$_undertext."\n";
1834 print $fh ' </div>'."\n";
1837 print $fh ' <div id="command">'."\n";
1839 print $fh ' [<span id="ongh" class="'.$timer_color.'">'.$timer.'</span>';
1840 print $fh ':<span id="ongm" class="'.$timer_color.'">'.$timer.'</span>';
1841 print $fh ':<span id="ongs" class="'.$timer_color.'">'.$timer.'</span>]<br>'."\n";
1843 if ($undertext ne '') {
1844 print $fh '><a href="'.$_2words_url.'">'.$_undertext.'</a><span class="inp">_</span>'."\n";
1846 print $fh " </div>\n";
1849 print $fh " </div>\n";
1851 print_html_body_end($fh, $ong_state == STATE->{'inactive'});
1852 print_html_end($fh);
1858 sub write_static_viewer_page {
1865 my $prev_frame_data_ref,
1866 my $next_frame_data_ref,
1874 my %prev_frame_data;
1875 my %next_frame_data;
1880 $frame = int($frame);
1881 my $prev_frame = $frame -1;
1882 my $next_frame = $frame +1;
1884 %state = (ref ($state_ref)) ?
1887 my $ong_state = int($state{'state'});
1888 my $last_frame = int($state{'last'});
1890 unless ($ong_state > STATE->{'inactive'}) {
1895 ($frame < $last_frame) || (
1896 ($frame <= $last_frame) &&
1897 ($ong_state >= STATE->{'end'})
1904 %settings = (ref ($settings_ref)) ?
1907 %default = (ref ($default_ref)) ?
1911 %frame_data = (ref ($frame_data_ref)) ?
1913 read_frame_data($frame);
1915 %prev_frame_data = (ref ($prev_frame_data_ref)) ?
1916 %$prev_frame_data_ref : (
1917 ($prev_frame >= 0) ?
1918 read_frame_data($prev_frame) :
1922 %next_frame_data = (ref ($next_frame_data_ref)) ?
1923 %$next_frame_data_ref :
1924 read_frame_data($next_frame);
1926 %words_data = (ref ($words_data_ref)) ?
1933 %frame_data = merge_settings(\%default, \%frame_data);
1934 %prev_frame_data = merge_settings(\%default, \%prev_frame_data);
1935 %next_frame_data = merge_settings(\%default, \%next_frame_data);
1937 $file = get_page_file($frame, \%frame_data, \%settings);
1938 $file = join_path(PATH_SEPARATOR(), WWW_PATH(), $file);
1940 return print_viewer_page(
1947 'timer_unlocked'=> 3, # not relevant
1948 'timer' => 0, # not relevant
1950 'show_command' => 1,
1951 'text_mode' => TEXT_MODE->{'normal'},
1952 'words_page' => 0, # not relevant
1964 sub write_static_goto {
1965 (my $state_ref, my $settings_ref, my $goto_ref) = @_;
1970 %state = (ref ($state_ref)) ?
1973 %settings = (ref ($settings_ref)) ?
1976 %goto_list = (ref ($goto_ref)) ?
1989 # ONG the frame + attachment & stuff. NOT update state file.
1992 my $ID, my $ongtime, my $timer, my $update, my $print,
1993 my $settings_ref, my $default_ref, my $data_ref, my $goto_ref
1999 my $frame_data_path;
2007 my %frame_full_data;
2010 if ($ongtime eq '') {
2017 elsif ($ID eq 'c') {
2049 %settings = (ref ($settings_ref)) ?
2052 %default = (ref ($default_ref)) ? %$default_ref : read_default();
2053 $frame_data_path = $cfrt ?
2054 DATA_NOACCESS_PATH() :
2055 join_path(PATH_SEPARATOR(), DATA_PATH(), $frame);
2056 %frame_data = (ref ($data_ref)) ?
2058 read_frame_data($frame_data_path);
2059 %frame_full_data = merge_settings(\%default, \%frame_data);
2060 @files = (get_frame_file($frame, \%frame_full_data, \%settings), );
2062 %goto_list = (ref ($goto_ref)) ?
2065 for (my $i=0; ;$i+=1) {
2066 my %file_data = read_attachment($i);
2067 if ($file_data{'frame'} eq '') {
2070 if (int($file_data{'frame'}) != $frame) {
2073 if ($file_data{'content'} ne '') {
2076 unshift @files, $file_data{'filename'};
2080 ($frame_full_data{'ongtime'} eq '')
2082 $frame_data {'ongtime'} = $ongtime;
2083 $frame_full_data{'ongtime'} = $ongtime;
2089 ($frame_full_data{'timer'} eq '')
2092 $frame_data{'timer'} = int($timer);
2096 $r = write_frame_data($frame_data_path, \%frame_data);
2098 print STDERR "fail writing $frame_data_path\n";
2100 print "write frame data fail\n";
2105 $goto_list{'title-' .$frame} = $frame_full_data{'title'};
2106 $goto_list{'ongtime-'.$frame} = $frame_full_data{'ongtime'};
2107 $r = write_goto('', \%goto_list);
2109 print STDERR "fail writing ".DATA_LIST_PATH()."\n";
2111 print "write GOTO list fail\n";
2117 foreach my $file (@files) {
2118 $in_path = join_path(PATH_SEPARATOR(), DATA_PATH(), $file);
2119 $out_path = join_path(PATH_SEPARATOR(), WWW_PATH() , $file);
2121 print $in_path.' -> '.$out_path;
2123 $r = copy_encoded($in_path, $out_path);
2125 print (($r) ? " OK\n" : " FAIL\n");
2128 print STDERR "fail copy $in_path $out_path\n";
2137 sub get_frame_file {
2138 (my $frame, my $frame_data, my $settings) = @_;
2142 if ($frame_data->{'frame'} ne '') {
2143 $file = $frame_data->{'frame'};
2146 $pattern = validate_filename($settings->{'frame'}, '%d.%ext');
2149 int($frame), $frame_data->{'ext'}
2152 return validate_filename($file);
2156 (my $frame, my $frame_data, my $settings) = @_;
2163 if ($frame_data->{'page'} ne '') {
2164 $file = $frame_data->{'page'};
2167 $pattern = validate_filename($settings->{'frame'}, '%d.%ext');
2173 return validate_filename($file);
2176 sub validate_filename {
2177 (my $filename, my $fallback) = @_;
2178 if ($fallback eq '') {
2184 if ($filename =~ /^\./) {
2187 if (index($filename, PATH_SEPARATOR()) >= 0) {
2193 sub validate_frame_data {
2195 my %data = %$data_in;
2197 if ($data{'ongtime'} ne '') {
2198 $data{'ongtime'} = int($data{'ongtime'});
2200 if ($data{'timer'} ne '') {
2201 $data{'timer'} = int($data{'timer'});
2203 if ($data{'width'} ne '') {
2204 $data{'width'} = int($data{'width'});
2206 if ($data{'height'} ne '') {
2207 $data{'height'} = int($data{'height'});
2209 if ($data{'page'} ne '') {
2210 $data{'page'} = validate_filename($data{'page'});
2212 if ($data{'frame'} ne '') {
2213 $data{'frame'} = validate_filename($data{'frame'});
2219 sub validate_settings {
2221 my %data = %$data_in;
2223 if ($data{'ongtime'} ne '') {
2224 $data{'ongtime'} = int($data{'ongtime'});
2226 if ($data{'dynamicongtime'} ne '') {
2227 $data{'dynamicongtime'} = int($data{'dynamicongtime'});
2229 if ($data{'firstongtime'} ne '') {
2230 $data{'firstongtime'} = int($data{'firstongtime'});
2232 if ($data{'last'} ne '') {
2233 $data{'last'} = int($data{'last'});
2235 $data{'frame'} = validate_filename($data{'frame'}, '%d.%s');
2240 sub validate_state {
2242 my %data = %$data_in;
2244 if ($data{'state'} ne '') {
2245 $data{'state'} = int($data{'state'});
2247 if ($data{'last'} ne '') {
2248 $data{'last'} = int($data{'last'});
2250 if ($data{'nextong'} ne '') {
2251 $data{'nextong'} = int($data{'nextong'});
2257 sub validate_words_list {
2258 (my $data_in, my $not_list) = @_;
2259 my %data = %$data_in;
2261 if ($data{'ongtime'} ne '') {
2262 $data{'ongtime'} = int($data{'ongtime'});
2267 foreach my $ID (split(/\r?\n/, $data{'content'})) {
2268 $ID = validate_filename($ID);
2270 $id_list .= $ID."\n";
2273 $data{'content'} = $id_list;
2277 foreach my $ID (@{$data{'content'}}) {
2279 $ID = validate_filename($ID);
2284 $data{'content'} = [@id_list];
2290 sub validate_words {
2292 my %data = %$data_in;
2294 if ($data{'posttime'} ne '') {
2295 $data{'posttime'} = int($data{'posttime'});
2297 if ($data{'edittime'} ne '') {
2298 $data{'edittime'} = int($data{'edittime'});
2304 sub validate_story {
2306 my %data = %$data_in;
2308 if ($data{'id'} ne '') {
2309 $data{'id'} = int($data{'id'});
2311 if ($data{'pass'} ne '') {
2312 $data{'pass'} = int($data{'pass'});
2314 if ($data{'state'} ne '') {
2315 $data{'state'} = int($data{'state'});
2323 my %data = %$data_in;
2325 foreach my $key (keys %data) {
2326 if ($key =~ /^ongtime-([0-9]+)$/) {
2327 my $new_key = 'ongtime-'.int($1);
2328 $data{$new_key} = int($data{$key});
2329 if ($new_key != $key) {
2338 sub validate_attachment {
2340 my %data = %$data_in;
2342 if ($data{'frame'} ne '') {
2343 $data{'frame'} = int($data{'frame'});
2345 $data{'filename'} = validate_filename($data{'filename'});
2350 sub validate_coincidence {
2352 my %data = %$data_in;
2354 if ($data{'server'} ne '') {
2355 $data{'server'} = int($data{'server'});
2361 sub read_frame_data {
2362 (my $f, my $default) = @_;
2366 if (ref ($f)) { # already open file
2369 elsif ($f =~ /^[0-9]+$/) { # frame ID
2370 $file = join_path(PATH_SEPARATOR(), DATA_PATH(), int($&));
2372 elsif ($f =~ /^(c(frt)?)|(noaccess)$/) { # CFRT (no access)
2373 $file = DATA_NOACCESS_PATH();
2375 elsif ($f =~ /^d(efault)?$/) { # default
2376 $file = DATA_DEFAULT_PATH();
2378 elsif ($f ne '') { # path
2382 $file = DATA_DEFAULT_PATH();
2385 %data = read_data_file($file);
2386 if (ref ($default)) {
2387 %data = merge_settings($default, \%data);
2389 elsif ($default ne '') {
2390 my %default_data = read_data_file(DATA_DEFAULT_PATH());
2391 %data = merge_settings(\%default_data, \%data);
2394 return validate_frame_data(\%data);
2397 sub write_frame_data {
2398 (my $f, my $data) = @_;
2401 if (ref ($f)) { # already open file
2404 elsif ($f =~ /^[0-9]+$/) { # frame ID
2405 $file = join_path(PATH_SEPARATOR(), DATA_PATH(), int($&));
2407 elsif ($f =~ /^(c(frt)?)|(noaccess)$/) { # CFRT (no access)
2408 return 0; # forbidden
2410 elsif ($f =~ /^d(efault)?$/) { # default
2411 return 0; # forbidden
2413 elsif ($f ne '') { # path
2417 return 0; # forbidden
2420 my %_data = validate_frame_data($data);
2422 return write_data_file($file, \%_data);
2426 return read_frame_data('default');
2431 return read_frame_data('noaccess', $default);
2439 if (ref ($f)) { # already open file
2442 elsif ($f ne '') { # path
2446 $file = DATA_SETTINGS_PATH();
2449 %data = read_data_file($file);
2451 return validate_settings(\%data);
2459 if (ref ($f)) { # already open file
2462 elsif ($f ne '') { # path
2466 $file = DATA_STATE_PATH();
2469 %data = read_data_file($file);
2471 return validate_state(\%data);
2475 (my $f, my $data) = @_;
2478 if (ref ($f)) { # already open file
2481 elsif ($f ne '') { # path
2485 $file = PERL_DATA_STATE_PATH();
2488 my %_data = validate_state($data);
2490 return write_data_file($file, \%_data);
2493 sub read_words_list {
2494 (my $f, my $header_only, my $not_list) = @_;
2498 if (ref ($f)) { # already open file
2501 elsif ($f =~ /^[0-9]+$/) { # frame ID
2502 $file = join_path(PATH_SEPARATOR(), DATA_WORDS_PATH(), int($&));
2504 elsif ($f ne '') { # path
2507 else { # which frame ???
2508 return ('posts' => 0);
2511 %data = read_data_file(
2516 not $not_list # as list
2519 return validate_words_list(\%data, $not_list);
2522 sub write_words_list {
2523 (my $f, my $data) = @_;
2526 if (ref ($f)) { # already open file
2529 elsif ($f =~ /^[0-9]+$/) { # frame ID
2530 $file = join_path(PATH_SEPARATOR(), DATA_WORDS_PATH(), int($&));
2532 elsif ($f ne '') { # path
2535 else { # which frame ???
2539 my %_data = validate_words_list($data);
2541 return write_data_file(
2552 (my $f, my $default) = @_;
2556 if (ref ($f)) { # already open file
2559 elsif ($f =~ /^[0-9]+\.[0-9\.]+$/) { # post ID
2560 $file = join_path(PATH_SEPARATOR(), DATA_WORDS_PATH(), $&);
2562 elsif ($f ne '') { # path
2565 else { # which post ???
2569 %data = read_data_file($file);
2571 return validate_words(\%data);
2575 (my $f, my $data) = @_;
2578 if (ref ($f)) { # already open file
2581 elsif ($f =~ /^[0-9\.]+$/) { # post ID
2582 $file = join_path(PATH_SEPARATOR(), DATA_WORDS_PATH(), $&);
2584 elsif ($f ne '') { # path
2587 else { # which post ???
2591 my %_data = validate_words($data);
2593 return write_data_file($file, \%_data);
2601 if (ref ($f)) { # already open file
2604 elsif ($f =~ /^[0-9]+$/) { # story ID
2605 $file = DATA_STORY_PATH().int($&);
2607 elsif ($f ne '') { # path
2611 $file = DATA_STORY_PATH();
2614 %data = read_data_file($file);
2616 return validate_story(\%data);
2620 (my $f, my $data) = @_;
2623 if (ref ($f)) { # already open file
2626 elsif ($f =~ /^[0-9]+$/) { # story ID
2627 $file = DATA_STORY_PATH().int($&);
2629 elsif ($f ne '') { # path
2633 $file = DATA_STORY_PATH();
2636 my %_data = validate_story($data);
2638 return write_data_file($file, \%_data);
2646 if (ref ($f)) { # already open file
2649 elsif ($f ne '') { # path
2653 $file = DATA_LIST_PATH();
2656 %data = read_data_file($file);
2658 return validate_goto(\%data);
2662 (my $f, my $data) = @_;
2665 if (ref ($f)) { # already open file
2668 elsif ($f ne '') { # path
2672 $file = DATA_LIST_PATH();
2675 my %_data = validate_goto($data);
2677 return write_data_file($file, \%_data);
2680 sub read_attachment {
2681 (my $f, my $default) = @_;
2685 if (ref ($f)) { # already open file
2688 elsif ($f =~ /^[0-9]+$/) { # attachment ID
2689 $file = DATA_ATTACH_PATH().int($&);
2691 elsif ($f ne '') { # path
2698 %data = read_data_file($file);
2700 return validate_attachment(\%data);
2703 sub read_coincidence {
2708 if (ref ($f)) { # already open file
2711 elsif ($f ne '') { # path
2715 $file = DATA_COIN_PATH();
2718 %data = read_data_file($file);
2720 return validate_coincidence(\%data);
2728 if (ref ($f)) { # already open file
2731 elsif ($f =~ /^[0-9]+$/) { # chat ID
2732 $file = DATA_CHAT_PATH().int($&);
2734 elsif ($f ne '') { # path
2738 $file = DATA_CHAT_PATH();
2741 return read_data_file($file);
2747 (my $f, my $data) = @_;
2750 if (ref ($f)) { # already open file
2753 elsif ($f =~ /^[0-9]+$/) { # chat ID
2754 $file = DATA_CHAT_PATH().int($&);
2756 elsif ($f ne '') { # path
2760 $file = DATA_CHAT_PATH();
2765 return write_data_file($file, $data);