Inheritance: extends Parser
Beispiel #1
0
 /**
  * @return void
  */
 public function process(Texy\BlockParser $parser, $content, Texy\HtmlElement $el)
 {
     if ($parser->isIndented()) {
         $parts = preg_split('#(\\n(?! )|\\n{2,})#', $content, -1, PREG_SPLIT_NO_EMPTY);
     } else {
         $parts = preg_split('#(\\n{2,})#', $content, -1, PREG_SPLIT_NO_EMPTY);
     }
     foreach ($parts as $s) {
         $s = trim($s);
         if ($s === '') {
             continue;
         }
         // try to find modifier
         $mod = NULL;
         if ($mx = Regexp::match($s, '#' . Texy\Patterns::MODIFIER_H . '(?=\\n|\\z)#sUm', Regexp::OFFSET_CAPTURE)) {
             list($mMod) = $mx[1];
             $s = trim(substr_replace($s, '', $mx[0][1], strlen($mx[0][0])));
             if ($s === '') {
                 continue;
             }
             $mod = new Texy\Modifier();
             $mod->setProperties($mMod);
         }
         $res = $this->texy->invokeAroundHandlers('paragraph', $parser, [$s, $mod]);
         if ($res) {
             $el->insert(NULL, $res);
         }
     }
 }
Beispiel #2
0
 /**
  * Callback for:.
  *
  * > They went in single file, running like hounds on a strong scent,
  * and an eager light was in their eyes. Nearly due west the broad
  * swath of the marching Orcs tramped its ugly slot; the sweet grass
  * of Rohan had been bruised and blackened as they passed.
  * >:http://www.mycom.com/tolkien/twotowers.html
  *
  * @return Texy\HtmlElement|string|FALSE
  */
 public function pattern(Texy\BlockParser $parser, array $matches)
 {
     list(, $mMod, $mPrefix, $mContent) = $matches;
     // [1] => .(title)[class]{style}<>
     // [2] => spaces |
     // [3] => ... / LINK
     $texy = $this->texy;
     $el = new Texy\HtmlElement('blockquote');
     $mod = new Texy\Modifier($mMod);
     $mod->decorate($texy, $el);
     $content = '';
     $spaces = '';
     do {
         if ($mPrefix === ':') {
             $mod->cite = $texy->blockQuoteModule->citeLink($mContent);
             $content .= "\n";
         } else {
             if ($spaces === '') {
                 $spaces = max(1, strlen($mPrefix));
             }
             $content .= $mContent . "\n";
         }
         if (!$parser->next("#^>(?:|(\\ {1,{$spaces}}|:)(.*))()\$#mA", $matches)) {
             break;
         }
         /*
         			if ($mPrefix === '>') {
         				$content .= $mPrefix . $mContent . "\n";
         			} elseif ($mPrefix === ':') {
         				$mod->cite = $texy->blockQuoteModule->citeLink($mContent);
         				$content .= "\n";
         			} else {
         				if ($spaces === '') $spaces = max(1, strlen($mPrefix));
         				$content .= $mContent . "\n";
         			}
         			if (!$parser->next("#^\\>(?:(\\>|\\ {1,$spaces}|:)(.*))?()$#mA", $matches)) break;
         */
         list(, $mPrefix, $mContent) = $matches;
     } while (TRUE);
     $el->attrs['cite'] = $mod->cite;
     $el->parseBlock($texy, $content, $parser->isIndented());
     // no content?
     if (!$el->count()) {
         return FALSE;
     }
     // event listener
     $texy->invokeHandlers('afterBlockquote', [$parser, $el, $mod]);
     return $el;
 }
