web/wp-includes/Text/Diff/Renderer.php
author ymh
Fri, 12 Mar 2010 13:29:04 +0000
changeset 1 0d28b7c10758
permissions -rw-r--r--
First commit
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
1
0d28b7c10758 First commit
ymh
parents:
diff changeset
     1
<?php
0d28b7c10758 First commit
ymh
parents:
diff changeset
     2
/**
0d28b7c10758 First commit
ymh
parents:
diff changeset
     3
 * A class to render Diffs in different formats.
0d28b7c10758 First commit
ymh
parents:
diff changeset
     4
 *
0d28b7c10758 First commit
ymh
parents:
diff changeset
     5
 * This class renders the diff in classic diff format. It is intended that
0d28b7c10758 First commit
ymh
parents:
diff changeset
     6
 * this class be customized via inheritance, to obtain fancier outputs.
0d28b7c10758 First commit
ymh
parents:
diff changeset
     7
 *
0d28b7c10758 First commit
ymh
parents:
diff changeset
     8
 * $Horde: framework/Text_Diff/Diff/Renderer.php,v 1.21 2008/01/04 10:07:50 jan Exp $
0d28b7c10758 First commit
ymh
parents:
diff changeset
     9
 *
0d28b7c10758 First commit
ymh
parents:
diff changeset
    10
 * Copyright 2004-2008 The Horde Project (http://www.horde.org/)
0d28b7c10758 First commit
ymh
parents:
diff changeset
    11
 *
0d28b7c10758 First commit
ymh
parents:
diff changeset
    12
 * See the enclosed file COPYING for license information (LGPL). If you did
0d28b7c10758 First commit
ymh
parents:
diff changeset
    13
 * not receive this file, see http://opensource.org/licenses/lgpl-license.php.
0d28b7c10758 First commit
ymh
parents:
diff changeset
    14
 *
0d28b7c10758 First commit
ymh
parents:
diff changeset
    15
 * @package Text_Diff
0d28b7c10758 First commit
ymh
parents:
diff changeset
    16
 */
0d28b7c10758 First commit
ymh
parents:
diff changeset
    17
