2 // PukiWiki - Yet another WikiWikiWeb clone
4 // Copyright 2003-2017 PukiWiki Development Team
5 // License: GPL v2 or (at your option) any later version
7 // Issue tracker plugin (See Also bugtrack plugin)
9 // tracker_listで表示しないページ名(正規表現で)
10 // 'SubMenu'ページ および '/'を含むページを除外する
11 define('TRACKER_LIST_EXCLUDE_PATTERN','#^SubMenu$|/#');
13 //define('TRACKER_LIST_EXCLUDE_PATTERN','#(?!)#');
15 // 項目の取り出しに失敗したページを一覧に表示する
16 define('TRACKER_LIST_SHOW_ERROR_PAGE',TRUE);
18 function plugin_tracker_convert()
22 $script = get_base_uri();
23 if (PKWK_READONLY) return ''; // Show nothing
25 $base = $refer = $vars['page'];
27 $config_name = 'default';
32 $args = func_get_args();
36 $options = array_splice($args,2);
38 $args[1] = get_fullname($args[1],$base);
39 $base = is_pagename($args[1]) ? $args[1] : $base;
41 $config_name = ($args[0] != '') ? $args[0] : $config_name;
42 list($config_name,$form) = array_pad(explode('/',$config_name,2),2,$form);
46 $config = new Config('plugin/tracker/'.$config_name);
50 return "<p>config file '".htmlsc($config_name)."' not found.</p>";
53 $config->config_name = $config_name;
55 $fields = plugin_tracker_get_fields($base,$refer,$config);
57 $form = $config->page.'/'.$form;
60 return "<p>config file '".make_pagelink($form)."' not found.</p>";
62 $retval = convert_html(plugin_tracker_get_source($form));
65 foreach (array_keys($fields) as $name)
67 $replace = $fields[$name]->get_tag();
68 if (is_a($fields[$name],'Tracker_field_hidden'))
73 $retval = str_replace("[$name]",$replace,$retval);
76 <form enctype="multipart/form-data" action="$script" method="post">
84 function plugin_tracker_action()
86 global $post, $vars, $now;
88 if (PKWK_READONLY) die_message('PKWK_READONLY prohibits editing');
90 $config_name = array_key_exists('_config',$post) ? $post['_config'] : '';
92 $config = new Config('plugin/tracker/'.$config_name);
95 return "<p>config file '".htmlsc($config_name)."' not found.</p>";
97 $config->config_name = $config_name;
98 $source = $config->page.'/page';
100 $refer = array_key_exists('_refer',$post) ? $post['_refer'] : $post['_base'];
102 if (!is_pagename($refer))
105 'msg'=>'cannot write',
106 'body'=>'page name ('.htmlsc($refer).') is not valid.'
109 if (!is_page($source))
112 'msg'=>'cannot write',
113 'body'=>'page template ('.htmlsc($source).') is not exist.'
117 $base = $post['_base'];
118 if (!is_pagename($base))
121 'msg'=>'cannot write',
122 'body'=>'page name ('.htmlsc($base).') is not valid.'
125 $name = (array_key_exists('_name',$post)) ? $post['_name'] : '';
126 $_page = (array_key_exists('_page',$post)) ? $post['_page'] : '';
127 if (is_pagename($_page)) {
128 // Create _page page if _page is in parameters
129 $page = $real = $_page;
130 } else if (is_pagename($name)) {
131 // Create "$base/$name" page if _name is in parameters
133 $page = get_fullname('./' . $name, $base);
137 if (!is_pagename($page) || is_page($page)) {
138 // Need new page name => Get last article number + 1
139 $page_list = plugin_tracker_get_page_list($base, false);
140 usort($page_list, '_plugin_tracker_list_paganame_compare');
141 if (count($page_list) === 0) {
144 $latest_page = $page_list[count($page_list) - 1]['name'];
145 $num = intval(substr($latest_page, strlen($base) + 1)) + 1;
148 $page = $base . '/' . $num;
151 $postdata = plugin_tracker_get_source($source);
154 $_post = array_merge($post,$_FILES);
155 $_post['_date'] = $now;
156 $_post['_page'] = $page;
157 $_post['_name'] = $name;
158 $_post['_real'] = $real;
159 // $_post['_refer'] = $_post['refer'];
161 $fields = plugin_tracker_get_fields($page,$refer,$config);
163 // Creating an empty page, before attaching files
164 touch(get_filename($page));
166 foreach (array_keys($fields) as $key)
168 $value = array_key_exists($key,$_post) ?
169 $fields[$key]->format_value($_post[$key]) : '';
171 foreach (array_keys($postdata) as $num)
173 if (trim($postdata[$num]) == '')
177 $postdata[$num] = str_replace(
179 ($postdata[$num]{0} == '|' or $postdata[$num]{0} == ':') ?
180 str_replace('|','|',$value) : $value,
186 // Writing page data, without touch
187 page_write($page, join('', $postdata));
189 header('Location: ' . get_page_uri($page, PKWK_URI_ROOT));
194 * Page_list comparator
196 function _plugin_tracker_list_paganame_compare($a, $b)
198 return strnatcmp($a['name'], $b['name']);
202 * Get page list for "$page/"
204 function plugin_tracker_get_page_list($page, $needs_filetime) {
205 $page_list = array();
206 $pattern = $page . '/';
207 $pattern_len = strlen($pattern);
208 foreach (get_existpages() as $p) {
209 if (strncmp($p, $pattern, $pattern_len) === 0 && pkwk_ctype_digit(substr($p, $pattern_len))) {
210 if ($needs_filetime) {
211 $page_list[] = array('name'=>$p,'filetime'=>get_filetime($p));
213 $page_list[] = array('name'=>$p);
222 function plugin_tracker_inline()
226 if (PKWK_READONLY) return ''; // Show nothing
228 $args = func_get_args();
229 if (count($args) < 3)
233 $body = array_pop($args);
234 list($config_name,$field) = $args;
236 $config = new Config('plugin/tracker/'.$config_name);
238 if (!$config->read())
240 return "config file '".htmlsc($config_name)."' not found.";
243 $config->config_name = $config_name;
245 $fields = plugin_tracker_get_fields($vars['page'],$vars['page'],$config);
246 $fields[$field]->default_value = $body;
247 return $fields[$field]->get_tag();
251 function plugin_tracker_get_fields($base,$refer,&$config)
253 global $now,$_tracker_messages;
258 '_date'=>'text', // 投稿日時
259 '_update'=>'date', // 最終更新
260 '_past'=>'past', // 経過(passage)
261 '_page'=>'page', // ページ名
262 '_name'=>'text', // 指定されたページ名
263 '_real'=>'real', // 実際のページ名
264 '_refer'=>'page', // 参照元(フォームのあるページ)
265 '_base'=>'page', // 基準ページ
266 '_submit'=>'submit' // 追加ボタン
269 $class = 'Tracker_field_'.$class;
270 $fields[$field] = new $class(array($field,$_tracker_messages["btn$field"],'','20',''),$base,$refer,$config);
273 foreach ($config->get('fields') as $field)
275 // 0=>項目名 1=>見出し 2=>形式 3=>オプション 4=>デフォルト値
276 $class = 'Tracker_field_'.$field[2];
277 if (!class_exists($class))
279 $class = 'Tracker_field_text';
283 $fields[$field[0]] = new $class($field,$base,$refer,$config);
298 var $sort_type = SORT_REGULAR;
301 function Tracker_field($field,$page,$refer,&$config)
303 $this->__construct($field, $page, $refer, $config);
305 function __construct($field,$page,$refer,&$config)
311 $this->name = $field[0];
312 $this->title = $field[1];
313 $this->values = explode(',',$field[3]);
314 $this->default_value = $field[4];
316 $this->refer = $refer;
317 $this->config = &$config;
318 $this->data = array_key_exists($this->name,$post) ? $post[$this->name] : '';
323 function get_style($str)
327 function format_value($value)
331 function format_cell($str)
335 function get_value($value)
340 class Tracker_field_text extends Tracker_field
342 var $sort_type = SORT_STRING;
346 $s_name = htmlsc($this->name);
347 $s_size = htmlsc($this->values[0]);
348 $s_value = htmlsc($this->default_value);
349 return "<input type=\"text\" name=\"$s_name\" size=\"$s_size\" value=\"$s_value\" />";
352 class Tracker_field_page extends Tracker_field_text
354 var $sort_type = SORT_STRING;
356 function format_value($value)
360 $value = strip_bracket($value);
361 if (is_pagename($value))
363 $value = "[[$value]]";
365 return parent::format_value($value);
368 class Tracker_field_real extends Tracker_field_text
370 var $sort_type = SORT_REGULAR;
372 class Tracker_field_title extends Tracker_field_text
374 var $sort_type = SORT_STRING;
376 function format_cell($str)
382 class Tracker_field_textarea extends Tracker_field
384 var $sort_type = SORT_STRING;
388 $s_name = htmlsc($this->name);
389 $s_cols = htmlsc($this->values[0]);
390 $s_rows = htmlsc($this->values[1]);
391 $s_value = htmlsc($this->default_value);
392 return "<textarea name=\"$s_name\" cols=\"$s_cols\" rows=\"$s_rows\">$s_value</textarea>";
394 function format_cell($str)
396 $str = preg_replace('/[\r\n]+/','',$str);
397 if (!empty($this->values[2]) and strlen($str) > ($this->values[2] + 3))
399 $str = mb_substr($str,0,$this->values[2]).'...';
404 class Tracker_field_format extends Tracker_field
406 var $sort_type = SORT_STRING;
408 var $styles = array();
409 var $formats = array();
411 function Tracker_field_format($field,$page,$refer,&$config)
413 $this->__construct($field, $page, $refer, $config);
415 function __construct($field,$page,$refer,&$config)
417 parent::__construct($field,$page,$refer,$config);
419 foreach ($this->config->get($this->name) as $option)
421 list($key,$style,$format) = array_pad(array_map(create_function('$a','return trim($a);'),$option),3,'');
424 $this->styles[$key] = $style;
428 $this->formats[$key] = $format;
434 $s_name = htmlsc($this->name);
435 $s_size = htmlsc($this->values[0]);
436 return "<input type=\"text\" name=\"$s_name\" size=\"$s_size\" />";
438 function get_key($str)
440 return ($str == '') ? 'IS NULL' : 'IS NOT NULL';
442 function format_value($str)
446 return join(', ',array_map(array($this,'format_value'),$str));
448 $key = $this->get_key($str);
449 return array_key_exists($key,$this->formats) ? str_replace('%s',$str,$this->formats[$key]) : $str;
451 function get_style($str)
453 $key = $this->get_key($str);
454 return array_key_exists($key,$this->styles) ? $this->styles[$key] : '%s';
457 class Tracker_field_file extends Tracker_field_format
459 var $sort_type = SORT_STRING;
463 $s_name = htmlsc($this->name);
464 $s_size = htmlsc($this->values[0]);
465 return "<input type=\"file\" name=\"$s_name\" size=\"$s_size\" />";
467 function format_value($str)
469 if (array_key_exists($this->name,$_FILES))
471 require_once(PLUGIN_DIR.'attach.inc.php');
472 $result = attach_upload($_FILES[$this->name],$this->page);
473 if ($result['result']) // アップロード成功
475 return parent::format_value($this->page.'/'.$_FILES[$this->name]['name']);
478 // ファイルが指定されていないか、アップロードに失敗
479 return parent::format_value('');
482 class Tracker_field_radio extends Tracker_field_format
484 var $sort_type = SORT_NUMERIC;
488 $s_name = htmlsc($this->name);
491 foreach ($this->config->get($this->name) as $option)
493 $s_option = htmlsc($option[0]);
494 $checked = trim($option[0]) == trim($this->default_value) ? ' checked="checked"' : '';
496 $s_id = '_p_tracker_' . $s_name . '_' . $this->id . '_' . $id;
497 $retval .= '<input type="radio" name="' . $s_name . '" id="' . $s_id .
498 '" value="' . $s_option . '"' . $checked . ' />' .
499 '<label for="' . $s_id . '">' . $s_option . '</label>' . "\n";
504 function get_key($str)
508 function get_value($value)
510 static $options = array();
511 if (!array_key_exists($this->name,$options))
513 $options[$this->name] = array_flip(array_map(create_function('$arr','return $arr[0];'),$this->config->get($this->name)));
515 return array_key_exists($value,$options[$this->name]) ? $options[$this->name][$value] : $value;
518 class Tracker_field_select extends Tracker_field_radio
520 var $sort_type = SORT_NUMERIC;
522 function get_tag($empty=FALSE)
524 $s_name = htmlsc($this->name);
525 $s_size = (array_key_exists(0,$this->values) and is_numeric($this->values[0])) ?
526 ' size="'.htmlsc($this->values[0]).'"' : '';
527 $s_multiple = (array_key_exists(1,$this->values) and strtolower($this->values[1]) == 'multiple') ?
528 ' multiple="multiple"' : '';
529 $retval = "<select name=\"{$s_name}[]\"$s_size$s_multiple>\n";
532 $retval .= " <option value=\"\"></option>\n";
534 $defaults = array_flip(preg_split('/\s*,\s*/',$this->default_value,-1,PREG_SPLIT_NO_EMPTY));
535 foreach ($this->config->get($this->name) as $option)
537 $s_option = htmlsc($option[0]);
538 $selected = array_key_exists(trim($option[0]),$defaults) ? ' selected="selected"' : '';
539 $retval .= " <option value=\"$s_option\"$selected>$s_option</option>\n";
541 $retval .= "</select>";
546 class Tracker_field_checkbox extends Tracker_field_radio
548 var $sort_type = SORT_NUMERIC;
550 function get_tag($empty=FALSE)
552 $s_name = htmlsc($this->name);
553 $defaults = array_flip(preg_split('/\s*,\s*/',$this->default_value,-1,PREG_SPLIT_NO_EMPTY));
556 foreach ($this->config->get($this->name) as $option)
558 $s_option = htmlsc($option[0]);
559 $checked = array_key_exists(trim($option[0]),$defaults) ?
560 ' checked="checked"' : '';
562 $s_id = '_p_tracker_' . $s_name . '_' . $this->id . '_' . $id;
563 $retval .= '<input type="checkbox" name="' . $s_name .
564 '[]" id="' . $s_id . '" value="' . $s_option . '"' . $checked . ' />' .
565 '<label for="' . $s_id . '">' . $s_option . '</label>' . "\n";
571 class Tracker_field_hidden extends Tracker_field_radio
573 var $sort_type = SORT_NUMERIC;
575 function get_tag($empty=FALSE)
577 $s_name = htmlsc($this->name);
578 $s_default = htmlsc($this->default_value);
579 $retval = "<input type=\"hidden\" name=\"$s_name\" value=\"$s_default\" />\n";
584 class Tracker_field_submit extends Tracker_field
588 $s_title = htmlsc($this->title);
589 $s_page = htmlsc($this->page);
590 $s_refer = htmlsc($this->refer);
591 $s_config = htmlsc($this->config->config_name);
594 <input type="submit" value="$s_title" />
595 <input type="hidden" name="plugin" value="tracker" />
596 <input type="hidden" name="_refer" value="$s_refer" />
597 <input type="hidden" name="_base" value="$s_page" />
598 <input type="hidden" name="_config" value="$s_config" />
602 class Tracker_field_date extends Tracker_field
604 var $sort_type = SORT_NUMERIC;
606 function format_cell($timestamp)
608 return format_date($timestamp);
611 class Tracker_field_past extends Tracker_field
613 var $sort_type = SORT_NUMERIC;
615 function format_cell($timestamp)
617 return get_passage($timestamp,FALSE);
619 function get_value($value)
621 return UTIME - $value;
624 ///////////////////////////////////////////////////////////////////////////
626 function plugin_tracker_list_convert()
631 $page = $refer = $vars['page'];
638 $args = func_get_args();
639 switch (count($args))
642 $limit = is_numeric($args[3]) ? $args[3] : $limit;
646 $args[1] = get_fullname($args[1],$page);
647 $page = is_pagename($args[1]) ? $args[1] : $page;
649 $config = ($args[0] != '') ? $args[0] : $config;
650 list($config,$list) = array_pad(explode('/',$config,2),2,$list);
653 return plugin_tracker_getlist($page,$refer,$config,$list,$order,$limit);
655 function plugin_tracker_list_action()
657 global $vars,$_tracker_messages;
659 $page = $refer = $vars['refer'];
660 $s_page = make_pagelink($page);
661 $config = $vars['config'];
662 $list = array_key_exists('list',$vars) ? $vars['list'] : 'list';
663 $order = array_key_exists('order',$vars) ? $vars['order'] : '_real:SORT_DESC';
666 'msg' => $_tracker_messages['msg_list'],
667 'body'=> str_replace('$1',$s_page,$_tracker_messages['msg_back']).
668 plugin_tracker_getlist($page,$refer,$config,$list,$order)
671 function plugin_tracker_getlist($page,$refer,$config_name,$list,$order='',$limit=NULL)
673 $config = new Config('plugin/tracker/'.$config_name);
675 if (!$config->read())
677 return "<p>config file '".htmlsc($config_name)."' is not exist.</p>";
680 $config->config_name = $config_name;
682 if (!is_page($config->page.'/'.$list))
684 return "<p>config file '".make_pagelink($config->page.'/'.$list)."' not found.</p>";
687 $list = new Tracker_list($page,$refer,$config,$list);
689 return $list->toString($limit);
705 function Tracker_list($page,$refer,&$config,$list)
707 $this->__construct($page, $refer, $config, $list);
709 function __construct($page,$refer,&$config,$list)
712 $this->config = &$config;
714 $this->fields = plugin_tracker_get_fields($page,$refer,$config);
716 $pattern = join('',plugin_tracker_get_source($config->page.'/page'));
717 // ブロックプラグインをフィールドに置換
718 // #commentなどで前後に文字列の増減があった場合に、[_block_xxx]に吸い込ませるようにする
719 $pattern = preg_replace('/^\#([^\(\s]+)(?:\((.*)\))?\s*$/m','[_block_$1]',$pattern);
723 $this->pattern_fields = array();
724 $pattern = preg_split('/\\\\\[(\w+)\\\\\]/',preg_quote($pattern,'/'),-1,PREG_SPLIT_DELIM_CAPTURE);
725 while (count($pattern))
727 $this->pattern .= preg_replace('/\s+/','\\s*','(?>\\s*'.trim(array_shift($pattern)).'\\s*)');
730 $field = array_shift($pattern);
731 $this->pattern_fields[] = $field;
732 $this->pattern .= '(.*?)';
736 $this->rows = array();
738 $pattern_len = strlen($pattern);
739 foreach (get_existpages() as $_page)
741 if (strpos($_page,$pattern) === 0)
743 $name = substr($_page,$pattern_len);
744 if (preg_match(TRACKER_LIST_EXCLUDE_PATTERN,$name))
748 $this->add($_page,$name);
752 function add($page,$name)
754 static $moved = array();
757 if (array_key_exists($name,$this->rows))
762 $source = plugin_tracker_get_source($page);
763 if (preg_match('/move\sto\s(.+)/',$source[0],$matches))
765 $page = strip_bracket(trim($matches[1]));
766 if (array_key_exists($page,$moved) or !is_page($page))
770 $moved[$page] = TRUE;
771 return $this->add($page,$name);
773 $source = join('',preg_replace('/^(\*{1,3}.*)\[#[A-Za-z][\w-]+\](.*)$/','$1$2',$source));
776 $this->rows[$name] = array(
777 '_page' => "[[$page]]",
778 '_refer' => $this->page,
780 '_update'=> get_filetime($page),
781 '_past' => get_filetime($page)
783 if ($this->rows[$name]['_match'] = preg_match("/{$this->pattern}/s",$source,$matches))
785 array_shift($matches);
786 foreach ($this->pattern_fields as $key=>$field)
788 $this->rows[$name][$field] = trim($matches[$key]);
792 function compare($a, $b)
794 foreach ($this->sort_keys as $sort_key)
796 $field = $sort_key['field'];
797 $dir = $sort_key['dir'];
798 $f = $this->fields[$field];
799 $sort_type = $f->sort_type;
800 $aVal = isset($a[$field]) ? $f->get_value($a[$field]) : '';
801 $bVal = isset($b[$field]) ? $f->get_value($b[$field]) : '';
802 $c = strnatcmp($aVal, $bVal) * ($dir === SORT_ASC ? 1 : -1);
803 if ($c === 0) continue;
808 function sort($order)
814 $names = array_flip(array_keys($this->fields));
815 $this->order = array();
816 foreach (explode(';',$order) as $item)
818 list($key,$dir) = array_pad(explode(':',$item),1,'ASC');
819 if (!array_key_exists($key,$names))
823 switch (strtoupper($dir))
838 $this->order[$key] = $dir;
840 $sort_keys = array();
841 foreach ($this->order as $field=>$order)
843 if (!array_key_exists($field,$names))
847 $sort_keys[] = array('field' => $field, 'dir' => $order);
849 $this->sort_keys = $sort_keys;
850 usort($this->rows, array($this, 'compare'));
852 function replace_item($arr)
854 $params = explode(',',$arr[1]);
855 $name = array_shift($params);
860 else if (array_key_exists($name,$this->items))
862 $str = $this->items[$name];
863 if (array_key_exists($name,$this->fields))
865 $str = $this->fields[$name]->format_cell($str);
870 return $this->pipe ? str_replace('|','|',$arr[0]) : $arr[0];
872 $style = count($params) ? $params[0] : $name;
873 if (array_key_exists($style,$this->items)
874 and array_key_exists($style,$this->fields))
876 $str = sprintf($this->fields[$style]->get_style($this->items[$style]),$str);
878 return $this->pipe ? str_replace('|','|',$str) : $str;
880 function replace_title($arr)
882 $script = get_base_uri();
883 $field = $sort = $arr[1];
884 if ($sort == '_name' or $sort == '_page')
888 if (!array_key_exists($field,$this->fields))
894 $order = $this->order;
896 if (is_array($order) && isset($order[$sort]))
898 // BugTrack2/106: Only variables can be passed by reference from PHP 5.0.5
899 $order_keys = array_keys($order); // with array_shift();
901 $index = array_flip($order_keys);
902 $pos = 1 + $index[$sort];
903 $b_end = ($sort == array_shift($order_keys));
904 $b_order = ($order[$sort] == SORT_ASC);
905 $dir = ($b_end xor $b_order) ? SORT_ASC : SORT_DESC;
906 $arrow = '&br;'.($b_order ? '↑' : '↓')."($pos)";
908 unset($order[$sort], $order_keys);
910 $title = $this->fields[$field]->title;
911 $r_page = rawurlencode($this->page);
912 $r_config = rawurlencode($this->config->config_name);
913 $r_list = rawurlencode($this->list);
914 $_order = array("$sort:$dir");
915 if (is_array($order))
916 foreach ($order as $key=>$value)
917 $_order[] = "$key:$value";
918 $r_order = rawurlencode(join(';',$_order));
920 return "[[$title$arrow>$script?plugin=tracker_list&refer=$r_page&config=$r_config&list=$r_list&order=$r_order]]";
922 function toString($limit=NULL)
924 global $_tracker_messages;
929 if ($limit !== NULL and count($this->rows) > $limit)
931 $source = str_replace(
933 array(count($this->rows),$limit),
934 $_tracker_messages['msg_limit'])."\n";
935 $this->rows = array_splice($this->rows,0,$limit);
937 if (count($this->rows) == 0)
941 foreach (plugin_tracker_get_source($this->config->page.'/'.$this->list) as $line)
943 if (preg_match('/^\|(.+)\|[hHfFcC]$/',$line))
945 $source .= preg_replace_callback('/\[([^\[\]]+)\]/',array(&$this,'replace_title'),$line);
952 foreach ($this->rows as $key=>$row)
954 if (!TRACKER_LIST_SHOW_ERROR_PAGE and !$row['_match'])
959 foreach ($body as $line)
961 if (trim($line) == '')
966 $this->pipe = ($line{0} == '|' or $line{0} == ':');
967 $source .= preg_replace_callback('/\[([^\[\]]+)\]/',array(&$this,'replace_item'),$line);
970 return convert_html($source);
973 function plugin_tracker_get_source($page)
975 $source = get_source($page);
976 // Delete anchor part of Headings (Example: "*Heading1 [#id] AAA" to "*Heading1 AAA")
977 $s2 = preg_replace('/^(\*{1,3}.*)\[#[A-Za-z][\w-]+\](.*)$/m','$1$2',$source);
979 $s3 = preg_replace('/^#freeze\s*$/im', '', $s2);
980 // Delete #author line
981 $s4 = preg_replace('/^#author\b[^\r\n]*$/im', '', $s3);