Beispiel #3
0
 /**
  * Callback for:.
  *
  * .(title)[class]{style}>
  * |------------------
  * | xxx | xxx | xxx | .(..){..}[..]
  * |------------------
  * | aa | bb | cc |
  *
  * @return HtmlElement|string|FALSE
  */
 public function patternTable(Texy\BlockParser $parser, array $matches)
 {
     if ($this->disableTables) {
         return FALSE;
     }
     list(, $mMod) = $matches;
     // [1] => .(title)[class]{style}<>_
     $texy = $this->texy;
     $el = new HtmlElement('table');
     $mod = new Modifier($mMod);
     $mod->decorate($texy, $el);
     $parser->moveBackward();
     if ($parser->next('#^\\|(\\#|\\=){2,}(?![|\\#=+])(.+)\\1*\\|? *' . Patterns::MODIFIER_H . '?()$#Um', $matches)) {
         list(, , $mContent, $mMod) = $matches;
         // [1] => # / =
         // [2] => ....
         // [3] => .(title)[class]{style}<>
         $caption = $el->create('caption');
         $mod = new Modifier($mMod);
         $mod->decorate($texy, $caption);
         $caption->parseLine($texy, $mContent);
     }
     $isHead = FALSE;
     $colModifier = [];
     $prevRow = [];
     // rowSpan building helper
     $rowCounter = 0;
     $colCounter = 0;
     $elPart = NULL;
     $lineMode = FALSE;
     // rows must be separated by lines
     while (TRUE) {
         if ($parser->next('#^\\|([=-])[+|=-]{2,}$#Um', $matches)) {
             // line
             if ($lineMode) {
                 if ($matches[1] === '=') {
                     $isHead = !$isHead;
                 }
             } else {
                 $isHead = !$isHead;
                 $lineMode = $matches[1] === '=';
             }
             $prevRow = [];
             continue;
         }
         if ($parser->next('#^\\|(.*)(?:|\\|\\ *' . Patterns::MODIFIER_HV . '?)()$#U', $matches)) {
             // smarter head detection
             if ($rowCounter === 0 && !$isHead && $parser->next('#^\\|[=-][+|=-]{2,}$#Um', $foo)) {
                 $isHead = TRUE;
                 $parser->moveBackward();
             }
             if ($elPart === NULL) {
                 $elPart = $el->create($isHead ? 'thead' : 'tbody');
             } elseif (!$isHead && $elPart->getName() === 'thead') {
                 $this->finishPart($elPart);
                 $elPart = $el->create('tbody');
             }
             // PARSE ROW
             list(, $mContent, $mMod) = $matches;
             // [1] => ....
             // [2] => .(title)[class]{style}<>_
             $elRow = new HtmlElement('tr');
             $mod = new Modifier($mMod);
             $mod->decorate($texy, $elRow);
             $rowClass = $rowCounter % 2 === 0 ? $this->oddClass : $this->evenClass;
             if ($rowClass && !isset($mod->classes[$this->oddClass]) && !isset($mod->classes[$this->evenClass])) {
                 $elRow->attrs['class'][] = $rowClass;
             }
             $col = 0;
             $elCell = NULL;
             // special escape sequence \|
             $mContent = str_replace('\\|', "", $mContent);
             $mContent = Regexp::replace($mContent, '#(\\[[^\\]]*)\\|#', "\$1");
             // HACK: support for [..|..]
             foreach (explode('|', $mContent) as $cell) {
                 $cell = strtr($cell, "", '|');
                 // rowSpan
                 if (isset($prevRow[$col]) && ($lineMode || ($matches = Regexp::match($cell, '#\\^\\ *$|\\*??(.*)\\ +\\^$#AU')))) {
                     $prevRow[$col]->rowSpan++;
                     if (!$lineMode) {
                         $cell = isset($matches[1]) ? $matches[1] : '';
                     }
                     $prevRow[$col]->text .= "\n" . $cell;
                     $col += $prevRow[$col]->colSpan;
                     $elCell = NULL;
                     continue;
                 }
                 // colSpan
                 if ($cell === '' && $elCell) {
                     $elCell->colSpan++;
                     unset($prevRow[$col]);
                     $col++;
                     continue;
                 }
                 // common cell
                 $matches = Regexp::match($cell, '#(\\*??)\\ *' . Patterns::MODIFIER_HV . '??(.*)' . Patterns::MODIFIER_HV . '?\\ *()$#AU');
                 if (!$matches) {
                     continue;
                 }
                 list(, $mHead, $mModCol, $mContent, $mMod) = $matches;
                 // [1] => * ^
                 // [2] => .(title)[class]{style}<>_
                 // [3] => ....
                 // [4] => .(title)[class]{style}<>_
                 if ($mModCol) {
                     $colModifier[$col] = new Modifier($mModCol);
                 }
                 if (isset($colModifier[$col])) {
                     $mod = clone $colModifier[$col];
                 } else {
                     $mod = new Modifier();
                 }
                 $mod->setProperties($mMod);
                 $elCell = new TableCellElement();
                 $elCell->setName($isHead || $mHead === '*' ? 'th' : 'td');
                 $mod->decorate($texy, $elCell);
                 $elCell->text = $mContent;
                 $elRow->add($elCell);
                 $prevRow[$col] = $elCell;
                 $col++;
             }
             // even up with empty cells
             while ($col < $colCounter) {
                 if (isset($prevRow[$col]) && $lineMode) {
                     $prevRow[$col]->rowSpan++;
                     $prevRow[$col]->text .= "\n";
                 } else {
                     $elCell = new TableCellElement();
                     $elCell->setName($isHead ? 'th' : 'td');
                     if (isset($colModifier[$col])) {
                         $colModifier[$col]->decorate($texy, $elCell);
                     }
                     $elRow->add($elCell);
                     $prevRow[$col] = $elCell;
                 }
                 $col++;
             }
             $colCounter = $col;
             if ($elRow->count()) {
                 $elPart->add($elRow);
                 $rowCounter++;
             } else {
                 // redundant row
                 foreach ($prevRow as $elCell) {
                     $elCell->rowSpan--;
                 }
             }
             continue;
         }
         break;
     }
     if ($elPart === NULL) {
         // invalid table
         return FALSE;
     }
     if ($elPart->getName() === 'thead') {
         // thead is optional, tbody is required
         $elPart->setName('tbody');
     }
     $this->finishPart($elPart);
     // event listener
     $texy->invokeHandlers('afterTable', [$parser, $el, $mod]);
     return $el;
 }