class Text_Diff_Renderer {
0d28b7c10758 First commit
ymh
parents:
diff changeset
    18
0d28b7c10758 First commit
ymh
parents:
diff changeset
    19
    /**
0d28b7c10758 First commit
ymh
parents:
diff changeset
    20
     * Number of leading context "lines" to preserve.
0d28b7c10758 First commit
ymh
parents:
diff changeset
    21
     *
0d28b7c10758 First commit
ymh
parents:
diff changeset
    22
     * This should be left at zero for this class, but subclasses may want to
0d28b7c10758 First commit
ymh
parents:
diff changeset
    23
     * set this to other values.
0d28b7c10758 First commit
ymh
parents:
diff changeset
    24
     */
0d28b7c10758 First commit
ymh
parents:
diff changeset
    25
    var $_leading_context_lines = 0;
0d28b7c10758 First commit
ymh
parents:
diff changeset
    26
0d28b7c10758 First commit
ymh
parents:
diff changeset
    27
    /**
0d28b7c10758 First commit
ymh
parents:
diff changeset
    28
     * Number of trailing context "lines" to preserve.
0d28b7c10758 First commit
ymh
parents:
diff changeset
    29
     *
0d28b7c10758 First commit
ymh
parents:
diff changeset
    30
     * This should be left at zero for this class, but subclasses may want to
0d28b7c10758 First commit
ymh
parents:
diff changeset
    31
     * set this to other values.
0d28b7c10758 First commit
ymh
parents:
diff changeset
    32
     */
0d28b7c10758 First commit
ymh
parents:
diff changeset
    33
    var $_trailing_context_lines = 0;
0d28b7c10758 First commit
ymh
parents:
diff changeset
    34
0d28b7c10758 First commit
ymh
parents:
diff changeset
    35
    /**
0d28b7c10758 First commit
ymh
parents:
diff changeset
    36
     * Constructor.
0d28b7c10758 First commit
ymh
parents:
diff changeset
    37
     */
0d28b7c10758 First commit
ymh
parents:
diff changeset
    38
    function Text_Diff_Renderer($params = array())
0d28b7c10758 First commit
ymh
parents:
diff changeset
    39
    {
0d28b7c10758 First commit
ymh
parents:
diff changeset
    40
        foreach ($params as $param => $value) {
0d28b7c10758 First commit
ymh
parents:
diff changeset
    41
            $v = '_' . $param;
0d28b7c10758 First commit
ymh
parents:
diff changeset
    42
            if (isset($this->$v)) {
0d28b7c10758 First commit
ymh
parents:
diff changeset
    43
                $this->$v = $value;
0d28b7c10758 First commit
ymh
parents:
diff changeset
    44
            }
0d28b7c10758 First commit
ymh
parents:
diff changeset
    45
        }
0d28b7c10758 First commit
ymh
parents:
diff changeset
    46
    }
0d28b7c10758 First commit
ymh
parents:
diff changeset
    47
0d28b7c10758 First commit
ymh
parents:
diff changeset
    48
    /**
0d28b7c10758 First commit
ymh
parents:
diff changeset
    49
     * Get any renderer parameters.
0d28b7c10758 First commit
ymh
parents:
diff changeset
    50
     *
0d28b7c10758 First commit
ymh
parents:
diff changeset
    51
     * @return array  All parameters of this renderer object.
0d28b7c10758 First commit
ymh
parents:
diff changeset
    52
     */
0d28b7c10758 First commit
ymh
parents:
diff changeset
    53
    function getParams()
0d28b7c10758 First commit
ymh
parents:
diff changeset
    54
    {
0d28b7c10758 First commit
ymh
parents:
diff changeset
    55
        $params = array();
0d28b7c10758 First commit
ymh
parents:
diff changeset
    56
        foreach (get_object_vars($this) as $k => $v) {
0d28b7c10758 First commit
ymh
parents:
diff changeset
    57
            if ($k[0] == '_') {
0d28b7c10758 First commit
ymh
parents:
diff changeset
    58
                $params[substr($k, 1)] = $v;
0d28b7c10758 First commit
ymh
parents:
diff changeset
    59
            }
0d28b7c10758 First commit
ymh
parents:
diff changeset
    60
        }
0d28b7c10758 First commit
ymh
parents:
diff changeset
    61
0d28b7c10758 First commit
ymh
parents:
diff changeset
    62
        return $params;
0d28b7c10758 First commit
ymh
parents:
diff changeset
    63
    }
0d28b7c10758 First commit
ymh
parents:
diff changeset
    64
0d28b7c10758 First commit
ymh
parents:
diff changeset
    65
    /**
0d28b7c10758 First commit
ymh
parents:
diff changeset
    66
     * Renders a diff.
0d28b7c10758 First commit
ymh
parents:
diff changeset
    67
     *
0d28b7c10758 First commit
ymh
parents:
diff changeset
    68
     * @param Text_Diff $diff  A Text_Diff object.
0d28b7c10758 First commit
ymh
parents:
diff changeset
    69
     *
0d28b7c10758 First commit
ymh
parents:
diff changeset
    70
     * @return string  The formatted output.
0d28b7c10758 First commit
ymh
parents:
diff changeset
    71
     */
0d28b7c10758 First commit
ymh
parents:
diff changeset
    72
    function render($diff)
0d28b7c10758 First commit
ymh
parents:
diff changeset
    73
    {
0d28b7c10758 First commit
ymh
parents:
diff changeset
    74
        $xi = $yi = 1;
0d28b7c10758 First commit
ymh
parents:
diff changeset
    75
        $block = false;
0d28b7c10758 First commit
ymh
parents:
diff changeset
    76
        $context = array();
0d28b7c10758 First commit
ymh
parents:
diff changeset
    77
0d28b7c10758 First commit
ymh
parents:
diff changeset
    78
        $nlead = $this->_leading_context_lines;
0d28b7c10758 First commit
ymh
parents:
diff changeset
    79
        $ntrail = $this->_trailing_context_lines;
0d28b7c10758 First commit
ymh
parents:
diff changeset
    80
0d28b7c10758 First commit
ymh
parents:
diff changeset
    81
        $output = $this->_startDiff();
0d28b7c10758 First commit
ymh
parents:
diff changeset
    82
0d28b7c10758 First commit
ymh
parents:
diff changeset
    83
        $diffs = $diff->getDiff();
0d28b7c10758 First commit
ymh
parents:
diff changeset
    84
        foreach ($diffs as $i => $edit) {
0d28b7c10758 First commit
ymh
parents:
diff changeset
    85
            /* If these are unchanged (copied) lines, and we want to keep
0d28b7c10758 First commit
ymh
parents:
diff changeset
    86
             * leading or trailing context lines, extract them from the copy
0d28b7c10758 First commit
ymh
parents:
diff changeset
    87
             * block. */
0d28b7c10758 First commit
ymh
parents:
diff changeset
    88
            if (is_a($edit, 'Text_Diff_Op_copy')) {
0d28b7c10758 First commit
ymh
parents:
diff changeset
    89
                /* Do we have any diff blocks yet? */
0d28b7c10758 First commit
ymh
parents:
diff changeset
    90
                if (is_array($block)) {
0d28b7c10758 First commit
ymh
parents:
diff changeset
    91
                    /* How many lines to keep as context from the copy
0d28b7c10758 First commit
ymh
parents:
diff changeset
    92
                     * block. */
0d28b7c10758 First commit
ymh
parents:
diff changeset
    93
                    $keep = $i == count($diffs) - 1 ? $ntrail : $nlead + $ntrail;
0d28b7c10758 First commit
ymh
parents:
diff changeset
    94
                    if (count($edit->orig) <= $keep) {
0d28b7c10758 First commit
ymh
parents:
diff changeset
    95
                        /* We have less lines in the block than we want for
0d28b7c10758 First commit
ymh
parents:
diff changeset
    96
                         * context => keep the whole block. */
0d28b7c10758 First commit
ymh
parents:
diff changeset
    97
                        $block[] = $edit;
0d28b7c10758 First commit
ymh
parents:
diff changeset
    98
                    } else {
0d28b7c10758 First commit
ymh
parents:
diff changeset
    99
                        if ($ntrail) {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   100
                            /* Create a new block with as many lines as we need
0d28b7c10758 First commit
ymh
parents:
diff changeset
   101
                             * for the trailing context. */
0d28b7c10758 First commit
ymh
parents:
diff changeset
   102
                            $context = array_slice($edit->orig, 0, $ntrail);
0d28b7c10758 First commit
ymh
parents:
diff changeset
   103
                            $block[] = &new Text_Diff_Op_copy($context);
0d28b7c10758 First commit
ymh
parents:
diff changeset
   104
                        }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   105
                        /* @todo */
0d28b7c10758 First commit
ymh
parents:
diff changeset
   106
                        $output .= $this->_block($x0, $ntrail + $xi - $x0,
0d28b7c10758 First commit
ymh
parents:
diff changeset
   107
                                                 $y0, $ntrail + $yi - $y0,
0d28b7c10758 First commit
ymh
parents:
diff changeset
   108
                                                 $block);
0d28b7c10758 First commit
ymh
parents:
diff changeset
   109
                        $block = false;
0d28b7c10758 First commit
ymh
parents:
diff changeset
   110
                    }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   111
                }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   112
                /* Keep the copy block as the context for the next block. */
0d28b7c10758 First commit
ymh
parents:
diff changeset
   113
                $context = $edit->orig;
0d28b7c10758 First commit
ymh
parents:
diff changeset
   114
            } else {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   115
                /* Don't we have any diff blocks yet? */
0d28b7c10758 First commit
ymh
parents:
diff changeset
   116
                if (!is_array($block)) {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   117
                    /* Extract context lines from the preceding copy block. */
0d28b7c10758 First commit
ymh
parents:
diff changeset
   118
                    $context = array_slice($context, count($context) - $nlead);
0d28b7c10758 First commit
ymh
parents:
diff changeset
   119
                    $x0 = $xi - count($context);
0d28b7c10758 First commit
ymh
parents:
diff changeset
   120
                    $y0 = $yi - count($context);
0d28b7c10758 First commit
ymh
parents:
diff changeset
   121
                    $block = array();
0d28b7c10758 First commit
ymh
parents:
diff changeset
   122
                    if ($context) {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   123
                        $block[] = &new Text_Diff_Op_copy($context);
0d28b7c10758 First commit
ymh
parents:
diff changeset
   124
                    }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   125
                }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   126
                $block[] = $edit;
0d28b7c10758 First commit
ymh
parents:
diff changeset
   127
            }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   128
0d28b7c10758 First commit
ymh
parents:
diff changeset
   129
            if ($edit->orig) {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   130
                $xi += count($edit->orig);
0d28b7c10758 First commit
ymh
parents:
diff changeset
   131
            }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   132
            if ($edit->final) {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   133
                $yi += count($edit->final);
0d28b7c10758 First commit
ymh
parents:
diff changeset
   134
            }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   135
        }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   136
0d28b7c10758 First commit
ymh
parents:
diff changeset
   137
        if (is_array($block)) {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   138
            $output .= $this->_block($x0, $xi - $x0,
0d28b7c10758 First commit
ymh
parents:
diff changeset
   139
                                     $y0, $yi - $y0,
0d28b7c10758 First commit
ymh
parents:
diff changeset
   140
                                     $block);
0d28b7c10758 First commit
ymh
parents:
diff changeset
   141
        }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   142
0d28b7c10758 First commit
ymh
parents:
diff changeset
   143
        return $output . $this->_endDiff();
0d28b7c10758 First commit
ymh
parents:
diff changeset
   144
    }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   145
0d28b7c10758 First commit
ymh
parents:
diff changeset
   146
    function _block($xbeg, $xlen, $ybeg, $ylen, &$edits)
0d28b7c10758 First commit
ymh
parents:
diff changeset
   147
    {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   148
        $output = $this->_startBlock($this->_blockHeader($xbeg, $xlen, $ybeg, $ylen));
0d28b7c10758 First commit
ymh
parents:
diff changeset
   149
0d28b7c10758 First commit
ymh
parents:
diff changeset
   150
        foreach ($edits as $edit) {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   151
            switch (strtolower(get_class($edit))) {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   152
            case 'text_diff_op_copy':
0d28b7c10758 First commit
ymh
parents:
diff changeset
   153
                $output .= $this->_context($edit->orig);
0d28b7c10758 First commit
ymh
parents:
diff changeset
   154
                break;
0d28b7c10758 First commit
ymh
parents:
diff changeset
   155
0d28b7c10758 First commit
ymh
parents:
diff changeset
   156
            case 'text_diff_op_add':
0d28b7c10758 First commit
ymh
parents:
diff changeset
   157
                $output .= $this->_added($edit->final);
0d28b7c10758 First commit
ymh
parents:
diff changeset
   158
                break;
0d28b7c10758 First commit
ymh
parents:
diff changeset
   159
0d28b7c10758 First commit
ymh
parents:
diff changeset
   160
            case 'text_diff_op_delete':
0d28b7c10758 First commit
ymh
parents:
diff changeset
   161
                $output .= $this->_deleted($edit->orig);
0d28b7c10758 First commit
ymh
parents:
diff changeset
   162
                break;
0d28b7c10758 First commit
ymh
parents:
diff changeset
   163
0d28b7c10758 First commit
ymh
parents:
diff changeset
   164
            case 'text_diff_op_change':
0d28b7c10758 First commit
ymh
parents:
diff changeset
   165
                $output .= $this->_changed($edit->orig, $edit->final);
0d28b7c10758 First commit
ymh
parents:
diff changeset
   166
                break;
0d28b7c10758 First commit
ymh
parents:
diff changeset
   167
            }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   168
        }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   169
0d28b7c10758 First commit
ymh
parents:
diff changeset
   170
        return $output . $this->_endBlock();
0d28b7c10758 First commit
ymh
parents:
diff changeset
   171
    }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   172
0d28b7c10758 First commit
ymh
parents:
diff changeset
   173
    function _startDiff()
0d28b7c10758 First commit
ymh
parents:
diff changeset
   174
    {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   175
        return '';
0d28b7c10758 First commit
ymh
parents:
diff changeset
   176
    }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   177
0d28b7c10758 First commit
ymh
parents:
diff changeset
   178
    function _endDiff()
0d28b7c10758 First commit
ymh
parents:
diff changeset
   179
    {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   180
        return '';
0d28b7c10758 First commit
ymh
parents:
diff changeset
   181
    }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   182
0d28b7c10758 First commit
ymh
parents:
diff changeset
   183
    function _blockHeader($xbeg, $xlen, $ybeg, $ylen)
0d28b7c10758 First commit
ymh
parents:
diff changeset
   184
    {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   185
        if ($xlen > 1) {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   186
            $xbeg .= ',' . ($xbeg + $xlen - 1);
0d28b7c10758 First commit
ymh
parents:
diff changeset
   187
        }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   188
        if ($ylen > 1) {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   189
            $ybeg .= ',' . ($ybeg + $ylen - 1);
0d28b7c10758 First commit
ymh
parents:
diff changeset
   190
        }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   191
0d28b7c10758 First commit
ymh
parents:
diff changeset
   192
        // this matches the GNU Diff behaviour
0d28b7c10758 First commit
ymh
parents:
diff changeset
   193
        if ($xlen && !$ylen) {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   194
            $ybeg--;
0d28b7c10758 First commit
ymh
parents:
diff changeset
   195
        } elseif (!$xlen) {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   196
            $xbeg--;
0d28b7c10758 First commit
ymh
parents:
diff changeset
   197
        }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   198
0d28b7c10758 First commit
ymh
parents:
diff changeset
   199
        return $xbeg . ($xlen ? ($ylen ? 'c' : 'd') : 'a') . $ybeg;
0d28b7c10758 First commit
ymh
parents:
diff changeset
   200
    }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   201
0d28b7c10758 First commit
ymh
parents:
diff changeset
   202
    function _startBlock($header)
0d28b7c10758 First commit
ymh
parents:
diff changeset
   203
    {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   204
        return $header . "\n";
0d28b7c10758 First commit
ymh
parents:
diff changeset
   205
    }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   206
0d28b7c10758 First commit
ymh
parents:
diff changeset
   207
    function _endBlock()
0d28b7c10758 First commit
ymh
parents:
diff changeset
   208
    {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   209
        return '';
0d28b7c10758 First commit
ymh
parents:
diff changeset
   210
    }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   211
0d28b7c10758 First commit
ymh
parents:
diff changeset
   212
    function _lines($lines, $prefix = ' ')
0d28b7c10758 First commit
ymh
parents:
diff changeset
   213
    {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   214
        return $prefix . implode("\n$prefix", $lines) . "\n";
0d28b7c10758 First commit
ymh
parents:
diff changeset
   215
    }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   216
0d28b7c10758 First commit
ymh
parents:
diff changeset
   217
    function _context($lines)
0d28b7c10758 First commit
ymh
parents:
diff changeset
   218
    {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   219
        return $this->_lines($lines, '  ');
0d28b7c10758 First commit
ymh
parents:
diff changeset
   220
    }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   221
0d28b7c10758 First commit
ymh
parents:
diff changeset
   222
    function _added($lines)
0d28b7c10758 First commit
ymh
parents:
diff changeset
   223
    {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   224
        return $this->_lines($lines, '> ');
0d28b7c10758 First commit
ymh
parents:
diff changeset
   225
    }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   226
0d28b7c10758 First commit
ymh
parents:
diff changeset
   227
    function _deleted($lines)
0d28b7c10758 First commit
ymh
parents:
diff changeset
   228
    {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   229
        return $this->_lines($lines, '< ');
0d28b7c10758 First commit
ymh
parents:
diff changeset
   230
    }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   231
0d28b7c10758 First commit
ymh
parents:
diff changeset
   232
    function _changed($orig, $final)
0d28b7c10758 First commit
ymh
parents:
diff changeset
   233
    {
0d28b7c10758 First commit
ymh
parents:
diff changeset
   234
        return $this->_deleted($orig) . "---\n" . $this->_added($final);
0d28b7c10758 First commit
ymh
parents:
diff changeset
   235
    }
0d28b7c10758 First commit
ymh
parents:
diff changeset
   236
0d28b7c10758 First commit
ymh
parents:
diff changeset
   237
}