/[pub]/suikawiki/script/wiki.cgi
Suika

Contents of /suikawiki/script/wiki.cgi

Parent Directory Parent Directory | Revision Log Revision Log


Revision 1.5 - (hide annotations) (download)
Mon Feb 4 15:22:00 2002 UTC (22 years, 9 months ago) by wakaba
Branch: MAIN
Changes since 1.4: +7 -5 lines
implement <uri:foo:bar> format.

1 wakaba 1.2 #!/usr/bin/perl
2     use lib "../lib";
3     use CGI::Carp 'fatalsToBrowser';
4     use Algorithm::Diff qw(traverse_sequences);
5     # use strict;
6     #
7     # yukiwiki.cgi - Yet another WikiWikiWeb clone.
8     #
9     # Copyright (C) 2000,2001 by Hiroshi Yuki.
10     # <hyuki@hyuki.com>
11     # http://www.hyuki.com/yukiwiki/
12     #
13     # This program is free software; you can redistribute it and/or modify
14     # it under the terms of the GNU General Public License as published by
15     # the Free Software Foundation; either version 2 of the License, or
16     # (at your option) any later version.
17     #
18     # This program is distributed in the hope that it will be useful,
19     # but WITHOUT ANY WARRANTY; without even the implied warranty of
20     # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21     # GNU General Public License for more details.
22     #
23 wakaba 1.4 # $Id: wiki.cgi,v 1.3 2002/02/04 14:36:18 wakaba Exp $
24 wakaba 1.2 ##############################
25     my $version = "1.6.6";
26     ##############################
27     # 単独テストのときには 1 にする。
28     my $testing = 0;
29     ##############################
30     # 漢字ライブラリ
31     my $jcodelib = 'jcode.pl';
32     ##############################
33     # 保存・表示の漢字コード
34     my $kanjicode = 'euc'; # 'sjis' 'euc'
35     my $charset = 'euc-jisx0213'; # 'Shift_JIS' 'EUC-JP'
36     ##############################
37     # dbmopenが使えるなら1、使えないなら0
38     my $dbmopen = 0;
39     ##############################
40     # データベース名(.pag, .dir, .dbなどは不要)
41     # $dbmopen = 1のときはデータベース名、
42     # $dbmopen = 0のときはディレクトリ名になる。
43 wakaba 1.3 my $dbname = './wikidata';
44 wakaba 1.2 my $diffdbname = './wikidiff';
45     ##############################
46     # 修正者の氏名(自由に変更してください)
47     my $modifier = 'suika';
48     ##############################
49     # 修正者のWebページ(自由に変更してください)
50     my $modifierlink = 'http://suika.fam.cx/';
51     ##############################
52     # このページのURL
53     my $thisurl = 'wiki';
54     ##############################
55     # 開始ページ名
56     my $toppage = 'HomePage';
57     ##############################
58     # 最終更新ページ名
59     my $whatsnew = 'RecentChanges';
60     ##############################
61     # 最終更新に掲載するページ数
62     my $maxnew = 50;
63     ##############################
64     # アイコンファイル名(カラー版)
65     my $iconfile = '';
66     ##############################
67     # アイコンファイル名(モノクロ版)
68     # my $iconfile = '';
69     ##############################
70     # ページを変更したときにtouchするファイル(''なら何もしない)
71     my $touchfile = 'touch.txt';
72     ##############################
73     # プレビュー用の背景色
74     my $preview_color = '#FFCCCC';
75     ##############################
76     # 全ページのスタイル
77     my $style = <<'EOD';
78     pre, dl, ul, ol, p, blockquote { line-height:120%; }
79     a { text-decoration: none; }
80     a:link { color: #0000FF; background-color: #FFFFFF; }
81     a:visited { color: #9900CC; background-color: #FFFFFF; }
82     a:hover { text-decoration: underline; }
83     EOD
84     ##############################
85     # テキスト入力部分の大きさ
86     my $cols = 80;
87     my $rows = 20;
88     ##############################
89     my %form = ();
90     my %database = ();
91     my %diffbase = ();
92     my $diff_text = '';
93     my @diff_added = ();
94     my @diff_deleted = ();
95     my $msgrefA;
96     my $msgrefB;
97     ##############################
98     # 編集不可ページ名一覧
99     my @uneditable = ( $whatsnew );
100     ##############################
101     # リンク用の正規表現
102     # YukiWikiのリンクは2種類ある。
103     #
104     # (1) WikiName (RecentChangesとかFrontPageのようなもの)
105     # (2) BracketName ([[結城浩]]とか[[トラブルシュート]]のようなもの)
106     #
107     # ※シフトJISの2バイト目には ']' が来うるので、
108     # 文字']'を1つ多くとるようにしている。
109     #
110     my $WikiName = '([A-Z][a-z]+([A-Z][a-z]+)+)';
111     my $BracketName = '\[\[([^>\s]+?\]?)\]\]';
112    
113     # アイコン部分のタグ
114     my $IconTag = ''; #<<"EOD";
115     #<a href="http://www.hyuki.com/yukiwiki/"><img src="$iconfile"
116     # border="0" width="80" height="80" alt="[YukiWiki]" /></a>
117     #EOD
118    
119     require "$jcodelib";
120    
121     &init_form($kanjicode);
122    
123     if ($testing) {
124     %form = (
125     # 'mycmd' => 'write',
126     'mycmd' => 'read',
127     #'mycmd' => 'search',
128     #'mycmd' => 'edit',
129     'mymsg' => <<"EOD",
130     はじめまして。
131     これからいろいろ書き込みますね。
132     LinkPageも見てください。
133     TestPageはどうでしょうか。
134     どうぞよろしく。
135     http://www.hyuki.com/
136     [[結城浩]]
137     EOD
138     'mypage' => '<結城浩>',
139     'myword' => '結',
140     # '3C8C8B8FE98D5F3E' => '',
141     # 'TestPage' => '',
142     );
143     }
144     &main;
145     exit(0);
146    
147     # メイン
148     sub main {
149     &normalize_form;
150     if ($dbmopen) {
151     if (!dbmopen(%database, $dbname, 0666)) {
152     &print_error("(dbmopen) $dbname が作れません。");
153     }
154     } else {
155     if (!tie(%database, "YukiWikiDB", $dbname)) {
156     &print_error("(tie error)");
157     }
158     }
159    
160     # myspecial対応
161     foreach (keys %form) {
162     if (/^myspecial_(.*)/) {
163     $form{mycmd} = $1;
164     last;
165     }
166     }
167    
168     if ($form{mycmd} eq 'read') {
169     &do_read;
170     } elsif ($form{mycmd} eq 'preview') {
171     &do_preview;
172     } elsif ($form{mycmd} eq 'write') {
173     &do_write;
174     } elsif ($form{mycmd} eq 'edit') {
175     &do_edit;
176     } elsif ($form{mycmd} eq 'reedit') {
177     &do_reedit;
178     } elsif ($form{mycmd} eq 'search') {
179     &do_search;
180     } elsif ($form{mycmd} eq 'list') {
181     &do_list;
182     } elsif ($form{mycmd} eq 'diff') {
183     &do_diff;
184     } else {
185     $form{mypage} = $toppage;
186     &do_read;
187     }
188     if ($dbmopen) {
189     dbmclose(%database);
190     } else {
191     untie(%database);
192     }
193     }
194    
195     # ページの表示
196     sub do_read {
197     my $page_name = $form{mypage};
198     my $percent_name = &encode_percent($page_name);
199     &print_header($page_name);
200     print qq|<h1>$IconTag<a href="$thisurl?mycmd=search;myword=$percent_name">$page_name</a></h1>\n|;
201     &print_toolbar($page_name);
202     print &convert_html(&get_page($page_name));
203     &print_footer;
204     }
205    
206     # ページの編集
207     sub do_edit {
208     if (not &is_editable($form{mypage})) {
209     # 編集不可ページは表示のみ
210     &do_read;
211     return;
212     }
213     &editpage(&get_page($form{mypage}));
214     }
215    
216     # ページの再編集
217     sub do_reedit {
218     if (not &is_editable($form{mypage})) {
219     # 編集不可ページは表示のみ
220     &do_read;
221     } else {
222     &editpage($form{mymsg});
223     }
224     }
225    
226     sub editpage {
227     my $page_msg = shift;
228     my $page_name = $form{mypage};
229     my $digest = &calc_message_digest($page_msg);
230     &print_header($page_name);
231     print qq|<h1>$IconTag${page_name}の編集</h1>\n|;
232     &print_toolbar($page_name);
233     $page_msg = &escape($page_msg);
234     print <<"EOD";
235     <form action="$thisurl" method="post">
236 wakaba 1.4 <!--<input type="hidden" name="mycmd" value="preview">-->
237 wakaba 1.2 <input type="hidden" name="mypage" value="$page_name">
238     <input type="hidden" name="mydigest" value="$digest">
239     <textarea cols="$cols" rows="$rows" name="mymsg" wrap="virtual">$page_msg</textarea><br>
240 wakaba 1.4 <input type="submit" name="myspecial_preview" value="確認">
241     <input type="submit" name="myspecial_write" value="確認せず変更">
242 wakaba 1.2 </form>
243     <hr>
244     <h3>テキスト整形のルール</h3>
245    
246     <p>通常は入力した文字がそのまま出力されますが、
247     以下のルールに従ってテキスト整形を行うことができます。</p>
248    
249     <ul>
250     <li>
251     空行は段落の区切りとなります。
252    
253     <li>
254     HTMLのタグは書けません。
255    
256     <li>
257     ''強調''のようにシングルクォート二つではさむと、強調になります。
258    
259     <li>
260     '''更に強調'''のようにシングルクォート三つではさむと、更に強調になります。
261    
262     <li>
263     ----のようにマイナス4つがあると、水平線になります。
264    
265     <li>
266     *を行頭に書くと大見出しになります。
267    
268     <li>
269     **を行頭に書くと小見出しになります。
270    
271     <li>
272     -を行頭に書くと箇条書きになります。- -- --- の3レベルがあります。
273    
274     <li>
275     :を行頭に書くと用語と解説文が作れます。
276    
277     <pre>
278     :用語1:いろいろ書いた解説文1
279     :用語2:いろいろ書いた解説文2
280     :用語3:いろいろ書いた解説文3
281     </pre>
282    
283     <li>
284     http://www.hyuki.com/ のようなURLは自動的にリンクになります。
285    
286     <li>
287     YukiWikiのように大文字小文字を混ぜた英文字列を書くと、
288     YukiWikiのページ名になります。
289    
290     <li>
291     [[結城浩]]のように二重の大かっこ[[ ]]でくくった文字列を書くと、
292     YukiWikiのページ名になります。
293     大かっこの中にはスペースを含めてはいけません。
294     日本語も使えます。
295    
296     <li>
297     行頭がスペースで始まっていると、
298     その段落は整形済み扱われます。
299     プログラムを書き込むときに使うと便利です。
300    
301     <li>
302     &gt; を行頭に書くと、
303     引用文が書けます。
304     &gt; の数が多いとインデントが深くなります(3レベルまで)。
305    
306     </ul>
307     EOD
308     &print_footer;
309     }
310    
311     # ページの検索
312     sub do_search {
313     if ($form{myword}) {
314     &print_header('検索結果');
315     print qq|<h1>$IconTag$form{myword}の検索結果</h1>\n|;
316     &print_toolbar();
317     print qq|<ul>\n|;
318     my $count = 0;
319     foreach my $page_name (sort keys %database) { # sortするのは無謀かな
320     if ($database{$page_name} =~ /\Q$form{'myword'}\E/) {
321     my $encoded = &encode_percent($page_name);
322     print qq|<li><a href="$thisurl?mycmd=read;mypage=$encoded">$page_name</a>\n|;
323     $count++;
324     }
325     }
326     print qq|</ul>\n|;
327     if ($count > 0) {
328     print qq|<p><strong>$form{myword}</strong>を含むページは、上に示す<strong>$count</strong> ページです。</p>\n|;
329     } else {
330     print qq|<p><strong>$form{myword}</strong>を含むページは見つかりません。</p>\n|;
331     }
332     } else {
333     &print_header('単語検索');
334     print qq|<h1>$IconTag単語検索</h1>\n|;
335     &print_toolbar();
336     }
337     print <<"EOD";
338     <p>
339     <form action="$thisurl" method="post">
340     <input type="hidden" name="mycmd" value="search">
341     <input type="text" name="myword" size="20" value="$form{myword}">
342     <input type="submit" value="単語検索">
343     </form>
344     </p>
345     EOD
346     &print_footer;
347     }
348    
349     # ページの一覧
350     sub do_list {
351     &print_header('ページ一覧');
352     print qq|<h1>$IconTag ページ一覧</h1>\n|;
353     &print_toolbar();
354     print qq|<ul>\n|;
355     foreach my $page_name (sort keys %database) { # sortするのは無謀かな
356     my $encoded = &encode_percent($page_name);
357     print qq|<li><a href="$thisurl?mycmd=read;mypage=$encoded">$page_name</a>\n|
358     }
359     print qq|</ul>\n|;
360     &print_footer;
361     }
362    
363     # プレビュー
364     sub do_preview {
365     my $page_name = $form{mypage};
366     my $escapedmsg = &escape($form{mymsg});
367     &print_header($page_name);
368     print qq|<h1>$IconTag${page_name}のプレビュー</h1>\n|;
369     &print_toolbar($page_name);
370     # local $percent_name = &encode_percent($page_name);
371     print qq|<p>以下のプレビューを確認して、よければページ下部のボタンで更新してください。</p>\n|;
372     if ($form{mymsg}) {
373     print qq|<table width="100%" bgcolor="$preview_color" ><tr><td>\n|;
374     # print &convert_html($escapedmsg);
375     print &convert_html($form{mymsg});
376     print qq|</td></tr></table>\n|;
377     } else {
378     print qq|<p>(ページの内容は空です。更新するとこのページは<strong>削除</strong>されます。)</p>\n|;
379     }
380     &print_preview_buttons($page_name, $escapedmsg, $form{mydigest});
381     &print_footer;
382     }
383    
384     sub print_preview_buttons {
385     my ($page_name, $escapedmsg, $digest) = @_;
386     print <<"EOD";
387     <form action="$thisurl" method="post">
388     <textarea cols="$cols" rows="$rows" name="mymsg" wrap="virtual">$escapedmsg</textarea>
389     <br />
390     <input type="hidden" name="mypage" value="$page_name">
391     <input type="hidden" name="mydigest" value="$digest">
392     <input type="submit" name="myspecial_preview" value="再度プレビュー">
393     <input type="submit" name="myspecial_write" value="ページの更新">
394     </form>
395     EOD
396     }
397    
398     # 書き込む
399     sub do_write {
400     if (not &is_editable($form{mypage})) {
401     # 編集不可ページは表示のみ
402     &do_read;
403     return;
404     }
405    
406     my $page_name = $form{mypage};
407    
408     # digestを使って、更新の衝突チェック
409     my $original_digest = &calc_message_digest(&get_page($page_name));
410     if ($form{mydigest} ne $original_digest) {
411     &print_header($page_name);
412     print qq|<h1>$IconTag${page_name}で【更新の衝突】が起きました</h1>\n|;
413     print <<"EOD";
414     <p>あなたがこのページを編集している間に、
415     他の人が同じページを更新してしまったようです。
416     </p><p>
417     以下に、あなたの編集したテキストがありますので、
418     あなたの編集内容が失われないように、
419     いますぐ、メモ帳などにコピー&ペーストしてください。
420     </p><p>
421     コピー&ペーストが済んでから、
422     最新の内容を見て再度編集し直してください。
423     最新の内容は
424     <a target="_blank" href="$thisurl?mycmd=read;mypage=$form{mypage}">$form{mypage}</a>
425     で見ることができます。
426     </p>
427     EOD
428     # &print_toolbar($page_name);
429     &print_preview_buttons($page_name, &escape($form{mymsg}), $form{mydigest});
430     &print_footer;
431     return;
432     }
433    
434     # diff生成
435     {
436     &opendiff;
437     my @msg1 = split(/\n/, &get_page($page_name));
438     my @msg2 = split(/\n/, $form{mymsg});
439     $msgrefA = \@msg1;
440     $msgrefB = \@msg2;
441     &diff_check;
442     $diffbase{$form{mypage}} = $diff_text;
443     $diff_text = '';
444     &closediff;
445     }
446    
447     &print_header($page_name);
448     &set_page($page_name, $form{mymsg});
449     if ($form{mymsg}) {
450     print qq|<h1>$IconTag${page_name}を更新しました</h1>\n|;
451     &print_toolbar($page_name);
452     print &convert_html(&get_page($page_name));
453     } else {
454     print qq|<h1>$IconTag${page_name}を削除しました</h1>\n|;
455     &print_toolbar($page_name);
456     print qq|<p>${page_name}を削除しました。</p>\n|;
457     }
458     &print_footer;
459     # 更新されたのでタッチしておく。
460     if ($touchfile) {
461     open(FILE, "> $touchfile");
462     print FILE "\n";
463     close(FILE);
464     }
465     }
466    
467     # ページの変更点
468     sub do_diff {
469     if (not &is_editable($form{mypage})) {
470     # 編集不可ページは表示のみ
471     &do_read;
472     return;
473     }
474     &opendiff;
475     &print_header($form{mypage} . 'の変更点');
476     print qq|<h1>$IconTag <a href="$thisurl?mycmd=read&mypage=$form{mypage}">$form{mypage}</a>の変更点</h1>\n|;
477     &print_toolbar();
478     $_ = &escape($diffbase{$form{mypage}});
479     print <<"EOD";
480     <ul>
481     <li>追加された行は<ins>青色</ins>です。
482     <li>削除された行は<del>赤色</del>です。
483     <li><a href="$thisurl?mycmd=read;mypage=$form{mypage}">$form{mypage}</a>へ行く。
484     </ul>
485     <hr />
486     EOD
487     print qq|<pre>\n|;
488     foreach (split(/\n/, $_)) {
489     if (/^\+(.*)/) {
490     print qq|<ins>$1</ins>\n|;
491     } elsif (/^\-(.*)/) {
492     print qq|<del>$1</del>\n|;
493     } elsif (/^\=(.*)/) {
494     print qq|$1\n|;
495     } else {
496     print qq|??? $_\n|;
497     }
498     }
499     print qq|</pre>\n|;
500     &print_footer;
501     &closediff;
502     }
503    
504     sub opendiff {
505     if ($dbmopen) {
506     if (!dbmopen(%diffbase, $diffdbname, 0666)) {
507     &print_error("(dbmopen) $diffdbname が作れません。");
508     }
509     } else {
510     if (!tie(%diffbase, "YukiWikiDB", $diffdbname)) {
511     &print_error("(tie error)");
512     }
513     }
514     }
515    
516     sub closediff {
517     if ($dbmopen) {
518     dbmclose(%diffbase);
519     } else {
520     untie(%diffbase);
521     }
522     }
523    
524     # フォームからの情報を連想配列 %form に入れる
525     # &init_form('euc');
526     sub init_form {
527     my ($charcode) = @_;
528     my $query;
529     if ($ENV{REQUEST_METHOD} =~ /^post$/i) {
530     read(STDIN, $query, $ENV{CONTENT_LENGTH});
531     } else {
532     $query = $ENV{QUERY_STRING};
533     }
534     my @assocarray = split(/[&;]/, $query);
535     foreach my $assoc (@assocarray) {
536     my ($property, $value) = split(/=/, $assoc);
537     $value =~ tr/+/ /;
538     $value =~ s/%([A-Fa-f0-9][A-Fa-f0-9])/pack("C", hex($1))/eg;
539     &jcode::convert(\$value, $charcode);
540     $form{$property} = $value;
541     }
542     }
543    
544     # エラーページを出力する
545     sub print_error {
546     my ($msg) = @_;
547     &print_header('Error');
548     print "<h1>$IconTag$msg</h1></body></html>";
549     exit(0);
550     }
551    
552     sub escape {
553     my ($line) = shift;
554     $line =~ s|<|&lt;|g;
555     $line =~ s|>|&gt;|g;
556     $line =~ s|"|&quot;|g;
557     # $line =~ s|\&|&amp;|g;
558     return $line;
559     }
560    
561     sub inline {
562     my ($line) = shift;
563     $line = &escape($line);
564     $line =~ s|'''([^']+?)'''|<strong>$1</strong>|g;
565     $line =~ s|''([^']+?)''|<em>$1</em>|g;
566     $line =~ s!
567     (
568 wakaba 1.5 (<(mailto|http|https|ftp|urn):[\x21-\x7E]*)> # Direct http://...
569 wakaba 1.2 |
570     ($WikiName) # LocalLinkLikeThis
571     |
572     ($BracketName) # [[日本語リンク]]
573     )
574     !
575     &make_link($1)
576     !gex;
577     return $line;
578     }
579    
580     # ページのタイトルからページの内容を得る
581     sub get_page {
582     my $page_name = shift;
583     return $database{$page_name};
584     }
585    
586     # ページの内容を与える
587     # &set_page($title, $txt)
588     sub set_page {
589     # ページを更新する
590     my $title = $_[0];
591     $database{$title} = $_[1];
592     # 空ページなら削除する
593     unless ($database{$title}) {
594     delete $database{$title};
595     }
596     # RecentChangesを更新する
597     my $delim = ' - ';
598     my @pages = split(/\n/, $database{$whatsnew});
599     my $datestr = &get_current_datestr;
600     unshift(@pages, qq|-$datestr$delim$title|);
601     # 同一ページの更新は最新のもののみにし、
602     # 存在しないページはスキップする。
603     my %count;
604     my @newpages;
605     foreach my $line (@pages) {
606     my ($prefix, $title) = split(/$delim/, $line);
607     $count{$title}++;
608     if ($count{$title} == 1 and exists($database{$title})) {
609     push(@newpages, qq|$prefix - $title|);
610     }
611     }
612     # ここで本当に更新
613     $database{$whatsnew} = join("\n", splice(@newpages, 0, $maxnew));
614     }
615    
616     # ページのヘッダを出力
617     sub print_header {
618     my $title = shift;
619     print <<"EOD";
620     Content-type: text/html
621    
622     <html><head>
623     <title>$title</title>
624     <style type="text/css">
625     $style
626     </style>
627     </head>
628     <body>
629     EOD
630     }
631    
632     # ツールバーを出力
633     sub print_toolbar {
634     my $page_name = shift;
635     my $percent_name = &encode_percent($page_name);
636     my $editlink = '';
637     if ($page_name ne '' and &is_editable($page_name)) {
638     $editlink = <<"EOD";
639     <a href="$thisurl?mycmd=edit;mypage=$percent_name">編集</a> |
640     <a href="$thisurl?mycmd=diff;mypage=$percent_name">差分</a> |
641     EOD
642     }
643     print <<"EOD";
644     <p>
645     [
646     <a href="$thisurl?mycmd=read;mypage=$toppage">トップ</a> |
647     <a href="$thisurl?mycmd=list">一覧</a> |
648     $editlink
649     <a href="$thisurl?mycmd=search">単語検索</a> |
650     <a href="$thisurl?mycmd=read;mypage=$whatsnew">最終更新</a>
651     ]
652     </p>
653     EOD
654     }
655    
656     # ページのフッタを出力
657     sub print_footer {
658     print <<"EOD";
659     <address>
660     <a href="http://www.hyuki.com/yukiwiki/">YukiWiki</a> 1.6.6 Copyright (C) 2000,2001 by <a href="http://www.hyuki.com/">Hiroshi Yuki.</a>
661     + <a href="$modifierlink">$modifier</a> ${version}.
662     [<a href="/" title="このサーバーの首頁">/</a>
663     <a href="/map" title="このサーバーの案内">地図</a>
664     <a href="/search/" title="このサーバーの検索">検索</a>]
665     </address>
666     </body></html>
667     EOD
668     }
669    
670     # URLやページの名前からリンクを作る
671     sub make_link {
672     my $name = shift;
673 wakaba 1.5 if ($name =~ /^(http|https|ftp).*?(\.png|\.jpeg|\.jpg)?$/) {
674 wakaba 1.2 if ($2) {
675     return qq|<a href="$name"><img border="0" src="$name" /></a>|;
676     } else {
677 wakaba 1.5 return qq|&lt;<a href="$name">$name</a>&gt;|;
678 wakaba 1.2 }
679     } elsif ($name =~ /^mailto:(.*)/) {
680     my $address = $1;
681 wakaba 1.5 return qq|&lt;<a href="$name">$address</a>&gt;|;
682     } elsif ($name =~ /^urn:[0-9A-Za-z_:-]+/) {
683     return qq|&lt;<a href="/uri-res/N2L?${name}">$name</a>&gt;|;
684 wakaba 1.2 } elsif ($database{$name}) {
685     my $percent_name = &encode_percent($name);
686     return qq|<a href="$thisurl?mycmd=read;mypage=$percent_name">$name</a>|;
687     } else {
688     my $percent_name = &encode_percent($name);
689     return qq|$name<a href="$thisurl?mycmd=edit;mypage=$percent_name">?</a>|;
690     }
691     }
692    
693     # %xx の形式にエンコードする
694     # これは、
695     # http://www.hyuki.com/yukiwiki/yukiwiki.cgi?mycmd=read&mypage=%3C%8C%8B%8F%E9%8D_%3E
696     # という形式のために使われる。
697     # '<結城浩>' → '%3C%8C%8B%8F%E9%8D_%3E'
698     sub encode_percent {
699     my $name = shift;
700     my $encoded = '';
701     foreach my $ch (split(//, $name)) {
702     if ($ch =~ /[A-Za-z0-9_]/) {
703     $encoded .= $ch;
704     } else {
705     $encoded .= '%' . sprintf("%02X", ord($ch));
706     }
707     }
708     return $encoded;
709     }
710    
711     # テキスト本体をHTMLに変換する
712     sub convert_html {
713     my ($txt) = shift;
714     my (@txt) = split(/\n/, $txt);
715     foreach (@txt) {
716     chomp;
717     if (/^\*\*(.*)/) {
718     push(@result, splice(@saved), '<h3>' . &inline($1) . '</h3>');
719     } elsif (/^\*(.*)/) {
720     push(@result, splice(@saved), '<h2>' . &inline($1) . '</h2>');
721     } elsif (/^----/) {
722     push(@result, splice(@saved), '<hr>');
723     } elsif (/^(-{1,3})(.*)/) {
724     &back_push('ul', length($1));
725     push(@result, '<li>' . &inline($2) . '</li>');
726     } elsif (/^:([^:]+):(.*)/) {
727     &back_push('dl', 1);
728     push(@result, '<dt>' . &inline($1) . '</dt>', '<dd>' . &inline($2) . '</dd>');
729     } elsif (/^(>{1,3})(.*)/) {
730     &back_push('blockquote', length($1));
731     push(@result, &inline($2));
732     } elsif (/^\s*$/) {
733     push(@result, splice(@saved));
734     unshift(@saved, "</p>");
735     push(@result, "<p>");
736     } elsif (/^(\s+.*)$/) {
737     &back_push('pre', 1);
738     push(@result, &escape($1)); # Not &inline, but &escape
739     } else {
740     push(@result, &inline($_));
741     }
742     }
743     push(@result, splice(@saved));
744     return join("\n", @result);
745     }
746    
747     # &back_push($tag, $count)
748     # $tagのタグを$levelレベルまで詰める。
749     sub back_push {
750     my ($tag, $level) = @_;
751     while (@saved > $level) {
752     push(@result, shift(@saved));
753     }
754     if ($saved[0] ne "</$tag>") {
755     push(@result, splice(@saved));
756     }
757     while (@saved < $level) {
758     unshift(@saved, "</$tag>");
759     push(@result, "<$tag>");
760     }
761     }
762    
763     # 編集可能ページか?
764     sub is_editable {
765     my ($pagename) = @_;
766     foreach (@uneditable) {
767     if ($pagename eq $_) {
768     return 0;
769     }
770     }
771     if (&is_valid_name($pagename)) {
772     return 1;
773     }
774     return 0;
775     }
776    
777     # Validな名前か?
778     sub is_valid_name {
779     my ($pagename) = @_;
780     if ($pagename =~ /^$WikiName$/) {
781     return 1;
782     } elsif ($pagename =~ /^$BracketName$/) {
783     return 1;
784     } else {
785     return 0;
786     }
787     }
788    
789     # 現在時刻の文字列を得る
790     sub get_current_datestr {
791     my (@wdays) = ( "日", "月", "火", "水", "木", "金", "土" );
792     my ($sec, $min, $hour, $mday, $mon, $year, $wday) = localtime(time);
793     return sprintf("%4d-%02d-%02d (%s) %02d:%02d:%02d",
794     $year + 1900, $mon + 1, $mday, $wdays[$wday], $hour, $min, $sec);
795     }
796    
797     # URL?SomePageや、
798     # URL?[[結城浩]]の形式だった場合、(not yet)
799     # 強制的にmycmdをreadにして$formの内容を設定する。
800     sub normalize_form {
801     foreach my $key (keys %form) {
802     if ($key =~ /^$WikiName$/) {
803     $form{mycmd} = 'read';
804     $form{mypage} = $1;
805     last;
806     } elsif ($key =~ /^$BracketName$/) {
807     $form{mycmd} = 'read';
808     $form{mypage} = $1;
809     last;
810     }
811     }
812     }
813    
814     # 変換テストを行なうときのサンプル
815     sub print_sample {
816     my $txt = &convert_html(<<"EOD");
817     *大見出し1
818     **小見出し1-1
819     -項目1
820     -項目2
821     -項目3
822     段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
823     段落1段落1段落1段落1段落1段落''強調''1段落1段落1段落1段落1段落1
824     段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
825    
826     段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2
827     段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2
828     段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2
829     **小見出し1-2
830     :用語1:いろいろ書いた解説文1と''強調単語''
831     段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
832     段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
833     段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
834     :用語2:いろいろ書いた解説文2
835     :用語3:いろいろ書いた解説文3
836     ----
837     *大見出し2
838     **小見出し2-1
839 wakaba 1.5 &lt;http://suika.fam.cx/&gt;
840 wakaba 1.2 **小見出し2-2
841    
842     [[結城浩]]
843    
844     段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
845     段落1段落1段落1段落'''強調'''1段落1段落1段落1段落1段落1段落1段落1
846     段落1段落1段落1段落'''''強い強調'''''1段落1段落1段落1段落1段落1段落1段落1段落1
847     >段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2
848     >段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2
849     >段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2
850    
851     レベル0レベル0レベル0レベル0レベル0レベル0
852    
853     >レベル1
854     >レベル1
855     >レベル1
856     >>レベル2
857     >>レベル2
858     >>レベル2
859     >>>レベル3
860     -はろ1
861     --はろ2
862     ろろろろ2
863     ---はろ3
864     --はろ2
865     ---はろ3
866     --はろ2
867     ---はろ3
868     >>>レベル3
869     >>>レベル3
870     >>>レベル3
871     EOD
872     print $txt;
873     exit;
874     }
875    
876     sub diff_check {
877     traverse_sequences(
878     $msgrefA, $msgrefB,
879     {
880     MATCH => \&df_match,
881     DISCARD_A => \&df_delete,
882     DISCARD_B => \&df_add,
883     }
884     );
885     &diff_flush;
886     }
887    
888     sub diff_flush {
889     $diff_text .= join('', map { "-$_\n" } splice(@diff_deleted));
890     $diff_text .= join('', map { "+$_\n" } splice(@diff_added));
891     }
892    
893     sub df_match {
894     my ($a, $b) = @_;
895     &diff_flush;
896     $diff_text .= "=$msgrefA->[$a]\n";
897     }
898    
899     sub df_delete {
900     my ($a, $b) = @_;
901     push(@diff_deleted, $msgrefA->[$a]);
902     }
903    
904     sub df_add {
905     my ($a, $b) = @_;
906     push(@diff_added, $msgrefB->[$b]);
907     }
908    
909     sub calc_message_digest { # You have to use MD5...
910     my $text = shift;
911     my @text = split(//, $text);
912     my $len = length($text);
913     my $checksum = 0;
914     foreach (@text) {
915     $checksum += ord($_);
916     $checksum = ($checksum * 2) % 65536 + (($checksum & 32768) ? 1 : 0); # 16bit rotate
917     }
918     return "$len:$checksum";
919     }
920    
921     # Definition of YukiWikiDB
922     package YukiWikiDB;
923    
924     my $debug = 1;
925    
926     # Constructor
927     sub new {
928     return shift->TIEHASH(@_);
929     }
930    
931     # tying
932     sub TIEHASH {
933     my ($class, $dbname) = @_;
934     my $self = {
935     dir => $dbname,
936     keys => [],
937     };
938     if (not -d $self->{dir}) {
939     if (!mkdir($self->{dir}, 0777)) {
940     print "mkdir(" . $self->{dir} . ") fail\n" if ($debug);
941     return undef;
942     }
943     }
944     return bless($self, $class);
945     }
946    
947     # Store
948     sub STORE {
949     my ($self, $key, $val) = @_;
950     my $file = &make_filename($self, $key);
951     if (open(FILE,"> $file")) {
952     binmode(FILE);
953     print FILE $val;
954     close(FILE);
955     return $self->{$key} = $val;
956     } else {
957     print "$file create error.";
958     }
959     }
960    
961     # Fetch
962     sub FETCH {
963     my ($self, $key) = @_;
964     my $file = &make_filename($self, $key);
965     if (open(FILE, $file)) {
966     local $/;
967     $self->{$key} = <FILE>;
968     close(FILE);
969     }
970     return $self->{$key};
971     }
972    
973     # Exists
974     sub EXISTS {
975     my ($self, $key) = @_;
976     my $file = &make_filename($self, $key);
977     return -e($file);
978     }
979    
980     # Delete
981     sub DELETE {
982     my ($self, $key) = @_;
983     my $file = &make_filename($self, $key);
984     unlink $file;
985     return delete $self->{$key};
986     }
987    
988     sub FIRSTKEY {
989     my ($self) = @_;
990     opendir(DIR, $self->{dir}) or die $self->{dir};
991     @{$self->{keys}} = grep /\.txt$/, readdir(DIR);
992     foreach my $name (@{$self->{keys}}) {
993     $name =~ s/\.txt$//;
994     $name =~ s/[0-9A-F][0-9A-F]/pack("C", hex($&))/eg;
995     }
996     return shift @{$self->{keys}};
997     }
998    
999     sub NEXTKEY {
1000     my ($self) = @_;
1001     return shift @{$self->{keys}};
1002     }
1003    
1004     sub make_filename {
1005     my ($self, $key) = @_;
1006     my $enkey = '';
1007     foreach my $ch (split(//, $key)) {
1008     $enkey .= sprintf("%02X", ord($ch));
1009     }
1010     return $self->{dir} . "/$enkey.txt";
1011     }
1012     __END__
1013    
1014     =head1 NAME
1015    
1016     YukiWiki - 自由にページを追加・削除・編集できるWebページ構築CGI
1017    
1018     Copyright (C) 2000,2001 by Hiroshi Yuki.
1019     結城浩 <hyuki@hyuki.com>
1020     http://www.hyuki.com/
1021     http://www.hyuki.com/yukiwiki/
1022    
1023     =head1 SYNOPSIS
1024    
1025     http://www.hyuki.com/yukiwiki/yukiwiki.cgi
1026    
1027     =head1 DESCRIPTION
1028    
1029     YukiWiki(結城ウィキィ)は参加者が自由にページを追加・削除・編集できる
1030     不思議なWebページ群を作るCGIです。
1031     Webで動作する掲示板とちょっと似ていますが、
1032     Web掲示板が単にメッセージを追加するだけなのに対して、
1033     YukiWikiは、Webページ全体を自由に変更することができます。
1034    
1035     YukiWikiは、Cunningham & CunninghamのWikiWikiWebの
1036     仕様を参考にして独自に作られました。
1037    
1038     YukiWikiはPerlで書かれたCGIスクリプトとして実現されていますので、
1039     Perlが動作するWebサーバならば比較的容易に設置できます。
1040    
1041     あとはdbmopenが使える環境ならば設置できます(Version 1.5.0以降ならdbmopenが使えなくても設置できます)。
1042    
1043    
1044     YukiWikiはフリーソフトです。
1045     ご自由にお使いください。
1046    
1047     =head1 設置方法
1048    
1049     =head2 入手
1050    
1051     YukiWikiの最新版は、
1052     http://www.hyuki.com/yukiwiki/
1053     から入手できます。
1054    
1055     =head2 ファイル一覧
1056    
1057     readme.txt ドキュメント
1058     yukiwiki.cgi YukiWiki本体
1059     yukiwiki.gif ロゴ(カラー版)
1060     yukimono.gif ロゴ(モノクロ版)
1061     jcode.pl 漢字コードライブラリ
1062    
1063     =head2 インストール
1064    
1065     =over 4
1066    
1067     =item 1.
1068    
1069     アーカイブを解く。
1070    
1071     =item 2.
1072    
1073     yukiwiki.cgiのはじめの方にある設定を確認します。
1074     通常は何もしなくてよいが、
1075     はじめは$touchfileを''にした方がよいでしょう。
1076    
1077     =item 3.
1078    
1079     yukiwiki.cgiとjcode.plを同じところに設置します。
1080    
1081     =item 4.
1082    
1083     サイズ0のyukiwiki.dbというファイルを設置します。
1084     (Perlシステムによってはyukiwiki.pag, yukiwiki.dir)
1085    
1086     =item 5.
1087    
1088     yukiwiki.cgiにブラウザからアクセスします。
1089    
1090     =back
1091    
1092     =head2 パーミッション
1093    
1094     ファイル パーミッション 転送モード
1095     yukiwiki.cgi 755 ASCII
1096     yukiwiki.gif 644 BINARY
1097     yukimono.gif 644 BINARY
1098     jcode.pl 644 ASCII
1099    
1100     $dbmopen = 1; にした場合:
1101     yukiwiki.db 666 BINARY
1102     (yukiwiki.pag, yukiwiki.dirの場合もあり)
1103    
1104     $dbmopen = 0; にした場合: (カレントディレクトリを777にしておく)
1105     . 777 (転送しない)
1106    
1107     =head1 データのバックアップ方法
1108    
1109     $dbmopen = 1;の場合は、
1110     データはすべてyukiwiki.db(.dir, .pag)に入る。
1111     これをバックアップすればよい。
1112    
1113     $dbmopen = 0;の場合は、
1114     yukiwikiというディレクトリができるので、
1115     これ以下をバックアップすればよい。
1116    
1117     =head1 新しいページの作り方
1118    
1119     =over 4
1120    
1121     =item 1.
1122    
1123     まず、適当なページ(例えばFrontPage)を選び、
1124     ページの下にある「編集」リンクをたどります。
1125    
1126     =item 2.
1127    
1128     するとテキスト入力ができる状態になるので、
1129     そこにNewPageのような単語
1130     (大文字小文字混在している英文字列)
1131     を書いて「保存」します。
1132    
1133     =item 3.
1134    
1135     保存すると、FrontPageのページが書き換わり、
1136     あなたが書いたNewPageという文字列の後ろに ? というリンクが表示されます。
1137     この ? はそのページがまだ存在しないことを示す印です。
1138    
1139     =item 4.
1140    
1141     その ? をクリックすると新しいページNewPageができますので、
1142     あなたの好きな文章をその新しいページに書いて保存します。
1143    
1144     =item 5.
1145    
1146     NewPageページができるとFrontPageの ? は消えて、リンクとなります。
1147    
1148     =back
1149    
1150     =head1 テキスト整形のルール
1151    
1152     =over 4
1153    
1154     =item *
1155    
1156     連続した複数行はフィルされて表示されます。
1157    
1158     =item *
1159    
1160     空行は段落C<< <p> >>の区切りとなります。
1161    
1162     =item *
1163    
1164     HTMLのタグは書けません。
1165    
1166     =item *
1167    
1168     B<''ボールド''>のようにシングルクォート二つではさむと、
1169     ボールドC<< <b> >>になります。
1170    
1171     =item *
1172    
1173     B<'''イタリック'''>のようにシングルクォート三つではさむと、
1174     イタリックC<< <i> >>になります。
1175    
1176     =item *
1177    
1178     B<---->のようにマイナス4つがあると、
1179     水平線C<< <hr> >>になります。
1180    
1181     =item *
1182    
1183     行をB<*>ではじめると、
1184     大見出しC<< <h2> >>になります。
1185    
1186     =item *
1187    
1188     行をB<**>ではじめると、
1189     小見出しC<< <h3> >>になります。
1190    
1191     =item *
1192    
1193     行をマイナス-ではじめると、
1194     箇条書きC<< <ul> >>になります。
1195     マイナスの数が増えるとレベルが下がります(3レベルまで)
1196    
1197     -項目1
1198     --項目1-1
1199     --項目1-2
1200     -項目2
1201     -項目3
1202     --項目3-1
1203     ---項目3-1-1
1204     ---項目3-1-2
1205     --項目3-2
1206    
1207     =item *
1208    
1209     コロンを使うと、
1210     用語と解説文のリストC<< <dl> >>が書けます。
1211    
1212     :用語1:いろいろ書いた解説文1
1213     :用語2:いろいろ書いた解説文2
1214     :用語3:いろいろ書いた解説文3
1215    
1216     =item *
1217    
1218     リンク
1219    
1220     =over 4
1221    
1222     =item *
1223    
1224     LinkToSomePageやFrontPageのように、
1225     英単語の最初の一文字を大文字にしたものが
1226     二つ以上連続したものはYukiWikiのページ名となり、
1227     それが文章中に含まれるとリンクになります。
1228    
1229     =item *
1230    
1231     http://www.hyuki.com/ のようなURLは自動的にリンクになります。
1232    
1233     =item *
1234    
1235     二重の大かっこ[[ ]]でくくった文字列も、
1236     YukiWikiのページ名になります。
1237     大かっこの中にはスペースを含めてはいけません。
1238     日本語も使えます。
1239    
1240     =back
1241    
1242     =item *
1243    
1244     行頭がスペースやタブで始まっていると、
1245     それは整形済みの段落C<< <pre> >>として扱われます。
1246     プログラムの表示などに使うと便利です。
1247    
1248    
1249     =item *
1250    
1251     行を > ではじめると、
1252     引用文C<< <blockquote> >>が書けます。
1253     >の数が多いとインデントが深くなります(3レベルまで)。
1254    
1255     >文章
1256     >文章
1257     >>さらなる引用
1258     >文章
1259    
1260     =back
1261    
1262     =head1 更新履歴
1263    
1264     =over 4
1265    
1266     =item *
1267    
1268     2001年10月20日、Version 1.6.6。
1269    
1270     更新の衝突対策。
1271     元ページの簡単なチェックサムを取っておき、
1272     更新前にチェックサムを比較する。
1273     修正個所はdigestという文字列を検索すれば分かる。
1274     本来はMD5などでちゃんとやった方がいいのだけれど。
1275    
1276     衝突時に表示されるメッセージなどは「極悪」さんのページを参考にした。
1277    
1278     =item *
1279    
1280     2001年10月17日、Version 1.6.5。
1281    
1282     プレビュー画面で、更新ボタンを押したときに送信される
1283     メッセージの内容をinput要素のtype="hidden"を使って埋め込むのをやめる。
1284     代わりに、textarea要素を使う。
1285     再プレビュー用にmyspecial_を導入。でもきれいな対策ではない。
1286    
1287     =item *
1288    
1289     2001年8月30日、Version 1.6.4。
1290    
1291     URLでダイレクトにページ名を指定しても、
1292     $WikiNameと$BracketName以外のページを作れないようにした。
1293     (is_valid_nameとis_editable参照)。
1294    
1295     =item *
1296    
1297     2001年8月30日、Version 1.6.3。
1298    
1299     RecentChangesを編集・再編集不可とした。
1300     編集不可ページは@uneditableにページ名を入れる。
1301    
1302     =item *
1303    
1304     2001年2月25日、Version 1.6.1, 1.6.2。
1305    
1306     差分機能のバグ修正。
1307     do_previewで'>'が扱えないバグを修正
1308     (ユーザからの指摘)。
1309    
1310     =item *
1311    
1312     2001年2月22日、Version 1.6.0。
1313     差分機能を実装した。
1314    
1315     =item *
1316    
1317     2001年2月19日、Version 1.5.4。
1318     画像ファイルへのリンクは画像にしてみた。
1319    
1320     =item *
1321    
1322     2001年2月19日、Version 1.5.3。
1323     RecentChangesの中に削除したページがあるのをやめた。
1324     use strict;で引っかかる部分を少し整理(完全ではない)。
1325    
1326     =item *
1327    
1328     2001年2月16日、Version 1.5.2。
1329     textareaに表示およびプレビューする前に < や > を &lt; や &gt; に変換した
1330     (do_preview, editpage, print_preview_buttons)。
1331    
1332     =item *
1333    
1334     2000年12月27日、Version 1.5.1。
1335     プレビュー画面を整理した。
1336    
1337     =item *
1338    
1339     2000年12月22日、Version 1.5.0。
1340     全体的にずいぶん書き直した。
1341     一覧を別途作成するようにした(do_list)。
1342     書き込む前に確認画面を出すようにした(do_preview)。
1343     テキストの書き方を編集画面に入れた(do_edit, do_reedit)。
1344     WhatsNew→RecentChanges、TopPage→FrontPageに変更した。
1345    
1346     =item *
1347    
1348     2000年12月20日、Version 1.1.0。
1349     tieを利用して、dbmopenが使えない場合でも動作するように修正。
1350     利用者の1人である「極悪」さんから
1351     送っていただいたコードを元にしています。
1352    
1353     =item *
1354    
1355     2000年9月5日、Version 1.0.2。
1356     <body color=...> → <body bgcolor=...>
1357     利用者からの指摘による。感謝。
1358    
1359     =item *
1360    
1361     2000年8月6日、Version 1.0.1を公開。
1362     C MAGAZINE(ソフトバンクパブリッシング)
1363     2000年10月号連載記事向け公開版。
1364     [[ ]] の最後が「望」のようにシフトJISで
1365     0x5Dになる場合の回避を行なった。
1366    
1367     =item *
1368    
1369     2000年8月5日、Version 1.0.0を公開。
1370    
1371     =item *
1372    
1373     2000年7月23日、Version 0.82を公開。
1374     編集時のリンクミス。
1375     <textarea>の属性変更。
1376    
1377     =item *
1378    
1379     2000年7月22日、Version 0.81を公開。
1380     ロゴを組み込む。
1381    
1382     =item *
1383    
1384     2000年7月21日、Version 0.80を公開。
1385     PODをCGI中に書き込む。
1386    
1387     =item *
1388    
1389     2000年7月19日、Version 0.70を公開。
1390     '''イタリック'''や、--、---、>>、>>>などを実装。
1391    
1392     =item *
1393    
1394     2000年7月18日、Version 0.60を公開。
1395     *太字*を''太字''に変更
1396    
1397     =item *
1398    
1399     2000年7月17日、Version 0.50を公開。
1400    
1401     =item *
1402    
1403     2000年7月17日、さらにいろいろ追加する。
1404    
1405     =item *
1406    
1407     2000年7月16日、いろいろ追加。
1408    
1409     =item *
1410    
1411     2000年7月15日、公開。
1412    
1413     =back
1414    
1415     =head1 TODO
1416    
1417     - テキスト表示モード
1418     - Charsetを明示。
1419     - textarea中の閉じタグ対応
1420     - メニューの英語表記付記
1421     - プレビューのボタンで、mymsgをinputのvalueに入れているが、改行をそのままvalueにいれてはいけないのではないか。
1422     - 「再編集」の機能はブラウザの back で充分ではないか。プレビューはもっとシンプルに。
1423     - ページタイトル(Wikiname)が検索にかかるようにする。
1424     - InterWiki風の機能「URLを隠しつつ、リンクを張る」
1425    
1426     =head1 作者
1427    
1428     Copyright (C) 2000 by Hiroshi Yuki.
1429     結城浩 <hyuki@hyuki.com>
1430     http://www.hyuki.com/
1431     http://www.hyuki.com/yukiwiki/
1432    
1433     質問、意見、バグ報告は hyuki@hyuki.com にメールしてください。
1434    
1435     =head1 配布条件
1436    
1437     YukiWikiは、
1438     GNU General Public Licenseにて公開します。
1439    
1440     YukiWikiはフリーソフトです。
1441     ご自由にお使いください。
1442     自分好みのYukiWikiが作れるようにシンプルにしてあります。
1443    
1444     =head1 謝辞
1445    
1446     本家のWikiWikiを作ったCunningham & Cunningham, Inc.に
1447     感謝します。
1448    
1449     YukiWikiを楽しんで使ってくださる
1450     ネット上の方々に感謝します。
1451    
1452     YukiWikiのロゴをデザインしてくださった橋本礼奈さん
1453     http://city.hokkai.or.jp/~reina/
1454     に感謝します。
1455    
1456     tieを使った版の元になるコードを送ってくださった
1457     「極悪」さんに感謝します。
1458    
1459     =head1 参照リンク
1460    
1461     =over 4
1462    
1463     =item *
1464    
1465     YukiWikiホームページ
1466     http://www.hyuki.com/yukiwiki/
1467    
1468     =item *
1469    
1470     本家のWikiWiki
1471     http://c2.com/cgi/wiki?WikiWikiWeb
1472    
1473     =item *
1474    
1475     本家のWikiWikiの作者(Cunningham & Cunningham, Inc.)
1476     http://c2.com/
1477    
1478     =item *
1479    
1480     YukiWikiのロゴデザインをしてくださった橋本礼奈さんのページ
1481     http://city.hokkai.or.jp/~reina/
1482    
1483     =back
1484    
1485     =cut

admin@suikawiki.org
ViewVC Help
Powered by ViewVC 1.1.24