Beispiel #4
0
 /**
  * Callback for single list item.
  * @return HtmlElement|FALSE
  */
 public function patternItem(BlockParser $parser, $bullet, $indented, $tag)
 {
     $spacesBase = $indented ? '\\ {1,}' : '';
     $patternItem = "#^\n?({$spacesBase}){$bullet}\\ *(\\S.*)?" . Patterns::MODIFIER_H . '?()$#mAUu';
     // first line with bullet
     $matches = NULL;
     if (!$parser->next($patternItem, $matches)) {
         return FALSE;
     }
     list(, $mIndent, $mContent, $mMod) = $matches;
     // [1] => indent
     // [2] => ...
     // [3] => .(title)[class]{style}<>
     $elItem = new HtmlElement($tag);
     $mod = new Modifier($mMod);
     $mod->decorate($this->texy, $elItem);
     // next lines
     $spaces = '';
     $content = ' ' . $mContent;
     // trick
     while ($parser->next('#^(\\n*)' . $mIndent . '(\\ {1,' . $spaces . '})(.*)()$#Am', $matches)) {
         list(, $mBlank, $mSpaces, $mContent) = $matches;
         // [1] => blank line?
         // [2] => spaces
         // [3] => ...
         if ($spaces === '') {
             $spaces = strlen($mSpaces);
         }
         $content .= "\n" . $mBlank . $mContent;
     }
     // parse content
     $elItem->parseBlock($this->texy, $content, TRUE);
     if (isset($elItem[0]) && $elItem[0] instanceof HtmlElement) {
         $elItem[0]->setName(NULL);
     }
     return $elItem;
 }