aboutsummaryrefslogtreecommitdiffstats
path: root/server/sonar-web/src/main/js/components/source-viewer/helpers/code-with-issue-locations-helper.js
blob: 0a1537002fde54ef5ffe639f287922ca6b56e245 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
import _ from 'underscore';

/**
 * Intersect two ranges
 * @param {number} s1 Start position of the first range
 * @param {number} e1 End position of the first range
 * @param {number} s2 Start position of the second range
 * @param {number} e2 End position of the second range
 * @returns {{from: number, to: number}}
 */
function intersect (s1, e1, s2, e2) {
  return { from: Math.max(s1, s2), to: Math.min(e1, e2) };
}


/**
 * Get the substring of a string
 * @param {string} str A string
 * @param {number} from "From" offset
 * @param {number} to "To" offset
 * @param {number} acc Global offset to eliminate
 * @returns {string}
 */
function part (str, from, to, acc) {
  // we do not want negative number as the first argument of `substr`
  return from >= acc ? str.substr(from - acc, to - from) : str.substr(0, to - from);
}


/**
 * Split a code html into tokens
 * @param {string} code
 * @returns {Array}
 */
function splitByTokens (code) {
  var container = document.createElement('div'),
      tokens = [];
  container.innerHTML = code;
  [].forEach.call(container.childNodes, function (node) {
    if (node.nodeType === 1) {
      // ELEMENT NODE
      tokens.push({ className: node.className, text: node.textContent });
    }
    if (node.nodeType === 3) {
      // TEXT NODE
      tokens.push({ className: '', text: node.nodeValue });
    }
  });
  return tokens;
}


/**
 * Highlight issue locations in the list of tokens
 * @param {Array} tokens
 * @param {Array} issueLocations
 * @param {string} className
 * @returns {Array}
 */
function highlightIssueLocations (tokens, issueLocations, className) {
  issueLocations.forEach(function (location) {
    var nextTokens = [],
        acc = 0;
    tokens.forEach(function (token) {
      var x = intersect(acc, acc + token.text.length, location.from, location.to);
      var p1 = part(token.text, acc, x.from, acc),
          p2 = part(token.text, x.from, x.to, acc),
          p3 = part(token.text, x.to, acc + token.text.length, acc);
      if (p1.length) {
        nextTokens.push({ className: token.className, text: p1 });
      }
      if (p2.length) {
        var newClassName = token.className.indexOf(className) === -1 ?
            [token.className, className].join(' ') : token.className;
        nextTokens.push({ className: newClassName, text: p2 });
      }
      if (p3.length) {
        nextTokens.push({ className: token.className, text: p3 });
      }
      acc += token.text.length;
    });
    tokens = nextTokens.slice();
  });
  return tokens;
}


/**
 * Generate an html string from the list of tokens
 * @param {Array} tokens
 * @returns {string}
 */
function generateHTML (tokens) {
  return tokens.map(function (token) {
    return '<span class="' + token.className + '">' + _.escape(token.text) + '</span>';
  }).join('');
}


/**
 * Take the initial source code, split by tokens,
 * highlight issues and generate result html
 * @param {string} code
 * @param {Array} issueLocations
 * @param {string} [optionalClassName]
 * @returns {string}
 */
function doTheStuff (code, issueLocations, optionalClassName) {
  var _code = code || '&nbsp;';
  var _issueLocations = issueLocations || [];
  var _className = optionalClassName ? optionalClassName : 'source-line-code-issue';
  return generateHTML(highlightIssueLocations(splitByTokens(_code), _issueLocations, _className));
}


if (typeof Handlebars !== 'undefined') {
  /**
   * Handlebars helper to highlight issue locations in the source code
   */
  Handlebars.registerHelper('codeWithIssueLocations', function (code, issueLocations) {
    return doTheStuff(code, issueLocations);
  });
}

export default doTheStuff;