*
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* - Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
* - Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
* - Neither the name of the Chris Boulton nor the names of its contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*
* @package DiffLib
* @author Chris Boulton
* @copyright (c) 2009 Chris Boulton
* @license New BSD License http://www.opensource.org/licenses/bsd-license.php
* @version 1.1
* @link http://github.com/chrisboulton/php-diff
*/
require_once dirname ( __FILE__ ) . '/../Abstract.php';
class Diff_Renderer_Html_Array extends Diff_Renderer_Abstract {
/**
*
* @var array Array of the default options that apply to this renderer.
*/
protected $defaultOptions = array (
'tabSize' => 4
);
/**
* Render and return an array structure suitable for generating HTML
* based differences.
* Generally called by subclasses that generate a
* HTML based diff and return an array of the changes to show in the diff.
*
* @return array An array of the generated chances, suitable for presentation in HTML.
*/
public function render() {
// As we'll be modifying a & b to include our change markers,
// we need to get the contents and store them here. That way
// we're not going to destroy the original data
$a = $this->diff->getA ();
$b = $this->diff->getB ();
$changes = array ();
$opCodes = $this->diff->getGroupedOpcodes ();
foreach ( $opCodes as $group ) {
$blocks = array ();
$lastTag = null;
$lastBlock = 0;
foreach ( $group as $code ) {
list ( $tag, $i1, $i2, $j1, $j2 ) = $code;
if ($tag == 'replace' && $i2 - $i1 == $j2 - $j1) {
for($i = 0; $i < ($i2 - $i1); ++ $i) {
$fromLine = $a [$i1 + $i];
$toLine = $b [$j1 + $i];
list ( $start, $end ) = $this->getChangeExtent ( $fromLine, $toLine );
if ($start != 0 || $end != 0) {
$last = $end + strlen ( $fromLine );
$fromLine = substr_replace ( $fromLine, "\0", $start, 0 );
$fromLine = substr_replace ( $fromLine, "\1", $last + 1, 0 );
$last = $end + strlen ( $toLine );
$toLine = substr_replace ( $toLine, "\0", $start, 0 );
$toLine = substr_replace ( $toLine, "\1", $last + 1, 0 );
$a [$i1 + $i] = $fromLine;
$b [$j1 + $i] = $toLine;
}
}
}
if ($tag != $lastTag) {
$blocks [] = array (
'tag' => $tag,
'base' => array (
'offset' => $i1,
'lines' => array ()
),
'changed' => array (
'offset' => $j1,
'lines' => array ()
)
);
$lastBlock = count ( $blocks ) - 1;
}
$lastTag = $tag;
if ($tag == 'equal') {
$lines = array_slice ( $a, $i1, ($i2 - $i1) );
$blocks [$lastBlock] ['base'] ['lines'] += $this->formatLines ( $lines );
$lines = array_slice ( $b, $j1, ($j2 - $j1) );
$blocks [$lastBlock] ['changed'] ['lines'] += $this->formatLines ( $lines );
} else {
if ($tag == 'replace' || $tag == 'delete') {
$lines = array_slice ( $a, $i1, ($i2 - $i1) );
$lines = $this->formatLines ( $lines );
$lines = str_replace ( array (
"\0",
"\1"
), array (
'',
''
), $lines );
$blocks [$lastBlock] ['base'] ['lines'] += $lines;
}
if ($tag == 'replace' || $tag == 'insert') {
$lines = array_slice ( $b, $j1, ($j2 - $j1) );
$lines = $this->formatLines ( $lines );
$lines = str_replace ( array (
"\0",
"\1"
), array (
'',
''
), $lines );
$blocks [$lastBlock] ['changed'] ['lines'] += $lines;
}
}
}
$changes [] = $blocks;
}
return $changes;
}
/**
* Given two strings, determine where the changes in the two strings
* begin, and where the changes in the two strings end.
*
* @param string $fromLine
* The first string.
* @param string $toLine
* The second string.
* @return array Array containing the starting position (0 by default) and the ending position (-1 by default)
*/
private function getChangeExtent($fromLine, $toLine) {
$start = 0;
$limit = min ( strlen ( $fromLine ), strlen ( $toLine ) );
while ( $start < $limit && $fromLine {$start} == $toLine {$start} ) {
++ $start;
}
$end = - 1;
$limit = $limit - $start;
while ( - $end <= $limit && substr ( $fromLine, $end, 1 ) == substr ( $toLine, $end, 1 ) ) {
-- $end;
}
return array (
$start,
$end + 1
);
}
/**
* Format a series of lines suitable for output in a HTML rendered diff.
* This involves replacing tab characters with spaces, making the HTML safe
* for output, ensuring that double spaces are replaced with etc.
*
* @param array $lines
* Array of lines to format.
* @return array Array of the formatted lines.
*/
private function formatLines($lines) {
$lines = array_map ( array (
$this,
'ExpandTabs'
), $lines );
$lines = array_map ( array (
$this,
'HtmlSafe'
), $lines );
foreach ( $lines as &$line ) {
$line = preg_replace_callback ( '# ( +)|^ #', __CLASS__ . "::fixSpaces", $line );
}
return $lines;
}
/**
* Replace a string containing spaces with a HTML representation using .
*
* @param string $matches
* Regex matches array.
* @return string The HTML representation of the string.
*/
public static function fixSpaces($matches) {
$spaces = isset ( $matches [1] ) ? $matches [1] : '';
$count = strlen ( $spaces );
if ($count == 0) {
return '';
}
$div = floor ( $count / 2 );
$mod = $count % 2;
return str_repeat ( ' ', $div ) . str_repeat ( ' ', $mod );
}
/**
* Replace tabs in a single line with a number of spaces as defined by the tabSize option.
*
* @param string $line
* The containing tabs to convert.
* @return string The line with the tabs converted to spaces.
*/
private function expandTabs($line) {
return str_replace ( "\t", str_repeat ( ' ', $this->options ['tabSize'] ), $line );
}
/**
* Make a string containing HTML safe for output on a page.
*
* @param string $string
* The string.
* @return string The string with the HTML characters replaced by entities.
*/
private function htmlSafe($string) {
return htmlspecialchars ( $string, ENT_NOQUOTES, 'UTF-8' );
}
}