[addon todopanel] TodoPanel
- Mikulas Dite
- Člen | 756
No problém je v tom panelu, ale nevíme kde, protože je problém v Nette. Tzn. zkusím zjistit, proč tohle nezachytává Laděnka a potom se uvidí.
- thool
- Člen | 6
WOOOHOOO!
Podařilo se mi tohle rozjet na prefixovém nette 2.0-beta – PHP 5.2
změny:
- záměna funkcí
callback($this, "xxxx")
zaArray($this, "xxxx")
- dále oddělání jmenných prostorů
- přidání prefixů
- přejmenování třídy
- záměna
$ignoreMask ?: array( ...
na($ignoreMask) ? $ignoreMask : array( ...
použití:
v bootstrap.php přidat řádek:
TodoPanel::register( $container->params['appDir'] );
Na to jaký jsem Nette neználek si teď připadám jak nějaký guru :-)
<?php
/**
* TodoPanel for Nette 2.0
*
* @author Mikuláš Dítě, Peter Ped Helcmanovsky
* @license MIT
*/
/* //PHP 5.3
namespace Panel;
use Nette\Diagnostics\Debugger;
use Nette\Utils\Finder;
use Nette\Diagnostics\IBarPanel;
use Nette\Object;
use Nette\Utils\SafeStream;
use Nette\Utils\Strings as String;
use Nette\Templating\FileTemplate;
use Nette\Latte\Engine;
class Todo extends Object implements IBarPanel
*/
class TodoPanel extends NObject implements IBarPanel
{
/** @var array|mixed stores found todos */
private $items = array();
/** @var array */
private $scanDir = array();
/** @var array any path or file containing one of the patterns to skip */
private $ignoreMask = array();
/** @var array */
private $commentBlockMask = array();
/** @var array patterns for "todo" comments to catch */
public $todoMask = array('TO\s?DO', 'FIX\s?ME', 'PENDING', 'XXX');
/** @var bool defines wheter the todo type should be visible */
public $showType = FALSE;
/**
* @param array|string $basedir path or paths to scan
* @param array $ignoreMask can use wildcards
*/
public function __construct($basedir, $ignoreMask = NULL)
{
if (is_array($basedir)) {
foreach ($basedir as $path) {
$this->addDirectory(realpath($path));
}
} else {
$this->addDirectory(realpath($basedir));
}
$this->setIgnoreMask( ($ignoreMask) ? $ignoreMask : array('.git', '.svn', 'cache', 'log', 'sessions', 'temp'));
$patterns = array(
array('~^(php|css|js)$~', '~/\*(?P<content>.*?)\*/~sm'),
array('~^(php|js)$~', '~//(?P<content>.*?)$~sm'),
array('~^(php|sh|ps1)$~', '~#(?P<content>.*?)$~sm'),
array('~^(latte|phtml)$~', '~{\*(?P<content>.*?)\*}~sm'),
array('~^(latte|phtml|html)$~', '~<!--(?P<content>.*?)-->~sm'),
array('~^(ini)$~', '~;(?P<content>.*?)$~sm'),
array('~^(bat)$~', '~^[ \t]*REM[ \t]+(?P<content>.*?)$~smi'),
);
foreach ($patterns as $pattern) {
call_user_func_array( Array($this, 'addPattern'), $pattern);
}
}
/**
* Renders HTML code for custom tab
* IDebugPanel
* @return void
*/
public function getTab()
{
return '<img src="">' .
'Todo (' . $this->getTodoCount() . ')';
}
/**
* Renders HTML code for custom panel
* IDebugPanel
* @return void
*/
public function getPanel()
{
ob_start();
$template = new NFileTemplate(dirname(__FILE__) . '/bar.todo.panel.latte');
$template->registerFilter(new NLatteFilter());
$template->todos = $this->getTodo();
$template->todoCount = $this->getTodoCount();
$template->showType = $this->showType;
$template->render();
return ob_get_clean();
}
/**
* IDebugPanel
* @return string
*/
public function getId()
{
return __CLASS__;
}
/**
* Registers panel to Debug bar
*/
public static function register($basedir = NULL, $ignoreMask = NULL)
{
NDebugger::addPanel(new self($basedir, $ignoreMask));
}
/**
* Add directory to list
* @param string
* @return void
* @throws DirectoryNotFoundException
*/
public function addDirectory($path)
{
$realpath = realpath($path);
if (!$realpath) {
throw new NDirectoryNotFoundException("Directory `$path` not found.");
}
if (!array_search($realpath, $this->scanDir)) {
$this->scanDir[] = $realpath;
}
}
/**
* Adds custom comment block pattern
* @param string $extension regex
* @param string $pattern regex Must contain group named `content`
*/
public function addPattern($extension, $pattern)
{
self::validatePattern($pattern);
$this->commentBlockMask[] = array('extension' => $extension, 'pattern' => $pattern);
}
/**
* Throws exception if custom pattern does not name comment block content group
* @example pattern `~block_start(?P<content>.*?)block_end~sm`
* @throws \InvalidArgumentException
* @param string $pattern regex
*/
private static function validatePattern($pattern)
{
if (NStrings::match($pattern, '~\\(\\?P<content>\\.\\*\\?\\)~') === NULL) {
throw new NInvalidArgumentException('Custom pattern `' . $pattern . '` does not contain a group named `content`.');
}
}
/**
* Files to ignore
* @example $todoPanel->setIgnoreMask(array('.git', 'app/sessions'));
* @param array $ignoreMask
*/
public function setIgnoreMask(array $ignoreMask, $merge = FALSE)
{
if ($merge) {
foreach ($ignoreMask as $mask) {
if (!array_search($mask, $this->ignoreMask)) {
$this->ignoreMask[] = $mask;
}
}
} else {
$this->ignoreMask = $ignoreMask;
}
}
/**
* Sum of found todos in browsed files
* @return int
*/
public function getTodoCount()
{
$count = 0;
foreach ($this->getTodo() as $file) {
$count += count($file);
}
return $count;
}
/**
* usort implementation
* @param array $compared
* @param array $todo
* @return int
*/
public function compareTodos($compared, $todo)
{
if ($compared['line'] == $todo['line']) {
return 0;
}
return $compared['line'] < $todo['line'] ? -1 : 1;
}
/**
* Wrapper for generateTodo, performace booster in one instance
*/
private function getTodo()
{
if (empty($this->items)) {
$this->items = $this->generateTodo();
}
return $this->items;
}
/**
* Returns array in format $filename => array($todos)
* @uses \Nette\SafeStream
* @throws \Nette\InvalidStateException
*/
private function generateTodo()
{
if (count($this->todoMask) === 0) {
throw new NInvalidStateException('No todo mask specified for TodoPanel.');
}
@NSafeStream::register(); //intentionally @ (prevents multiple registration warning)
$items = array();
foreach (NFinder::findFiles('*')->size('> 3B')->exclude('.*', '*/' . $this->ignoreMask . '/*')->from($this->scanDir) as $path => $file) {
$items[$path] = $this->parseFile($file);
}
return $items;
}
/**
* Reads pointed file and returns all comments found
* @param SplFileInfo $file
* @returns array
*/
private function parseFile($file)
{
$todos = array();
$stream = fopen("safe://" . $file->getRealPath(), 'r');
$content_original = $content = fread($stream, filesize("safe://" . $file->getRealPath()));
fclose($stream);
// Remove harcoded strings so we do not search in them
$content = NStrings::replace($content, '~("|\')(.|\\\"|\\\')*?("|\')~s', '\'\'');
$patterns = array();
foreach ($this->commentBlockMask as $pattern) {
if (NStrings::match(pathinfo($file, PATHINFO_EXTENSION), $pattern['extension'])) {
$patterns[] = $pattern['pattern'];
}
}
$matches = array();
// find block comments
foreach ($patterns as $pattern) {
$matches = array_merge($matches, NStrings::matchAll($content, $pattern));
$content = NStrings::replace($content, $pattern);
}
$comment_lines = array();
// split block comments by lines
foreach ($matches as $match) {
$comment_block = NStrings::trim($match['content']);
$comment_lines = array_merge($comment_lines, NStrings::split($comment_block, '~[\r\n]{1,2}~'));
}
foreach ($comment_lines as $key => $comment_content) {
$match = NStrings::match($comment_content, '~(^[@*\s-]*|[@*\s-])(?P<type>' . implode('|', $this->todoMask) . ')\s+(?P<todo>.*?)$~mi');
if ($match === NULL) {
continue;
}
$skip = 0;
foreach ($comment_lines as $tkey => $line) {
if ($tkey >= $key) {
break;
}
if ($line == $comment_content) {
$skip++;
}
}
$line = 0;
// assign line number
foreach (NStrings::split($content_original, '~\n~') as $line_number => $content_line) {
if (strpos($content_line, $comment_content) !== FALSE) {
if ($skip > 0) {
$skip--;
continue;
}
$line = $line_number + 1;
break;
}
}
$todos[] = array(
'line' => $line,
'type' => NStrings::lower($match['type']),
'content' => $match['todo'],
'link' => strtr(NDebugger::$editor, array('%file' => urlencode($file->getRealPath()), '%line' => $line)),
'file' => $file->getFilename(),
);
}
usort($todos, Array($this, 'compareTodos'));
return $todos;
}
}
// nazev tridy poupraven
Editoval thool (18. 8. 2011 15:44)
- Mikulas Dite
- Člen | 756
Fajn, akorát prefix I
značí interface, což ta třída není
:). Takže když už, tak BarTodo
(a ještě lépe
TodoPanel
).
- Marax
- Člen | 28
Darkry napsal(a):
a server mi při načítání stránky vyhodí:Tato webová stránka není dostupná. Chyba 101(net::ERR_CONNECTION_RESET): Připojení bylo resetováno.
Přitom funkce register() se vykoná, protože pokud v ní udělám chybu hodí mi to laděnku.
Taky se mi už několikrát stala chyba 101 a vždycky nějak záhadně zmizela po tom co jsem se v tom dobu šťoural. Až teď jsem zjistil,že to bylo tímhle doplňkem.
Dal jsem si logovat všechny soubory na začátku a konci metody parseFile a jediný který neprošel a na kterém to skončilo byl config.neon.
Pokud v neonu smažu všechny komentáře tak to jde.
- Matúš Matula
- Člen | 257
Rovnaka chyba 101 aj u mna, blizsie sa mi ale nepodarilo zistit, co ju sposobuje :/
Mikulas Dite napsal(a):
No problém je v tom panelu, ale nevíme kde, protože je problém v Nette. Tzn. zkusím zjistit, proč tohle nezachytává Laděnka a potom se uvidí.
Ziadny pokrok ani u teba?
- Matúš Matula
- Člen | 257
Pouzivam tak, ako je v dokumentacii, teda
<?php
\Panel\Todo::register($this->context->params['appDir'], array('.git', '.svn', 'cache', 'log', 'sessions', 'temp', 'config', '*.neon', ));
?>
- dada-amater
- Bronze Partner | 52
Neplanujes moznost instalace pres Composer? Bylo by to moc fajn. Dik