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

Contents of /suikawiki/script/wiki.cgi

Parent Directory Parent Directory | Revision Log Revision Log


Revision 1.8 - (hide annotations) (download)
Sun Mar 24 00:51:52 2002 UTC (22 years, 8 months ago) by wakaba
Branch: MAIN
Changes since 1.7: +1491 -1484 lines
2002-03-24  wakaba <w@suika.fam.cx>

	* wiki.cgi (inline): 
	- Ignore LinkLikeThis style anchor.
	- Allow [[space including anchor]].
	(make_link): Likewise.  Gets rid of angle branckets
	before make anchor.

1 wakaba 1.8 #!/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     # $Id: wiki.cgi,v 1.7 2002/02/04 15:27:22 wakaba Exp $
24     ##############################
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     my $dbname = './wikidata';
44     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 = '\[\[([^>\x09]+?\]?)\]\]';
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     <!--<input type="hidden" name="mycmd" value="preview">-->
237     <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     <input type="submit" name="myspecial_preview" value="確認">
241     <input type="submit" name="myspecial_write" value="確認せず変更">
242     </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     (?:&lt;(?:mailto|http|https|ftp|urn):[\x21-\x7E]*)&gt;
569     #| (?:$WikiName) # LocalLinkLikeThis
570     | (?:$BracketName) # [[日本語リンク]]
571     )
572     !
573     &make_link($1)
574     !gex;
575     return $line;
576     }
577    
578     # ページのタイトルからページの内容を得る
579     sub get_page {
580     my $page_name = shift;
581     return $database{$page_name};
582     }
583    
584     # ページの内容を与える
585     # &set_page($title, $txt)
586     sub set_page {
587     # ページを更新する
588     my $title = $_[0];
589     $database{$title} = $_[1];
590     # 空ページなら削除する
591     unless ($database{$title}) {
592     delete $database{$title};
593     }
594     # RecentChangesを更新する
595     my $delim = ' - ';
596     my @pages = split(/\n/, $database{$whatsnew});
597     my $datestr = &get_current_datestr;
598     unshift(@pages, qq|-$datestr$delim$title|);
599     # 同一ページの更新は最新のもののみにし、
600     # 存在しないページはスキップする。
601     my %count;
602     my @newpages;
603     foreach my $line (@pages) {
604     my ($prefix, $title) = split(/$delim/, $line);
605     $count{$title}++;
606     if ($count{$title} == 1 and exists($database{$title})) {
607     push(@newpages, qq|$prefix - $title|);
608     }
609     }
610     # ここで本当に更新
611     $database{$whatsnew} = join("\n", splice(@newpages, 0, $maxnew));
612     }
613    
614     # ページのヘッダを出力
615     sub print_header {
616     my $title = shift;
617     print <<"EOD";
618     Content-type: text/html
619    
620     <html><head>
621     <title>$title</title>
622     <style type="text/css">
623     $style
624     </style>
625     </head>
626     <body>
627     EOD
628     }
629    
630     # ツールバーを出力
631     sub print_toolbar {
632     my $page_name = shift;
633     my $percent_name = &encode_percent($page_name);
634     my $editlink = '';
635     if ($page_name ne '' and &is_editable($page_name)) {
636     $editlink = <<"EOD";
637     <a href="$thisurl?mycmd=edit;mypage=$percent_name">編集</a> |
638     <a href="$thisurl?mycmd=diff;mypage=$percent_name">差分</a> |
639     EOD
640     }
641     print <<"EOD";
642     <p>
643     [
644     <a href="$thisurl?mycmd=read;mypage=$toppage">トップ</a> |
645     <a href="$thisurl?mycmd=list">一覧</a> |
646     $editlink
647     <a href="$thisurl?mycmd=search">単語検索</a> |
648     <a href="$thisurl?mycmd=read;mypage=$whatsnew">最終更新</a>
649     ]
650     </p>
651     EOD
652     }
653    
654     # ページのフッタを出力
655     sub print_footer {
656     print <<"EOD";
657     <address>
658     <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>
659     + <a href="$modifierlink">$modifier</a> ${version}.
660     [<a href="/" title="このサーバーの首頁">/</a>
661     <a href="/map" title="このサーバーの案内">地図</a>
662     <a href="/search/" title="このサーバーの検索">検索</a>]
663     </address>
664     </body></html>
665     EOD
666     }
667    
668     # URLやページの名前からリンクを作る
669     sub make_link {
670     my $name = shift;
671     $name =~ s/^&lt;(.*)&gt;$/$1/;
672     $name =~ s/^\[\[(.*)\]\]$/$1/;
673     if ($name =~ /^(http|https|ftp).*?(\.png|\.jpeg|\.jpg)?$/) {
674     if ($2) {
675     return qq|<a href="$name"><img border="0" src="$name" /></a>|;
676     } else {
677     return qq|&lt;<a href="$name">$name</a>&gt;|;
678     }
679     } elsif ($name =~ /^mailto:(.*)/) {
680     my $address = $1;
681     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     } else {
685     my $name2 = $name; $name2 =~ tr/\x20/-/;
686     if ($database{$name2}) {
687     my $percent_name = &encode_percent($name2);
688     return qq|<a href="$thisurl?mycmd=read;mypage=$percent_name" class="wiki">$name</a>|;
689     } elsif ($database{'[['.$name2.']]'}) {
690     my $percent_name = &encode_percent('[['.$name2.']]');
691     return qq|<a href="$thisurl?mycmd=read;mypage=$percent_name" class="wiki">$name</a>|;
692     } else {
693     my $percent_name = &encode_percent($name2);
694     return qq|<a href="$thisurl?mycmd=edit;mypage=$percent_name" class="wiki newpage">$name?</a>|;
695     }
696     }
697     }
698    
699     # %xx の形式にエンコードする
700     # これは、
701     # http://www.hyuki.com/yukiwiki/yukiwiki.cgi?mycmd=read&mypage=%3C%8C%8B%8F%E9%8D_%3E
702     # という形式のために使われる。
703     # '<結城浩>' → '%3C%8C%8B%8F%E9%8D_%3E'
704     sub encode_percent {
705     my $name = shift;
706     my $encoded = '';
707     foreach my $ch (split(//, $name)) {
708     if ($ch =~ /[A-Za-z0-9_]/) {
709     $encoded .= $ch;
710     } else {
711     $encoded .= '%' . sprintf("%02X", ord($ch));
712     }
713     }
714     return $encoded;
715     }
716    
717     # テキスト本体をHTMLに変換する
718     sub convert_html {
719     my ($txt) = shift;
720     my (@txt) = split(/\n/, $txt);
721     foreach (@txt) {
722     chomp;
723     if (/^\*\*(.*)/) {
724     push(@result, splice(@saved), '<h3>' . &inline($1) . '</h3>');
725     } elsif (/^\*(.*)/) {
726     push(@result, splice(@saved), '<h2>' . &inline($1) . '</h2>');
727     } elsif (/^----/) {
728     push(@result, splice(@saved), '<hr>');
729     } elsif (/^(-{1,3})(.*)/) {
730     &back_push('ul', length($1));
731     push(@result, '<li>' . &inline($2) . '</li>');
732     } elsif (/^:([^:]+):(.*)/) {
733     &back_push('dl', 1);
734     push(@result, '<dt>' . &inline($1) . '</dt>', '<dd>' . &inline($2) . '</dd>');
735     } elsif (/^(>{1,3})(.*)/) {
736     &back_push('blockquote', length($1));
737     push(@result, &inline($2));
738     } elsif (/^\s*$/) {
739     push(@result, splice(@saved));
740     unshift(@saved, "</p>");
741     push(@result, "<p>");
742     } elsif (/^(\s+.*)$/) {
743     &back_push('pre', 1);
744     push(@result, &escape($1)); # Not &inline, but &escape
745     } else {
746     push(@result, &inline($_));
747     }
748     }
749     push(@result, splice(@saved));
750     return join("\n", @result);
751     }
752    
753     # &back_push($tag, $count)
754     # $tagのタグを$levelレベルまで詰める。
755     sub back_push {
756     my ($tag, $level) = @_;
757     while (@saved > $level) {
758     push(@result, shift(@saved));
759     }
760     if ($saved[0] ne "</$tag>") {
761     push(@result, splice(@saved));
762     }
763     while (@saved < $level) {
764     unshift(@saved, "</$tag>");
765     push(@result, "<$tag>");
766     }
767     }
768    
769     # 編集可能ページか?
770     sub is_editable {
771     my ($pagename) = @_;
772     foreach (@uneditable) {
773     if ($pagename eq $_) {
774     return 0;
775     }
776     }
777     if (&is_valid_name($pagename)) {
778     return 1;
779     }
780     return 0;
781     }
782    
783     # Validな名前か?
784     sub is_valid_name {
785     my ($pagename) = @_;
786     if ($pagename =~ /^$WikiName$/) {
787     return 1;
788     } elsif ($pagename =~ /^$BracketName$/) {
789     return 1;
790     } else {
791     return 0;
792     }
793     }
794    
795     # 現在時刻の文字列を得る
796     sub get_current_datestr {
797     my (@wdays) = ( "日", "月", "火", "水", "木", "金", "土" );
798     my ($sec, $min, $hour, $mday, $mon, $year, $wday) = localtime(time);
799     return sprintf("%4d-%02d-%02d (%s) %02d:%02d:%02d",
800     $year + 1900, $mon + 1, $mday, $wdays[$wday], $hour, $min, $sec);
801     }
802    
803     # URL?SomePageや、
804     # URL?[[結城浩]]の形式だった場合、(not yet)
805     # 強制的にmycmdをreadにして$formの内容を設定する。
806     sub normalize_form {
807     foreach my $key (keys %form) {
808     if ($key =~ /^$WikiName$/) {
809     $form{mycmd} = 'read';
810     $form{mypage} = $1;
811     last;
812     } elsif ($key =~ /^$BracketName$/) {
813     $form{mycmd} = 'read';
814     $form{mypage} = $1;
815     last;
816     }
817     }
818     }
819    
820     # 変換テストを行なうときのサンプル
821     sub print_sample {
822     my $txt = &convert_html(<<"EOD");
823     *大見出し1
824     **小見出し1-1
825     -項目1
826     -項目2
827     -項目3
828     段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
829     段落1段落1段落1段落1段落1段落''強調''1段落1段落1段落1段落1段落1
830     段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
831    
832     段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2
833     段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2
834     段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2
835     **小見出し1-2
836     :用語1:いろいろ書いた解説文1と''強調単語''
837     段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
838     段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
839     段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
840     :用語2:いろいろ書いた解説文2
841     :用語3:いろいろ書いた解説文3
842     ----
843     *大見出し2
844     **小見出し2-1
845     &lt;http://suika.fam.cx/&gt;
846     **小見出し2-2
847    
848     [[結城浩]]
849    
850     段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1段落1
851     段落1段落1段落1段落'''強調'''1段落1段落1段落1段落1段落1段落1段落1
852     段落1段落1段落1段落'''''強い強調'''''1段落1段落1段落1段落1段落1段落1段落1段落1
853     >段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2
854     >段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2
855     >段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2段落2
856    
857     レベル0レベル0レベル0レベル0レベル0レベル0
858    
859     >レベル1
860     >レベル1
861     >レベル1
862     >>レベル2
863     >>レベル2
864     >>レベル2
865     >>>レベル3
866     -はろ1
867     --はろ2
868     ろろろろ2
869     ---はろ3
870     --はろ2
871     ---はろ3
872     --はろ2
873     ---はろ3
874     >>>レベル3
875     >>>レベル3
876     >>>レベル3
877     EOD
878     print $txt;
879     exit;
880     }
881    
882     sub diff_check {
883     traverse_sequences(
884     $msgrefA, $msgrefB,
885     {
886     MATCH => \&df_match,
887     DISCARD_A => \&df_delete,
888     DISCARD_B => \&df_add,
889     }
890     );
891     &diff_flush;
892     }
893    
894     sub diff_flush {
895     $diff_text .= join('', map { "-$_\n" } splice(@diff_deleted));
896     $diff_text .= join('', map { "+$_\n" } splice(@diff_added));
897     }
898    
899     sub df_match {
900     my ($a, $b) = @_;
901     &diff_flush;
902     $diff_text .= "=$msgrefA->[$a]\n";
903     }
904    
905     sub df_delete {
906     my ($a, $b) = @_;
907     push(@diff_deleted, $msgrefA->[$a]);
908     }
909    
910     sub df_add {
911     my ($a, $b) = @_;
912     push(@diff_added, $msgrefB->[$b]);
913     }
914    
915     sub calc_message_digest { # You have to use MD5...
916     my $text = shift;
917     my @text = split(//, $text);
918     my $len = length($text);
919     my $checksum = 0;
920     foreach (@text) {
921     $checksum += ord($_);
922     $checksum = ($checksum * 2) % 65536 + (($checksum & 32768) ? 1 : 0); # 16bit rotate
923     }
924     return "$len:$checksum";
925     }
926    
927     # Definition of YukiWikiDB
928     package YukiWikiDB;
929    
930     my $debug = 1;
931    
932     # Constructor
933     sub new {
934     return shift->TIEHASH(@_);
935     }
936    
937     # tying
938     sub TIEHASH {
939     my ($class, $dbname) = @_;
940     my $self = {
941     dir => $dbname,
942     keys => [],
943     };
944     if (not -d $self->{dir}) {
945     if (!mkdir($self->{dir}, 0777)) {
946     print "mkdir(" . $self->{dir} . ") fail\n" if ($debug);
947     return undef;
948     }
949     }
950     return bless($self, $class);
951     }
952    
953     # Store
954     sub STORE {
955     my ($self, $key, $val) = @_;
956     my $file = &make_filename($self, $key);
957     if (open(FILE,"> $file")) {
958     binmode(FILE);
959     print FILE $val;
960     close(FILE);
961     return $self->{$key} = $val;
962     } else {
963     print "$file create error.";
964     }
965     }
966    
967     # Fetch
968     sub FETCH {
969     my ($self, $key) = @_;
970     my $file = &make_filename($self, $key);
971     if (open(FILE, $file)) {
972     local $/;
973     $self->{$key} = <FILE>;
974     close(FILE);
975     }
976     return $self->{$key};
977     }
978    
979     # Exists
980     sub EXISTS {
981     my ($self, $key) = @_;
982     my $file = &make_filename($self, $key);
983     return -e($file);
984     }
985    
986     # Delete
987     sub DELETE {
988     my ($self, $key) = @_;
989     my $file = &make_filename($self, $key);
990     unlink $file;
991     return delete $self->{$key};
992     }
993    
994     sub FIRSTKEY {
995     my ($self) = @_;
996     opendir(DIR, $self->{dir}) or die $self->{dir};
997     @{$self->{keys}} = grep /\.txt$/, readdir(DIR);
998     foreach my $name (@{$self->{keys}}) {
999     $name =~ s/\.txt$//;
1000     $name =~ s/[0-9A-F][0-9A-F]/pack("C", hex($&))/eg;
1001     }
1002     return shift @{$self->{keys}};
1003     }
1004    
1005     sub NEXTKEY {
1006     my ($self) = @_;
1007     return shift @{$self->{keys}};
1008     }
1009    
1010     sub make_filename {
1011     my ($self, $key) = @_;
1012     my $enkey = '';
1013     foreach my $ch (split(//, $key)) {
1014     $enkey .= sprintf("%02X", ord($ch));
1015     }
1016     return $self->{dir} . "/$enkey.txt";
1017     }
1018     __END__
1019    
1020     =head1 NAME
1021    
1022     YukiWiki - 自由にページを追加・削除・編集できるWebページ構築CGI
1023    
1024     Copyright (C) 2000,2001 by Hiroshi Yuki.
1025     結城浩 <hyuki@hyuki.com>
1026     http://www.hyuki.com/
1027     http://www.hyuki.com/yukiwiki/
1028    
1029     =head1 SYNOPSIS
1030    
1031     http://www.hyuki.com/yukiwiki/yukiwiki.cgi
1032    
1033     =head1 DESCRIPTION
1034    
1035     YukiWiki(結城ウィキィ)は参加者が自由にページを追加・削除・編集できる
1036     不思議なWebページ群を作るCGIです。
1037     Webで動作する掲示板とちょっと似ていますが、
1038     Web掲示板が単にメッセージを追加するだけなのに対して、
1039     YukiWikiは、Webページ全体を自由に変更することができます。
1040    
1041     YukiWikiは、Cunningham & CunninghamのWikiWikiWebの
1042     仕様を参考にして独自に作られました。
1043    
1044     YukiWikiはPerlで書かれたCGIスクリプトとして実現されていますので、
1045     Perlが動作するWebサーバならば比較的容易に設置できます。
1046    
1047     あとはdbmopenが使える環境ならば設置できます(Version 1.5.0以降ならdbmopenが使えなくても設置できます)。
1048    
1049    
1050     YukiWikiはフリーソフトです。
1051     ご自由にお使いください。
1052    
1053     =head1 設置方法
1054    
1055     =head2 入手
1056    
1057     YukiWikiの最新版は、
1058     http://www.hyuki.com/yukiwiki/
1059     から入手できます。
1060    
1061     =head2 ファイル一覧
1062    
1063     readme.txt ドキュメント
1064     yukiwiki.cgi YukiWiki本体
1065     yukiwiki.gif ロゴ(カラー版)
1066     yukimono.gif ロゴ(モノクロ版)
1067     jcode.pl 漢字コードライブラリ
1068    
1069     =head2 インストール
1070    
1071     =over 4
1072    
1073     =item 1.
1074    
1075     アーカイブを解く。
1076    
1077     =item 2.
1078    
1079     yukiwiki.cgiのはじめの方にある設定を確認します。
1080     通常は何もしなくてよいが、
1081     はじめは$touchfileを''にした方がよいでしょう。
1082    
1083     =item 3.
1084    
1085     yukiwiki.cgiとjcode.plを同じところに設置します。
1086    
1087     =item 4.
1088    
1089     サイズ0のyukiwiki.dbというファイルを設置します。
1090     (Perlシステムによってはyukiwiki.pag, yukiwiki.dir)
1091    
1092     =item 5.
1093    
1094     yukiwiki.cgiにブラウザからアクセスします。
1095    
1096     =back
1097    
1098     =head2 パーミッション
1099    
1100     ファイル パーミッション 転送モード
1101     yukiwiki.cgi 755 ASCII
1102     yukiwiki.gif 644 BINARY
1103     yukimono.gif 644 BINARY
1104     jcode.pl 644 ASCII
1105    
1106     $dbmopen = 1; にした場合:
1107     yukiwiki.db 666 BINARY
1108     (yukiwiki.pag, yukiwiki.dirの場合もあり)
1109    
1110     $dbmopen = 0; にした場合: (カレントディレクトリを777にしておく)
1111     . 777 (転送しない)
1112    
1113     =head1 データのバックアップ方法
1114    
1115     $dbmopen = 1;の場合は、
1116     データはすべてyukiwiki.db(.dir, .pag)に入る。
1117     これをバックアップすればよい。
1118    
1119     $dbmopen = 0;の場合は、
1120     yukiwikiというディレクトリができるので、
1121     これ以下をバックアップすればよい。
1122    
1123     =head1 新しいページの作り方
1124    
1125     =over 4
1126    
1127     =item 1.
1128    
1129     まず、適当なページ(例えばFrontPage)を選び、
1130     ページの下にある「編集」リンクをたどります。
1131    
1132     =item 2.
1133    
1134     するとテキスト入力ができる状態になるので、
1135     そこにNewPageのような単語
1136     (大文字小文字混在している英文字列)
1137     を書いて「保存」します。
1138    
1139     =item 3.
1140    
1141     保存すると、FrontPageのページが書き換わり、
1142     あなたが書いたNewPageという文字列の後ろに ? というリンクが表示されます。
1143     この ? はそのページがまだ存在しないことを示す印です。
1144    
1145     =item 4.
1146    
1147     その ? をクリックすると新しいページNewPageができますので、
1148     あなたの好きな文章をその新しいページに書いて保存します。
1149    
1150     =item 5.
1151    
1152     NewPageページができるとFrontPageの ? は消えて、リンクとなります。
1153    
1154     =back
1155    
1156     =head1 テキスト整形のルール
1157    
1158     =over 4
1159    
1160     =item *
1161    
1162     連続した複数行はフィルされて表示されます。
1163    
1164     =item *
1165    
1166     空行は段落C<< <p> >>の区切りとなります。
1167    
1168     =item *
1169    
1170     HTMLのタグは書けません。
1171    
1172     =item *
1173    
1174     B<''ボールド''>のようにシングルクォート二つではさむと、
1175     ボールドC<< <b> >>になります。
1176    
1177     =item *
1178    
1179     B<'''イタリック'''>のようにシングルクォート三つではさむと、
1180     イタリックC<< <i> >>になります。
1181    
1182     =item *
1183    
1184     B<---->のようにマイナス4つがあると、
1185     水平線C<< <hr> >>になります。
1186    
1187     =item *
1188    
1189     行をB<*>ではじめると、
1190     大見出しC<< <h2> >>になります。
1191    
1192     =item *
1193    
1194     行をB<**>ではじめると、
1195     小見出しC<< <h3> >>になります。
1196    
1197     =item *
1198    
1199     行をマイナス-ではじめると、
1200     箇条書きC<< <ul> >>になります。
1201     マイナスの数が増えるとレベルが下がります(3レベルまで)
1202    
1203     -項目1
1204     --項目1-1
1205     --項目1-2
1206     -項目2
1207     -項目3
1208     --項目3-1
1209     ---項目3-1-1
1210     ---項目3-1-2
1211     --項目3-2
1212    
1213     =item *
1214    
1215     コロンを使うと、
1216     用語と解説文のリストC<< <dl> >>が書けます。
1217    
1218     :用語1:いろいろ書いた解説文1
1219     :用語2:いろいろ書いた解説文2
1220     :用語3:いろいろ書いた解説文3
1221    
1222     =item *
1223    
1224     リンク
1225    
1226     =over 4
1227    
1228     =item *
1229    
1230     LinkToSomePageやFrontPageのように、
1231     英単語の最初の一文字を大文字にしたものが
1232     二つ以上連続したものはYukiWikiのページ名となり、
1233     それが文章中に含まれるとリンクになります。
1234    
1235     =item *
1236    
1237     http://www.hyuki.com/ のようなURLは自動的にリンクになります。
1238    
1239     =item *
1240    
1241     二重の大かっこ[[ ]]でくくった文字列も、
1242     YukiWikiのページ名になります。
1243     大かっこの中にはスペースを含めてはいけません。
1244     日本語も使えます。
1245    
1246     =back
1247    
1248     =item *
1249    
1250     行頭がスペースやタブで始まっていると、
1251     それは整形済みの段落C<< <pre> >>として扱われます。
1252     プログラムの表示などに使うと便利です。
1253    
1254    
1255     =item *
1256    
1257     行を > ではじめると、
1258     引用文C<< <blockquote> >>が書けます。
1259     >の数が多いとインデントが深くなります(3レベルまで)。
1260    
1261     >文章
1262     >文章
1263     >>さらなる引用
1264     >文章
1265    
1266     =back
1267    
1268     =head1 更新履歴
1269    
1270     =over 4
1271    
1272     =item *
1273    
1274     2001年10月20日、Version 1.6.6。
1275    
1276     更新の衝突対策。
1277     元ページの簡単なチェックサムを取っておき、
1278     更新前にチェックサムを比較する。
1279     修正個所はdigestという文字列を検索すれば分かる。
1280     本来はMD5などでちゃんとやった方がいいのだけれど。
1281    
1282     衝突時に表示されるメッセージなどは「極悪」さんのページを参考にした。
1283    
1284     =item *
1285    
1286     2001年10月17日、Version 1.6.5。
1287    
1288     プレビュー画面で、更新ボタンを押したときに送信される
1289     メッセージの内容をinput要素のtype="hidden"を使って埋め込むのをやめる。
1290     代わりに、textarea要素を使う。
1291     再プレビュー用にmyspecial_を導入。でもきれいな対策ではない。
1292    
1293     =item *
1294    
1295     2001年8月30日、Version 1.6.4。
1296    
1297     URLでダイレクトにページ名を指定しても、
1298     $WikiNameと$BracketName以外のページを作れないようにした。
1299     (is_valid_nameとis_editable参照)。
1300    
1301     =item *
1302    
1303     2001年8月30日、Version 1.6.3。
1304    
1305     RecentChangesを編集・再編集不可とした。
1306     編集不可ページは@uneditableにページ名を入れる。
1307    
1308     =item *
1309    
1310     2001年2月25日、Version 1.6.1, 1.6.2。
1311    
1312     差分機能のバグ修正。
1313     do_previewで'>'が扱えないバグを修正
1314     (ユーザからの指摘)。
1315    
1316     =item *
1317    
1318     2001年2月22日、Version 1.6.0。
1319     差分機能を実装した。
1320    
1321     =item *
1322    
1323     2001年2月19日、Version 1.5.4。
1324     画像ファイルへのリンクは画像にしてみた。
1325    
1326     =item *
1327    
1328     2001年2月19日、Version 1.5.3。
1329     RecentChangesの中に削除したページがあるのをやめた。
1330     use strict;で引っかかる部分を少し整理(完全ではない)。
1331    
1332     =item *
1333    
1334     2001年2月16日、Version 1.5.2。
1335     textareaに表示およびプレビューする前に < や > を &lt; や &gt; に変換した
1336     (do_preview, editpage, print_preview_buttons)。
1337    
1338     =item *
1339    
1340     2000年12月27日、Version 1.5.1。
1341     プレビュー画面を整理した。
1342    
1343     =item *
1344    
1345     2000年12月22日、Version 1.5.0。
1346     全体的にずいぶん書き直した。
1347     一覧を別途作成するようにした(do_list)。
1348     書き込む前に確認画面を出すようにした(do_preview)。
1349     テキストの書き方を編集画面に入れた(do_edit, do_reedit)。
1350     WhatsNew→RecentChanges、TopPage→FrontPageに変更した。
1351    
1352     =item *
1353    
1354     2000年12月20日、Version 1.1.0。
1355     tieを利用して、dbmopenが使えない場合でも動作するように修正。
1356     利用者の1人である「極悪」さんから
1357     送っていただいたコードを元にしています。
1358    
1359     =item *
1360    
1361     2000年9月5日、Version 1.0.2。
1362     <body color=...> → <body bgcolor=...>
1363     利用者からの指摘による。感謝。
1364    
1365     =item *
1366    
1367     2000年8月6日、Version 1.0.1を公開。
1368     C MAGAZINE(ソフトバンクパブリッシング)
1369     2000年10月号連載記事向け公開版。
1370     [[ ]] の最後が「望」のようにシフトJISで
1371     0x5Dになる場合の回避を行なった。
1372    
1373     =item *
1374    
1375     2000年8月5日、Version 1.0.0を公開。
1376    
1377     =item *
1378    
1379     2000年7月23日、Version 0.82を公開。
1380     編集時のリンクミス。
1381     <textarea>の属性変更。
1382    
1383     =item *
1384    
1385     2000年7月22日、Version 0.81を公開。
1386     ロゴを組み込む。
1387    
1388     =item *
1389    
1390     2000年7月21日、Version 0.80を公開。
1391     PODをCGI中に書き込む。
1392    
1393     =item *
1394    
1395     2000年7月19日、Version 0.70を公開。
1396     '''イタリック'''や、--、---、>>、>>>などを実装。
1397    
1398     =item *
1399    
1400     2000年7月18日、Version 0.60を公開。
1401     *太字*を''太字''に変更
1402    
1403     =item *
1404    
1405     2000年7月17日、Version 0.50を公開。
1406    
1407     =item *
1408    
1409     2000年7月17日、さらにいろいろ追加する。
1410    
1411     =item *
1412    
1413     2000年7月16日、いろいろ追加。
1414    
1415     =item *
1416    
1417     2000年7月15日、公開。
1418    
1419     =back
1420    
1421     =head1 TODO
1422    
1423     - テキスト表示モード
1424     - Charsetを明示。
1425     - textarea中の閉じタグ対応
1426     - メニューの英語表記付記
1427     - プレビューのボタンで、mymsgをinputのvalueに入れているが、改行をそのままvalueにいれてはいけないのではないか。
1428     - 「再編集」の機能はブラウザの back で充分ではないか。プレビューはもっとシンプルに。
1429     - ページタイトル(Wikiname)が検索にかかるようにする。
1430     - InterWiki風の機能「URLを隠しつつ、リンクを張る」
1431    
1432     =head1 作者
1433    
1434     Copyright (C) 2000 by Hiroshi Yuki.
1435     結城浩 <hyuki@hyuki.com>
1436     http://www.hyuki.com/
1437     http://www.hyuki.com/yukiwiki/
1438    
1439     質問、意見、バグ報告は hyuki@hyuki.com にメールしてください。
1440    
1441     =head1 配布条件
1442    
1443     YukiWikiは、
1444     GNU General Public Licenseにて公開します。
1445    
1446     YukiWikiはフリーソフトです。
1447     ご自由にお使いください。
1448     自分好みのYukiWikiが作れるようにシンプルにしてあります。
1449    
1450     =head1 謝辞
1451    
1452     本家のWikiWikiを作ったCunningham & Cunningham, Inc.に
1453     感謝します。
1454    
1455     YukiWikiを楽しんで使ってくださる
1456     ネット上の方々に感謝します。
1457    
1458     YukiWikiのロゴをデザインしてくださった橋本礼奈さん
1459     http://city.hokkai.or.jp/~reina/
1460     に感謝します。
1461    
1462     tieを使った版の元になるコードを送ってくださった
1463     「極悪」さんに感謝します。
1464    
1465     =head1 参照リンク
1466    
1467     =over 4
1468    
1469     =item *
1470    
1471     YukiWikiホームページ
1472     http://www.hyuki.com/yukiwiki/
1473    
1474     =item *
1475    
1476     本家のWikiWiki
1477     http://c2.com/cgi/wiki?WikiWikiWeb
1478    
1479     =item *
1480    
1481     本家のWikiWikiの作者(Cunningham & Cunningham, Inc.)
1482     http://c2.com/
1483    
1484     =item *
1485    
1486     YukiWikiのロゴデザインをしてくださった橋本礼奈さんのページ
1487     http://city.hokkai.or.jp/~reina/
1488    
1489     =back
1490    
1491     =cut

admin@suikawiki.org
ViewVC Help
Powered by ViewVC 1.1.24