line_highlighter.js 5.84 KB
Newer Older
1
/* eslint-disable func-names, space-before-function-paren, no-var, prefer-rest-params, wrap-iife, no-use-before-define, no-underscore-dangle, no-param-reassign, prefer-template, quotes, comma-dangle, prefer-arrow-callback, consistent-return, one-var, one-var-declaration-per-line, no-else-return, max-len */
2

3 4 5 6
// LineHighlighter
//
// Handles single- and multi-line selection and highlight for blob views.
//
7
require('vendor/jquery.scrollTo');
Fatih Acet's avatar
Fatih Acet committed
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
//
// ### Example Markup
//
//   <div id="blob-content-holder">
//     <div class="file-content">
//       <div class="line-numbers">
//         <a href="#L1" id="L1" data-line-number="1">1</a>
//         <a href="#L2" id="L2" data-line-number="2">2</a>
//         <a href="#L3" id="L3" data-line-number="3">3</a>
//         <a href="#L4" id="L4" data-line-number="4">4</a>
//         <a href="#L5" id="L5" data-line-number="5">5</a>
//       </div>
//       <pre class="code highlight">
//         <code>
//           <span id="LC1" class="line">...</span>
//           <span id="LC2" class="line">...</span>
//           <span id="LC3" class="line">...</span>
//           <span id="LC4" class="line">...</span>
//           <span id="LC5" class="line">...</span>
//         </code>
//       </pre>
//     </div>
//   </div>
//
Fatih Acet's avatar
Fatih Acet committed
33
(function() {
34
  var bind = function(fn, me) { return function() { return fn.apply(me, arguments); }; };
Fatih Acet's avatar
Fatih Acet committed
35 36

  this.LineHighlighter = (function() {
37
    // CSS class applied to highlighted lines
Fatih Acet's avatar
Fatih Acet committed
38 39
    LineHighlighter.prototype.highlightClass = 'hll';

40
    // Internal copy of location.hash so we're not dependent on `location` in tests
Fatih Acet's avatar
Fatih Acet committed
41 42 43 44 45
    LineHighlighter.prototype._hash = '';

    function LineHighlighter(hash) {
      var range;
      if (hash == null) {
46 47 48
        // Initialize a LineHighlighter object
        //
        // hash - String URL hash for dependency injection in tests
Fatih Acet's avatar
Fatih Acet committed
49 50 51 52 53 54 55 56 57 58 59 60
        hash = location.hash;
      }
      this.setHash = bind(this.setHash, this);
      this.highlightLine = bind(this.highlightLine, this);
      this.clickHandler = bind(this.clickHandler, this);
      this._hash = hash;
      this.bindEvents();
      if (hash !== '') {
        range = this.hashToRange(hash);
        if (range[0]) {
          this.highlightRange(range);
          $.scrollTo("#L" + range[0], {
61 62
            // Scroll to the first highlighted line on initial load
            // Offset -50 for the sticky top bar, and another -100 for some context
Fatih Acet's avatar
Fatih Acet committed
63 64 65 66 67 68 69
            offset: -150
          });
        }
      }
    }

    LineHighlighter.prototype.bindEvents = function() {
70
      $('#blob-content-holder').on('click', 'a[data-line-number]', this.clickHandler);
Fatih Acet's avatar
Fatih Acet committed
71 72 73 74 75 76 77 78 79
    };

    LineHighlighter.prototype.clickHandler = function(event) {
      var current, lineNumber, range;
      event.preventDefault();
      this.clearHighlight();
      lineNumber = $(event.target).closest('a').data('line-number');
      current = this.hashToRange(this._hash);
      if (!(current[0] && event.shiftKey)) {
80 81
        // If there's no current selection, or there is but Shift wasn't held,
        // treat this like a single-line selection.
Fatih Acet's avatar
Fatih Acet committed
82 83 84 85 86 87 88 89 90 91 92 93 94 95 96
        this.setHash(lineNumber);
        return this.highlightLine(lineNumber);
      } else if (event.shiftKey) {
        if (lineNumber < current[0]) {
          range = [lineNumber, current[0]];
        } else {
          range = [current[0], lineNumber];
        }
        this.setHash(range[0], range[1]);
        return this.highlightRange(range);
      }
    };

    LineHighlighter.prototype.clearHighlight = function() {
      return $("." + this.highlightClass).removeClass(this.highlightClass);
97
    // Unhighlight previously highlighted lines
Fatih Acet's avatar
Fatih Acet committed
98 99
    };

100 101 102 103 104 105 106 107 108 109 110
    // Convert a URL hash String into line numbers
    //
    // hash - Hash String
    //
    // Examples:
    //
    //   hashToRange('#L5')    # => [5, null]
    //   hashToRange('#L5-15') # => [5, 15]
    //   hashToRange('#foo')   # => [null, null]
    //
    // Returns an Array
Fatih Acet's avatar
Fatih Acet committed
111 112
    LineHighlighter.prototype.hashToRange = function(hash) {
      var first, last, matches;
113
      // ?L(\d+)(?:-(\d+))?$/)
Fatih Acet's avatar
Fatih Acet committed
114 115
      matches = hash.match(/^#?L(\d+)(?:-(\d+))?$/);
      if (matches && matches.length) {
116 117
        first = parseInt(matches[1], 10);
        last = matches[2] ? parseInt(matches[2], 10) : null;
Fatih Acet's avatar
Fatih Acet committed
118 119 120 121 122 123
        return [first, last];
      } else {
        return [null, null];
      }
    };

124 125 126
    // Highlight a single line
    //
    // lineNumber - Line number to highlight
Fatih Acet's avatar
Fatih Acet committed
127 128 129 130
    LineHighlighter.prototype.highlightLine = function(lineNumber) {
      return $("#LC" + lineNumber).addClass(this.highlightClass);
    };

131 132 133
    // Highlight all lines within a range
    //
    // range - Array containing the starting and ending line numbers
Fatih Acet's avatar
Fatih Acet committed
134 135 136 137
    LineHighlighter.prototype.highlightRange = function(range) {
      var i, lineNumber, ref, ref1, results;
      if (range[1]) {
        results = [];
138
        for (lineNumber = i = ref = range[0], ref1 = range[1]; ref <= ref1 ? i <= ref1 : i >= ref1; lineNumber = ref <= ref1 ? (i += 1) : (i -= 1)) {
Fatih Acet's avatar
Fatih Acet committed
139 140 141 142 143 144 145 146
          results.push(this.highlightLine(lineNumber));
        }
        return results;
      } else {
        return this.highlightLine(range[0]);
      }
    };

147
    // Set the URL hash string
Fatih Acet's avatar
Fatih Acet committed
148 149 150 151 152 153 154 155 156 157 158
    LineHighlighter.prototype.setHash = function(firstLineNumber, lastLineNumber) {
      var hash;
      if (lastLineNumber) {
        hash = "#L" + firstLineNumber + "-" + lastLineNumber;
      } else {
        hash = "#L" + firstLineNumber;
      }
      this._hash = hash;
      return this.__setLocationHash__(hash);
    };

159 160 161
    // Make the actual hash change in the browser
    //
    // This method is stubbed in tests.
Fatih Acet's avatar
Fatih Acet committed
162 163 164
    LineHighlighter.prototype.__setLocationHash__ = function(value) {
      return history.pushState({
        url: value
165 166
      // We're using pushState instead of assigning location.hash directly to
      // prevent the page from scrolling on the hashchange event
Fatih Acet's avatar
Fatih Acet committed
167 168 169 170 171
      }, document.title, value);
    };

    return LineHighlighter;
  })();
172
}).call(window);