BigW Consortium Gitlab

line_highlighter.js 6.07 KB
Newer Older
1
/* eslint-disable */
2 3 4 5
// LineHighlighter
//
// Handles single- and multi-line selection and highlight for blob views.
//
Fatih Acet committed
6 7
/*= require jquery.scrollTo */

8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31
//
// ### 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 committed
32 33 34 35
(function() {
  var bind = function(fn, me){ return function(){ return fn.apply(me, arguments); }; };

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

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

    function LineHighlighter(hash) {
      var range;
      if (hash == null) {
45 46 47
        // Initialize a LineHighlighter object
        //
        // hash - String URL hash for dependency injection in tests
Fatih Acet committed
48 49 50 51 52 53 54 55 56 57 58 59
        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], {
60 61
            // Scroll to the first highlighted line on initial load
            // Offset -50 for the sticky top bar, and another -100 for some context
Fatih Acet committed
62 63 64 65 66 67 68 69
            offset: -150
          });
        }
      }
    }

    LineHighlighter.prototype.bindEvents = function() {
      $('#blob-content-holder').on('mousedown', 'a[data-line-number]', this.clickHandler);
70 71 72 73 74 75
      // While it may seem odd to bind to the mousedown event and then throw away
      // the click event, there is a method to our madness.
      //
      // If not done this way, the line number anchor will sometimes keep its
      // active state even when the event is cancelled, resulting in an ugly border
      // around the link and/or a persisted underline text decoration.
Fatih Acet committed
76 77 78 79 80 81 82 83 84 85 86 87
      return $('#blob-content-holder').on('click', 'a[data-line-number]', function(event) {
        return event.preventDefault();
      });
    };

    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)) {
88 89
        // If there's no current selection, or there is but Shift wasn't held,
        // treat this like a single-line selection.
Fatih Acet committed
90 91 92 93 94 95 96 97 98 99 100 101 102 103 104
        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);
105
    // Unhighlight previously highlighted lines
Fatih Acet committed
106 107
    };

108 109 110 111 112 113 114 115 116 117 118
    // 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 committed
119 120
    LineHighlighter.prototype.hashToRange = function(hash) {
      var first, last, matches;
121
      //?L(\d+)(?:-(\d+))?$/)
Fatih Acet committed
122 123 124 125 126 127 128 129 130 131
      matches = hash.match(/^#?L(\d+)(?:-(\d+))?$/);
      if (matches && matches.length) {
        first = parseInt(matches[1]);
        last = matches[2] ? parseInt(matches[2]) : null;
        return [first, last];
      } else {
        return [null, null];
      }
    };

132 133 134
    // Highlight a single line
    //
    // lineNumber - Line number to highlight
Fatih Acet committed
135 136 137 138
    LineHighlighter.prototype.highlightLine = function(lineNumber) {
      return $("#LC" + lineNumber).addClass(this.highlightClass);
    };

139 140 141
    // Highlight all lines within a range
    //
    // range - Array containing the starting and ending line numbers
Fatih Acet committed
142 143 144 145 146 147 148 149 150 151 152 153 154
    LineHighlighter.prototype.highlightRange = function(range) {
      var i, lineNumber, ref, ref1, results;
      if (range[1]) {
        results = [];
        for (lineNumber = i = ref = range[0], ref1 = range[1]; ref <= ref1 ? i <= ref1 : i >= ref1; lineNumber = ref <= ref1 ? ++i : --i) {
          results.push(this.highlightLine(lineNumber));
        }
        return results;
      } else {
        return this.highlightLine(range[0]);
      }
    };

155
    // Set the URL hash string
Fatih Acet committed
156 157 158 159 160 161 162 163 164 165 166
    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);
    };

167 168 169
    // Make the actual hash change in the browser
    //
    // This method is stubbed in tests.
Fatih Acet committed
170 171 172 173
    LineHighlighter.prototype.__setLocationHash__ = function(value) {
      return history.pushState({
        turbolinks: false,
        url: value
174 175
      // We're using pushState instead of assigning location.hash directly to
      // prevent the page from scrolling on the hashchange event
Fatih Acet committed
176 177 178 179 180 181 182 183
      }, document.title, value);
    };

    return LineHighlighter;

  })();

}).call(this);