wiki.techinc.nl/includes/diff/DiffFormatter.php
Siebrand Mazeland abefd2c07e Update formatting for includes/diff
Change-Id: Id032a32ca6ebea3daf5313a05602c5e98603d983
2013-11-20 20:11:57 +01:00

225 lines
5.1 KiB
PHP

<?php
/**
* Base for diff rendering classes. Portions taken from phpwiki-1.3.3.
*
* Copyright © 2000, 2001 Geoffrey T. Dairiki <dairiki@dairiki.org>
* You may copy this code freely under the conditions of the GPL.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
* http://www.gnu.org/copyleft/gpl.html
*
* @file
* @ingroup DifferenceEngine
*/
/**
* Base class for diff formatters
*
* This class formats the diff in classic diff format.
* It is intended that this class be customized via inheritance,
* to obtain fancier outputs.
* @todo document
* @ingroup DifferenceEngine
*/
abstract class DiffFormatter {
/** @var int Number of leading context "lines" to preserve.
*
* This should be left at zero for this class, but subclasses
* may want to set this to other values.
*/
protected $leadingContextLines = 0;
/** @var int Number of trailing context "lines" to preserve.
*
* This should be left at zero for this class, but subclasses
* may want to set this to other values.
*/
protected $trailingContextLines = 0;
/**
* Format a diff.
*
* @param $diff Diff A Diff object.
* @return string The formatted output.
*/
public function format( $diff ) {
wfProfileIn( __METHOD__ );
$xi = $yi = 1;
$block = false;
$context = array();
$nlead = $this->leadingContextLines;
$ntrail = $this->trailingContextLines;
$this->startDiff();
foreach ( $diff->edits as $edit ) {
if ( $edit->type == 'copy' ) {
if ( is_array( $block ) ) {
if ( count( $edit->orig ) <= $nlead + $ntrail ) {
$block[] = $edit;
} else {
if ( $ntrail ) {
$context = array_slice( $edit->orig, 0, $ntrail );
$block[] = new DiffOp_Copy( $context );
}
$this->block( $x0, $ntrail + $xi - $x0,
$y0, $ntrail + $yi - $y0,
$block );
$block = false;
}
}
$context = $edit->orig;
} else {
if ( !is_array( $block ) ) {
$context = array_slice( $context, count( $context ) - $nlead );
$x0 = $xi - count( $context );
$y0 = $yi - count( $context );
$block = array();
if ( $context ) {
$block[] = new DiffOp_Copy( $context );
}
}
$block[] = $edit;
}
if ( $edit->orig ) {
$xi += count( $edit->orig );
}
if ( $edit->closing ) {
$yi += count( $edit->closing );
}
}
if ( is_array( $block ) ) {
$this->block( $x0, $xi - $x0,
$y0, $yi - $y0,
$block );
}
$end = $this->endDiff();
wfProfileOut( __METHOD__ );
return $end;
}
/**
* @param $xbeg
* @param $xlen
* @param $ybeg
* @param $ylen
* @param $edits
*/
protected function block( $xbeg, $xlen, $ybeg, $ylen, &$edits ) {
wfProfileIn( __METHOD__ );
$this->startBlock( $this->blockHeader( $xbeg, $xlen, $ybeg, $ylen ) );
foreach ( $edits as $edit ) {
if ( $edit->type == 'copy' ) {
$this->context( $edit->orig );
} elseif ( $edit->type == 'add' ) {
$this->added( $edit->closing );
} elseif ( $edit->type == 'delete' ) {
$this->deleted( $edit->orig );
} elseif ( $edit->type == 'change' ) {
$this->changed( $edit->orig, $edit->closing );
} else {
throw new MWException( "Unknown edit type: {$edit->type}" );
}
}
$this->endBlock();
wfProfileOut( __METHOD__ );
}
protected function startDiff() {
ob_start();
}
/**
* @return string
*/
protected function endDiff() {
$val = ob_get_contents();
ob_end_clean();
return $val;
}
/**
* @param $xbeg
* @param $xlen
* @param $ybeg
* @param $ylen
* @return string
*/
protected function blockHeader( $xbeg, $xlen, $ybeg, $ylen ) {
if ( $xlen > 1 ) {
$xbeg .= ',' . ( $xbeg + $xlen - 1 );
}
if ( $ylen > 1 ) {
$ybeg .= ',' . ( $ybeg + $ylen - 1 );
}
return $xbeg . ( $xlen ? ( $ylen ? 'c' : 'd' ) : 'a' ) . $ybeg;
}
protected function startBlock( $header ) {
echo $header . "\n";
}
protected function endBlock() {
}
/**
* @param $lines
* @param $prefix string
*/
protected function lines( $lines, $prefix = ' ' ) {
foreach ( $lines as $line ) {
echo "$prefix $line\n";
}
}
/**
* @param $lines
*/
protected function context( $lines ) {
$this->lines( $lines );
}
/**
* @param $lines
*/
protected function added( $lines ) {
$this->lines( $lines, '>' );
}
/**
* @param $lines
*/
protected function deleted( $lines ) {
$this->lines( $lines, '<' );
}
/**
* @param $orig
* @param $closing
*/
protected function changed( $orig, $closing ) {
$this->deleted( $orig );
echo "---\n";
$this->added( $closing );
}
}