BigW Consortium Gitlab

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

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

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

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

    function LineHighlighter(hash) {
      var range;
      if (hash == null) {
44 45 46
        // Initialize a LineHighlighter object
        //
        // hash - String URL hash for dependency injection in tests
Fatih Acet committed
47 48 49 50 51 52 53 54 55 56 57 58
        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], {
59 60
            // 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
61 62 63 64 65 66 67 68
            offset: -150
          });
        }
      }
    }

    LineHighlighter.prototype.bindEvents = function() {
      $('#blob-content-holder').on('mousedown', 'a[data-line-number]', this.clickHandler);
69 70 71 72 73 74
      // 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
75 76 77 78 79 80 81 82 83 84 85 86
      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)) {
87 88
        // If there's no current selection, or there is but Shift wasn't held,
        // treat this like a single-line selection.
Fatih Acet committed
89 90 91 92 93 94 95 96 97 98 99 100 101 102 103
        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);
104
    // Unhighlight previously highlighted lines
Fatih Acet committed
105 106
    };

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

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

138 139 140
    // Highlight all lines within a range
    //
    // range - Array containing the starting and ending line numbers
Fatih Acet committed
141 142 143 144 145 146 147 148 149 150 151 152 153
    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]);
      }
    };

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

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

    return LineHighlighter;

  })();

}).call(this);