Skip to:
Content

BuddyPress.org

Ticket #8001: 8001.tribute01.diff

File 8001.tribute01.diff, 268.3 KB (added by dcavins, 2 years ago)

Use Tribute for @-lookups of users.

Line 
1diff --git .gitignore .gitignore
2index 769dc445e..c4f380218 100644
3--- .gitignore
4+++ .gitignore
5@@ -14,8 +14,8 @@ lib-cov
6 pids
7 logs
8 results
9-src/vendor
10-vendor
11+#src/vendor
12+#vendor
13 
14 node_modules
15 npm-debug.log
16diff --git src/bp-activity/bp-activity-cssjs.php src/bp-activity/bp-activity-cssjs.php
17index ec14cd416..945a0ad75 100644
18--- src/bp-activity/bp-activity-cssjs.php
19+++ src/bp-activity/bp-activity-cssjs.php
20@@ -33,7 +33,7 @@ function bp_activity_mentions_script() {
21 
22        $min = bp_core_get_minified_asset_suffix();
23 
24-       wp_enqueue_script( 'bp-mentions', buddypress()->plugin_url . "bp-activity/js/mentions{$min}.js", array( 'jquery', 'jquery-atwho' ), bp_get_version(), true );
25+       wp_enqueue_script( 'bp-mentions', buddypress()->plugin_url . "bp-activity/js/mentions{$min}.js", array( 'zurb-tribute' ), bp_get_version(), true );
26        wp_enqueue_style( 'bp-mentions-css', buddypress()->plugin_url . "bp-activity/css/mentions{$min}.css", array(), bp_get_version() );
27 
28        wp_style_add_data( 'bp-mentions-css', 'rtl', true );
29diff --git src/bp-activity/css/mentions.css src/bp-activity/css/mentions.css
30index 7773c75ec..f7bc47d12 100644
31--- src/bp-activity/css/mentions.css
32+++ src/bp-activity/css/mentions.css
33@@ -1,107 +1,41 @@
34-.atwho-view {
35-       background: rgba(204, 204, 204, 0.8);
36-       border-radius: 2px;
37-       border: 1px solid rgb(204, 204, 204);
38-       box-shadow: 0 0 5px rgba(204, 204, 204, 0.25), 0 0 1px #fff;
39-       color: #d84800;
40-       display: none;
41-       font-family: sans-serif;
42-       margin-top: 18px;
43+.tribute-container {
44        position: absolute;
45        top: 0;
46-       z-index: 1000; /* >999 for wp-admin */
47-}
48-
49-/* rtl:ignore */
50-.atwho-view {
51        left: 0;
52+       height: auto;
53+       max-height: 300px;
54+       max-width: 500px;
55+       overflow: auto;
56+       display: block;
57+       z-index: 999999;
58 }
59-
60-.atwho-view ul {
61-       background: #fff;
62-       list-style: none;
63-       margin: auto;
64+.tribute-container ul {
65+       margin: 0;
66+       margin-top: 2px;
67        padding: 0;
68+       list-style: none;
69+       background: #efefef;
70 }
71-
72-.atwho-view ul li {
73-       border-bottom: 1px solid #efefef;
74-       box-sizing: content-box;
75+.tribute-container li {
76+       padding: 5px 5px;
77        cursor: pointer;
78-       display: block;
79-       font-size: 14px;
80-       height: 20px;
81-       line-height: 20px;
82-       margin: 0;
83-       overflow: hidden;
84-       padding: 5px 10px;
85-}
86-
87-.atwho-view img {
88-       border-radius: 2px;
89-       float: right;
90-       height: 20px;
91-       margin-top: 0;
92-       width: 20px;
93 }
94-
95-.atwho-view strong {
96-       background: #efefef;
97-       font-weight: 700;
98+.tribute-container li.highlight, .tribute-container li:hover {
99+       background: #ddd;
100 }
101-
102-.atwho-view .username strong {
103-       color: #d54e21;
104+.tribute-container li span {
105+       font-weight: bold;
106 }
107-
108-.atwho-view small {
109-       color: #aaa;
110-       float: right;
111-       font-size: smaller;
112-       font-weight: 400;
113-       margin: 0 10px 0 40px;
114+.tribute-container li.no-match {
115+       cursor: default;
116 }
117-
118-.atwho-view .cur {
119-       background: rgba(239, 239, 239, 0.5);
120+.tribute-container .menu-highlighted {
121+       font-weight: bold;
122 }
123-
124-@media (max-width: 900px) {
125-
126-       .atwho-view img {
127-               float: left;
128-               margin: 0 10px 0 0;
129-       }
130-}
131-
132-@media (max-width: 400px) {
133-
134-       .atwho-view ul li {
135-               font-size: 16px;
136-               line-height: 23px;
137-               padding: 13px;
138-       }
139-
140-       .atwho-view ul li img {
141-               height: 30px;
142-               margin-top: -5px;
143-               width: 30px;
144-       }
145-
146-       .atwho-view {
147-               border-radius: 0;
148-               left: 0 !important;
149-               width: 100%;
150-       }
151-
152-       .atwho-view ul li .username {
153-               display: inline-block;
154-               margin: -10px 0 0 0;
155-               padding: 10px 0;
156-       }
157-
158-       .atwho-view ul li small {
159-               display: inline-block;
160-               margin-left: 20px;
161-       }
162+.tribute-container li img {
163+       border-radius: 2px;
164+       float: right;
165+       height: 20px;
166+       /*margin-top: 0;*/
167+       width: 20px;
168 }
169diff --git src/bp-activity/js/mentions.js src/bp-activity/js/mentions.js
170index 2d0afa3ef..882f3690d 100644
171--- src/bp-activity/js/mentions.js
172+++ src/bp-activity/js/mentions.js
173@@ -16,241 +16,107 @@ window.bp = window.bp || {};
174        /**
175         * Adds BuddyPress @mentions to form inputs.
176         *
177-        * @param {array|object} options If array, becomes the suggestions' data source. If object, passed as config to $.atwho().
178+        * @param {array} defaultList If array, becomes the suggestions' default data source.
179         * @since 2.1.0
180         */
181-       $.fn.bp_mentions = function( options ) {
182-               if ( $.isArray( options ) ) {
183-                       options = { data: options };
184-               }
185-
186-               /**
187-                * Default options for at.js; see https://github.com/ichord/At.js/.
188-                */
189-               var suggestionsDefaults = {
190-                       delay:             200,
191-                       hideWithoutSuffix: true,
192-                       insertTpl:         '@${ID}',
193-                       limit:             10,
194-                       startWithSpace:    false,
195-                       suffix:            '',
196-
197-                       callbacks: {
198-                               /**
199-                                * Custom filter to only match the start of spaced words.
200-                                * Based on the core/default one.
201-                                *
202-                                * @param {string} query
203-                                * @param {array} data
204-                                * @param {string} search_key
205-                                * @return {array}
206-                                * @since 2.1.0
207-                                */
208-                               filter: function( query, data, search_key ) {
209-                                       var item, _i, _len, _results = [],
210-                                       regxp = new RegExp( '^' + query + '| ' + query, 'ig' ); // start of string, or preceded by a space.
211-
212-                                       for ( _i = 0, _len = data.length; _i < _len; _i++ ) {
213-                                               item = data[ _i ];
214-                                               if ( item[ search_key ].toLowerCase().match( regxp ) ) {
215-                                                       _results.push( item );
216-                                               }
217-                                       }
218-
219-                                       return _results;
220-                               },
221-
222-                               /**
223-                                * Removes some spaces around highlighted string and tweaks regex to allow spaces
224-                                * (to match display_name). Based on the core default.
225-                                *
226-                                * @param {unknown} li
227-                                * @param {string} query
228-                                * @return {string}
229-                                * @since 2.1.0
230-                                */
231-                               highlighter: function( li, query ) {
232-                                       if ( ! query ) {
233-                                               return li;
234-                                       }
235-
236-                                       var regexp = new RegExp( '>(\\s*|[\\w\\s]*)(' + this.at.replace( '+', '\\+') + '?' + query.replace( '+', '\\+' ) + ')([\\w ]*)\\s*<', 'ig' );
237-                                       return li.replace( regexp, function( str, $1, $2, $3 ) {
238-                                               return '>' + $1 + '<strong>' + $2 + '</strong>' + $3 + '<';
239-                                       });
240-                               },
241-
242-                               /**
243-                                * Reposition the suggestion list dynamically.
244-                                *
245-                                * @param {unknown} offset
246-                                * @since 2.1.0
247-                                */
248-                               before_reposition: function( offset ) {
249-                                       // get the iframe, if any, already applied with atwho
250-                                       var caret,
251-                                                       line,
252-                                                       iframeOffset,
253-                                                       move,
254-                                                       $view = $( '#atwho-ground-' + this.id + ' .atwho-view' ),
255-                                                       $body = $( 'body' ),
256-                                                       atwhoDataValue = this.$inputor.data( 'atwho' );
257-
258-                                       if ( 'undefined' !== atwhoDataValue && 'undefined' !== atwhoDataValue.iframe && null !== atwhoDataValue.iframe ) {
259-                                               caret = this.$inputor.caret( 'offset', { iframe: atwhoDataValue.iframe } );
260-                                               // Caret.js no longer calculates iframe caret position from the window (it's now just within the iframe).
261-                                               // We need to get the iframe offset from the window and merge that into our object.
262-                                               iframeOffset = $( atwhoDataValue.iframe ).offset();
263-                                               if ( 'undefined' !== iframeOffset ) {
264-                                                       caret.left += iframeOffset.left;
265-                                                       caret.top  += iframeOffset.top;
266-                                               }
267-                                       } else {
268-                                               caret = this.$inputor.caret( 'offset' );
269-                                       }
270-
271-                                       // If the caret is past horizontal half, then flip it, yo
272-                                       if ( caret.left > ( $body.width() / 2 ) ) {
273-                                               $view.addClass( 'right' );
274-                                               move = caret.left - offset.left - this.view.$el.width();
275-                                       } else {
276-                                               $view.removeClass( 'right' );
277-                                               move = caret.left - offset.left + 1;
278-                                       }
279-
280-                                       // If we're on a small screen, scroll to caret
281-                                       if ( $body.width() <= 400 ) {
282-                                               $( document ).scrollTop( caret.top - 6 );
283-                                       }
284+       $.fn.bp_mentions = function( defaultList ) {
285+               var debouncer = function(func, wait) {
286+                 let timeout;
287+
288+                 return function() {
289+                       var context = this;
290+                   var args = arguments;
291+
292+                   var callFunction = function() {
293+                       func.apply(context, args)
294+                   };
295+
296+                   clearTimeout(timeout);
297+                   timeout = setTimeout(callFunction, wait);
298+                 }
299+               };
300+
301+               var remoteSearch = function(text, cb) {
302+                       /**
303+                        * Immediately show the pre-created friends list, if it's populated,
304+                        * and the user has hesitated after hitting @ (no search text provided).
305+                        */
306+                       if ( text.length === 0 && $.isArray( defaultList ) && defaultList.length > 0) {
307+                               cb(defaultList);
308+                               return;
309+                       }
310 
311-                                       // New position is under the caret (never above) and positioned to follow
312-                                       // Dynamic sizing based on the input area (remove 'px' from end)
313-                                       line = parseInt( this.$inputor.css( 'line-height' ).substr( 0, this.$inputor.css( 'line-height' ).length - 2 ), 10 );
314-                                       if ( !line || line < 5 ) { // sanity check, and catch no line-height
315-                                               line = 19;
316-                                       }
317+                       mentionsItem = mentionsQueryCache[ text ];
318+                       if ( typeof mentionsItem === 'object' ) {
319+                               cb(mentionsItem);
320+                               return;
321+                       }
322 
323-                                       offset.top   = caret.top + line;
324-                                       offset.left += move;
325-                               },
326+                       var params = { 'action': 'bp_get_suggestions', 'term': text, 'type': 'members' };
327 
328-                               /**
329-                                * Override default behaviour which inserts junk tags in the WordPress Visual editor.
330-                                *
331-                                * @param {unknown} $inputor Element which we're inserting content into.
332-                                * @param {string) content The content that will be inserted.
333-                                * @param {string) suffix Applied to the end of the content string.
334-                                * @return {string}
335-                                * @since 2.1.0
336-                                */
337-                               inserting_wrapper: function( $inputor, content, suffix ) {
338-                                       return '' + content + suffix;
339-                               }
340+                       // Add the group ID to the request if group ID data is attached to the input.
341+                       if ( ".wp-editor-area" === $( this ).selector
342+                               && typeof document.activeElement !== undefined
343+                               && typeof document.activeElement.dataset !== undefined
344+                               && document.activeElement.dataset.suggestionsGroupId !== undefined
345+                               && $.isNumeric( document.activeElement.dataset.suggestionsGroupId ) ) {
346+                               params['group-id'] = parseInt( document.activeElement.dataset.suggestionsGroupId, 10 );
347                        }
348-               },
349 
350-               /**
351-                * Default options for our @mentions; see https://github.com/ichord/At.js/.
352-                */
353-               mentionsDefaults = {
354-                       callbacks: {
355+                       return $.getJSON( ajaxurl, params )
356                                /**
357-                                * If there are no matches for the query in this.data, then query BuddyPress.
358+                                * Success callback for the @suggestions lookup.
359                                 *
360-                                * @param {string} query Partial @mention to search for.
361-                                * @param {function} render_view Render page callback function.
362+                                * @param {object} response Details of users matching the query.
363                                 * @since 2.1.0
364-                                * @since 3.0.0. Renamed from "remote_filter" for at.js v1.5.4 support.
365                                 */
366-                               remoteFilter: function( query, render_view ) {
367-                                       var self = $( this ),
368-                                               params = {};
369-
370-                                       mentionsItem = mentionsQueryCache[ query ];
371-                                       if ( typeof mentionsItem === 'object' ) {
372-                                               render_view( mentionsItem );
373+                               .done(function( response ) {
374+                                       if ( ! response.success ) {
375+                                               cb([]);
376                                                return;
377                                        }
378 
379-                                       if ( self.xhr ) {
380-                                               self.xhr.abort();
381-                                       }
382-
383-                                       params = { 'action': 'bp_get_suggestions', 'term': query, 'type': 'members' };
384-
385-                                       if ( $.isNumeric( this.$inputor.data( 'suggestions-group-id' ) ) ) {
386-                                               params['group-id'] = parseInt( this.$inputor.data( 'suggestions-group-id' ), 10 );
387-                                       }
388-
389-                                       self.xhr = $.getJSON( ajaxurl, params )
390+                                       var data = $.map( response.data,
391                                                /**
392-                                                * Success callback for the @suggestions lookup.
393+                                                * Create a composite index to determine ordering of results;
394+                                                * nicename matches will appear on top.
395                                                 *
396-                                                * @param {object} response Details of users matching the query.
397+                                                * @param {array} suggestion A suggestion's original data.
398+                                                * @return {array} A suggestion's new data.
399                                                 * @since 2.1.0
400                                                 */
401-                                               .done(function( response ) {
402-                                                       if ( ! response.success ) {
403-                                                               return;
404-                                                       }
405+                                               function( suggestion ) {
406+                                                       suggestion.search = suggestion.search || suggestion.ID + ' ' + suggestion.name;
407+                                                       return suggestion;
408+                                               }
409+                                       );
410 
411-                                                       var data = $.map( response.data,
412-                                                               /**
413-                                                                * Create a composite index to determine ordering of results;
414-                                                                * nicename matches will appear on top.
415-                                                                *
416-                                                                * @param {array} suggestion A suggestion's original data.
417-                                                                * @return {array} A suggestion's new data.
418-                                                                * @since 2.1.0
419-                                                                */
420-                                                               function( suggestion ) {
421-                                                                       suggestion.search = suggestion.search || suggestion.ID + ' ' + suggestion.name;
422-                                                                       return suggestion;
423-                                                               }
424-                                                       );
425+                                       mentionsQueryCache[ text ] = data;
426+                                       cb(data);
427+                               });
428+               }
429 
430-                                                       mentionsQueryCache[ query ] = data;
431-                                                       render_view( data );
432-                                               });
433-                               }
434+               var tributeParams = {
435+                       values: debouncer( function (text, cb) {
436+                               remoteSearch(text, users => cb(users));
437+                       }, 250),
438+                       lookup: 'search',
439+                       fillAttr: 'ID',
440+                       menuItemTemplate: function (item) {
441+                               return '<img src="' + item.original.image + '" alt="Profile picture of ' + item.original.name + '"> @' + item.string;
442                        },
443+               };
444 
445-                       data: $.map( options.data,
446-                               /**
447-                                * Create a composite index to search against of nicename + display name.
448-                                * This will also determine ordering of results, so nicename matches will appear on top.
449-                                *
450-                                * @param {array} suggestion A suggestion's original data.
451-                                * @return {array} A suggestion's new data.
452-                                * @since 2.1.0
453-                                */
454-                               function( suggestion ) {
455-                                       suggestion.search = suggestion.search || suggestion.ID + ' ' + suggestion.name;
456-                                       return suggestion;
457-                               }
458-                       ),
459+               var tribute = new Tribute( tributeParams );
460 
461-                       at:         '@',
462-                       searchKey:  'search',
463-                       displayTpl: '<li data-value="@${ID}"><img src="${image}" /><span class="username">@${ID}</span><small>${name}</small></li>'
464-               },
465-
466-               opts = $.extend( true, {}, suggestionsDefaults, mentionsDefaults, options );
467-               return $.fn.atwho.call( this, opts );
468+               $( this ).each( function() {
469+                       tribute.attach( document.getElementById( $(this).attr("id") ) );
470+               });
471        };
472 
473        $( document ).ready(function() {
474                // Activity/reply, post comments, dashboard post 'text' editor.
475-               $( '.bp-suggestions, #comments form textarea, .wp-editor-area' ).bp_mentions( bp.mentions.users );
476+               $( '.bp-suggestions, #comments form textarea' ).bp_mentions( bp.mentions.users );
477        });
478 
479-       bp.mentions.tinyMCEinit = function() {
480-               if ( typeof window.tinyMCE === 'undefined' || window.tinyMCE.activeEditor === null || typeof window.tinyMCE.activeEditor === 'undefined' ) {
481-                       return;
482-               } else {
483-                       $( window.tinyMCE.activeEditor.contentDocument.activeElement )
484-                               .atwho( 'setIframe', $( '.wp-editor-wrap iframe' )[0] )
485-                               .bp_mentions( bp.mentions.users );
486-               }
487-       };
488 })( bp, jQuery );
489diff --git src/bp-core/bp-core-cssjs.php src/bp-core/bp-core-cssjs.php
490index 987deb642..4d9eff26a 100644
491--- src/bp-core/bp-core-cssjs.php
492+++ src/bp-core/bp-core-cssjs.php
493@@ -54,10 +54,6 @@ function bp_core_register_common_scripts() {
494                'bp-jquery-cookie'  => array( 'file' => "{$url}vendor/jquery-cookie{$min}.js", 'dependencies' => array( 'jquery' ), 'footer' => false ),
495                'bp-jquery-scroll-to' => array( 'file' => "{$url}vendor/jquery-scroll-to{$min}.js", 'dependencies' => array( 'jquery' ), 'footer' => false ),
496 
497-               // Version 2.1.
498-               'jquery-caret' => array( 'file' => "{$url}vendor/jquery.caret{$min}.js", 'dependencies' => array( 'jquery' ), 'footer' => true ),
499-               'jquery-atwho' => array( 'file' => "{$url}vendor/jquery.atwho{$min}.js", 'dependencies' => array( 'jquery', 'jquery-caret' ), 'footer' => true ),
500-
501                // Version 2.3.
502                'bp-plupload' => array( 'file' => "{$url}bp-plupload{$min}.js", 'dependencies' => array( 'plupload', 'jquery', 'json2', 'wp-backbone' ), 'footer' => true ),
503                'bp-avatar'   => array( 'file' => "{$url}avatar{$min}.js", 'dependencies' => array( 'jcrop' ), 'footer' => true ),
504@@ -69,6 +65,9 @@ function bp_core_register_common_scripts() {
505                // Version 2.7.
506                'bp-moment'    => array( 'file' => "{$url}vendor/moment-js/moment{$min}.js", 'dependencies' => array(), 'footer' => true ),
507                'bp-livestamp' => array( 'file' => "{$url}vendor/livestamp{$min}.js", 'dependencies' => array( 'jquery', 'bp-moment' ), 'footer' => true ),
508+
509+               // Version 5
510+               'zurb-tribute' => array( 'file' => "{$url}vendor/tribute{$min}.js", 'dependencies' => array(), 'footer' => true ),
511        );
512 
513        // Version 2.7 - Add Moment.js locale to our $scripts array if we found one.
514diff --git src/bp-core/js/vendor/jquery.atwho.js src/bp-core/js/vendor/jquery.atwho.js
515deleted file mode 100755
516index 795b6c67d..000000000
517--- src/bp-core/js/vendor/jquery.atwho.js
518+++ /dev/null
519@@ -1,1212 +0,0 @@
520-/**
521- * at.js - 1.5.4
522- * Copyright (c) 2017 chord.luo <chord.luo@gmail.com>;
523- * Homepage: http://ichord.github.com/At.js
524- * License: MIT
525- */
526-(function (root, factory) {
527-  if (typeof define === 'function' && define.amd) {
528-    // AMD. Register as an anonymous module unless amdModuleId is set
529-    define(["jquery"], function (a0) {
530-      return (factory(a0));
531-    });
532-  } else if (typeof exports === 'object') {
533-    // Node. Does not work with strict CommonJS, but
534-    // only CommonJS-like environments that support module.exports,
535-    // like Node.
536-    module.exports = factory(require("jquery"));
537-  } else {
538-    factory(jQuery);
539-  }
540-}(this, function ($) {
541-var DEFAULT_CALLBACKS, KEY_CODE;
542-
543-KEY_CODE = {
544-  ESC: 27,
545-  TAB: 9,
546-  ENTER: 13,
547-  CTRL: 17,
548-  A: 65,
549-  P: 80,
550-  N: 78,
551-  LEFT: 37,
552-  UP: 38,
553-  RIGHT: 39,
554-  DOWN: 40,
555-  BACKSPACE: 8,
556-  SPACE: 32
557-};
558-
559-DEFAULT_CALLBACKS = {
560-  beforeSave: function(data) {
561-    return Controller.arrayToDefaultHash(data);
562-  },
563-  matcher: function(flag, subtext, should_startWithSpace, acceptSpaceBar) {
564-    var _a, _y, match, regexp, space;
565-    flag = flag.replace(/[\-\[\]\/\{\}\(\)\*\+\?\.\\\^\$\|]/g, "\\$&");
566-    if (should_startWithSpace) {
567-      flag = '(?:^|\\s)' + flag;
568-    }
569-    _a = decodeURI("%C3%80");
570-    _y = decodeURI("%C3%BF");
571-    space = acceptSpaceBar ? "\ " : "";
572-    regexp = new RegExp(flag + "([A-Za-z" + _a + "-" + _y + "0-9_" + space + "\'\.\+\-]*)$|" + flag + "([^\\x00-\\xff]*)$", 'gi');
573-    match = regexp.exec(subtext);
574-    if (match) {
575-      return match[2] || match[1];
576-    } else {
577-      return null;
578-    }
579-  },
580-  filter: function(query, data, searchKey) {
581-    var _results, i, item, len;
582-    _results = [];
583-    for (i = 0, len = data.length; i < len; i++) {
584-      item = data[i];
585-      if (~new String(item[searchKey]).toLowerCase().indexOf(query.toLowerCase())) {
586-        _results.push(item);
587-      }
588-    }
589-    return _results;
590-  },
591-  remoteFilter: null,
592-  sorter: function(query, items, searchKey) {
593-    var _results, i, item, len;
594-    if (!query) {
595-      return items;
596-    }
597-    _results = [];
598-    for (i = 0, len = items.length; i < len; i++) {
599-      item = items[i];
600-      item.atwho_order = new String(item[searchKey]).toLowerCase().indexOf(query.toLowerCase());
601-      if (item.atwho_order > -1) {
602-        _results.push(item);
603-      }
604-    }
605-    return _results.sort(function(a, b) {
606-      return a.atwho_order - b.atwho_order;
607-    });
608-  },
609-  tplEval: function(tpl, map) {
610-    var error, error1, template;
611-    template = tpl;
612-    try {
613-      if (typeof tpl !== 'string') {
614-        template = tpl(map);
615-      }
616-      return template.replace(/\$\{([^\}]*)\}/g, function(tag, key, pos) {
617-        return map[key];
618-      });
619-    } catch (error1) {
620-      error = error1;
621-      return "";
622-    }
623-  },
624-  highlighter: function(li, query) {
625-    var regexp;
626-    if (!query) {
627-      return li;
628-    }
629-    regexp = new RegExp(">\\s*([^\<]*?)(" + query.replace("+", "\\+") + ")([^\<]*)\\s*<", 'ig');
630-    return li.replace(regexp, function(str, $1, $2, $3) {
631-      return '> ' + $1 + '<strong>' + $2 + '</strong>' + $3 + ' <';
632-    });
633-  },
634-  beforeInsert: function(value, $li, e) {
635-    return value;
636-  },
637-  beforeReposition: function(offset) {
638-    return offset;
639-  },
640-  afterMatchFailed: function(at, el) {}
641-};
642-
643-var App;
644-
645-App = (function() {
646-  function App(inputor) {
647-    this.currentFlag = null;
648-    this.controllers = {};
649-    this.aliasMaps = {};
650-    this.$inputor = $(inputor);
651-    this.setupRootElement();
652-    this.listen();
653-  }
654-
655-  App.prototype.createContainer = function(doc) {
656-    var ref;
657-    if ((ref = this.$el) != null) {
658-      ref.remove();
659-    }
660-    return $(doc.body).append(this.$el = $("<div class='atwho-container'></div>"));
661-  };
662-
663-  App.prototype.setupRootElement = function(iframe, asRoot) {
664-    var error, error1;
665-    if (asRoot == null) {
666-      asRoot = false;
667-    }
668-    if (iframe) {
669-      this.window = iframe.contentWindow;
670-      this.document = iframe.contentDocument || this.window.document;
671-      this.iframe = iframe;
672-    } else {
673-      this.document = this.$inputor[0].ownerDocument;
674-      this.window = this.document.defaultView || this.document.parentWindow;
675-      try {
676-        this.iframe = this.window.frameElement;
677-      } catch (error1) {
678-        error = error1;
679-        this.iframe = null;
680-        if ($.fn.atwho.debug) {
681-          throw new Error("iframe auto-discovery is failed.\nPlease use `setIframe` to set the target iframe manually.\n" + error);
682-        }
683-      }
684-    }
685-    return this.createContainer((this.iframeAsRoot = asRoot) ? this.document : document);
686-  };
687-
688-  App.prototype.controller = function(at) {
689-    var c, current, currentFlag, ref;
690-    if (this.aliasMaps[at]) {
691-      current = this.controllers[this.aliasMaps[at]];
692-    } else {
693-      ref = this.controllers;
694-      for (currentFlag in ref) {
695-        c = ref[currentFlag];
696-        if (currentFlag === at) {
697-          current = c;
698-          break;
699-        }
700-      }
701-    }
702-    if (current) {
703-      return current;
704-    } else {
705-      return this.controllers[this.currentFlag];
706-    }
707-  };
708-
709-  App.prototype.setContextFor = function(at) {
710-    this.currentFlag = at;
711-    return this;
712-  };
713-
714-  App.prototype.reg = function(flag, setting) {
715-    var base, controller;
716-    controller = (base = this.controllers)[flag] || (base[flag] = this.$inputor.is('[contentEditable]') ? new EditableController(this, flag) : new TextareaController(this, flag));
717-    if (setting.alias) {
718-      this.aliasMaps[setting.alias] = flag;
719-    }
720-    controller.init(setting);
721-    return this;
722-  };
723-
724-  App.prototype.listen = function() {
725-    return this.$inputor.on('compositionstart', (function(_this) {
726-      return function(e) {
727-        var ref;
728-        if ((ref = _this.controller()) != null) {
729-          ref.view.hide();
730-        }
731-        _this.isComposing = true;
732-        return null;
733-      };
734-    })(this)).on('compositionend', (function(_this) {
735-      return function(e) {
736-        _this.isComposing = false;
737-        setTimeout(function(e) {
738-          return _this.dispatch(e);
739-        });
740-        return null;
741-      };
742-    })(this)).on('keyup.atwhoInner', (function(_this) {
743-      return function(e) {
744-        return _this.onKeyup(e);
745-      };
746-    })(this)).on('keydown.atwhoInner', (function(_this) {
747-      return function(e) {
748-        return _this.onKeydown(e);
749-      };
750-    })(this)).on('blur.atwhoInner', (function(_this) {
751-      return function(e) {
752-        var c;
753-        if (c = _this.controller()) {
754-          c.expectedQueryCBId = null;
755-          return c.view.hide(e, c.getOpt("displayTimeout"));
756-        }
757-      };
758-    })(this)).on('click.atwhoInner', (function(_this) {
759-      return function(e) {
760-        return _this.dispatch(e);
761-      };
762-    })(this)).on('scroll.atwhoInner', (function(_this) {
763-      return function() {
764-        var lastScrollTop;
765-        lastScrollTop = _this.$inputor.scrollTop();
766-        return function(e) {
767-          var currentScrollTop, ref;
768-          currentScrollTop = e.target.scrollTop;
769-          if (lastScrollTop !== currentScrollTop) {
770-            if ((ref = _this.controller()) != null) {
771-              ref.view.hide(e);
772-            }
773-          }
774-          lastScrollTop = currentScrollTop;
775-          return true;
776-        };
777-      };
778-    })(this)());
779-  };
780-
781-  App.prototype.shutdown = function() {
782-    var _, c, ref;
783-    ref = this.controllers;
784-    for (_ in ref) {
785-      c = ref[_];
786-      c.destroy();
787-      delete this.controllers[_];
788-    }
789-    this.$inputor.off('.atwhoInner');
790-    return this.$el.remove();
791-  };
792-
793-  App.prototype.dispatch = function(e) {
794-    var _, c, ref, results;
795-    ref = this.controllers;
796-    results = [];
797-    for (_ in ref) {
798-      c = ref[_];
799-      results.push(c.lookUp(e));
800-    }
801-    return results;
802-  };
803-
804-  App.prototype.onKeyup = function(e) {
805-    var ref;
806-    switch (e.keyCode) {
807-      case KEY_CODE.ESC:
808-        e.preventDefault();
809-        if ((ref = this.controller()) != null) {
810-          ref.view.hide();
811-        }
812-        break;
813-      case KEY_CODE.DOWN:
814-      case KEY_CODE.UP:
815-      case KEY_CODE.CTRL:
816-      case KEY_CODE.ENTER:
817-        $.noop();
818-        break;
819-      case KEY_CODE.P:
820-      case KEY_CODE.N:
821-        if (!e.ctrlKey) {
822-          this.dispatch(e);
823-        }
824-        break;
825-      default:
826-        this.dispatch(e);
827-    }
828-  };
829-
830-  App.prototype.onKeydown = function(e) {
831-    var ref, view;
832-    view = (ref = this.controller()) != null ? ref.view : void 0;
833-    if (!(view && view.visible())) {
834-      return;
835-    }
836-    switch (e.keyCode) {
837-      case KEY_CODE.ESC:
838-        e.preventDefault();
839-        view.hide(e);
840-        break;
841-      case KEY_CODE.UP:
842-        e.preventDefault();
843-        view.prev();
844-        break;
845-      case KEY_CODE.DOWN:
846-        e.preventDefault();
847-        view.next();
848-        break;
849-      case KEY_CODE.P:
850-        if (!e.ctrlKey) {
851-          return;
852-        }
853-        e.preventDefault();
854-        view.prev();
855-        break;
856-      case KEY_CODE.N:
857-        if (!e.ctrlKey) {
858-          return;
859-        }
860-        e.preventDefault();
861-        view.next();
862-        break;
863-      case KEY_CODE.TAB:
864-      case KEY_CODE.ENTER:
865-      case KEY_CODE.SPACE:
866-        if (!view.visible()) {
867-          return;
868-        }
869-        if (!this.controller().getOpt('spaceSelectsMatch') && e.keyCode === KEY_CODE.SPACE) {
870-          return;
871-        }
872-        if (!this.controller().getOpt('tabSelectsMatch') && e.keyCode === KEY_CODE.TAB) {
873-          return;
874-        }
875-        if (view.highlighted()) {
876-          e.preventDefault();
877-          view.choose(e);
878-        } else {
879-          view.hide(e);
880-        }
881-        break;
882-      default:
883-        $.noop();
884-    }
885-  };
886-
887-  return App;
888-
889-})();
890-
891-var Controller,
892-  slice = [].slice;
893-
894-Controller = (function() {
895-  Controller.prototype.uid = function() {
896-    return (Math.random().toString(16) + "000000000").substr(2, 8) + (new Date().getTime());
897-  };
898-
899-  function Controller(app, at1) {
900-    this.app = app;
901-    this.at = at1;
902-    this.$inputor = this.app.$inputor;
903-    this.id = this.$inputor[0].id || this.uid();
904-    this.expectedQueryCBId = null;
905-    this.setting = null;
906-    this.query = null;
907-    this.pos = 0;
908-    this.range = null;
909-    if ((this.$el = $("#atwho-ground-" + this.id, this.app.$el)).length === 0) {
910-      this.app.$el.append(this.$el = $("<div id='atwho-ground-" + this.id + "'></div>"));
911-    }
912-    this.model = new Model(this);
913-    this.view = new View(this);
914-  }
915-
916-  Controller.prototype.init = function(setting) {
917-    this.setting = $.extend({}, this.setting || $.fn.atwho["default"], setting);
918-    this.view.init();
919-    return this.model.reload(this.setting.data);
920-  };
921-
922-  Controller.prototype.destroy = function() {
923-    this.trigger('beforeDestroy');
924-    this.model.destroy();
925-    this.view.destroy();
926-    return this.$el.remove();
927-  };
928-
929-  Controller.prototype.callDefault = function() {
930-    var args, error, error1, funcName;
931-    funcName = arguments[0], args = 2 <= arguments.length ? slice.call(arguments, 1) : [];
932-    try {
933-      return DEFAULT_CALLBACKS[funcName].apply(this, args);
934-    } catch (error1) {
935-      error = error1;
936-      return $.error(error + " Or maybe At.js doesn't have function " + funcName);
937-    }
938-  };
939-
940-  Controller.prototype.trigger = function(name, data) {
941-    var alias, eventName;
942-    if (data == null) {
943-      data = [];
944-    }
945-    data.push(this);
946-    alias = this.getOpt('alias');
947-    eventName = alias ? name + "-" + alias + ".atwho" : name + ".atwho";
948-    return this.$inputor.trigger(eventName, data);
949-  };
950-
951-  Controller.prototype.callbacks = function(funcName) {
952-    return this.getOpt("callbacks")[funcName] || DEFAULT_CALLBACKS[funcName];
953-  };
954-
955-  Controller.prototype.getOpt = function(at, default_value) {
956-    var e, error1;
957-    try {
958-      return this.setting[at];
959-    } catch (error1) {
960-      e = error1;
961-      return null;
962-    }
963-  };
964-
965-  Controller.prototype.insertContentFor = function($li) {
966-    var data, tpl;
967-    tpl = this.getOpt('insertTpl');
968-    data = $.extend({}, $li.data('item-data'), {
969-      'atwho-at': this.at
970-    });
971-    return this.callbacks("tplEval").call(this, tpl, data, "onInsert");
972-  };
973-
974-  Controller.prototype.renderView = function(data) {
975-    var searchKey;
976-    searchKey = this.getOpt("searchKey");
977-    data = this.callbacks("sorter").call(this, this.query.text, data.slice(0, 1001), searchKey);
978-    return this.view.render(data.slice(0, this.getOpt('limit')));
979-  };
980-
981-  Controller.arrayToDefaultHash = function(data) {
982-    var i, item, len, results;
983-    if (!$.isArray(data)) {
984-      return data;
985-    }
986-    results = [];
987-    for (i = 0, len = data.length; i < len; i++) {
988-      item = data[i];
989-      if ($.isPlainObject(item)) {
990-        results.push(item);
991-      } else {
992-        results.push({
993-          name: item
994-        });
995-      }
996-    }
997-    return results;
998-  };
999-
1000-  Controller.prototype.lookUp = function(e) {
1001-    var query, wait;
1002-    if (e && e.type === 'click' && !this.getOpt('lookUpOnClick')) {
1003-      return;
1004-    }
1005-    if (this.getOpt('suspendOnComposing') && this.app.isComposing) {
1006-      return;
1007-    }
1008-    query = this.catchQuery(e);
1009-    if (!query) {
1010-      this.expectedQueryCBId = null;
1011-      return query;
1012-    }
1013-    this.app.setContextFor(this.at);
1014-    if (wait = this.getOpt('delay')) {
1015-      this._delayLookUp(query, wait);
1016-    } else {
1017-      this._lookUp(query);
1018-    }
1019-    return query;
1020-  };
1021-
1022-  Controller.prototype._delayLookUp = function(query, wait) {
1023-    var now, remaining;
1024-    now = Date.now ? Date.now() : new Date().getTime();
1025-    this.previousCallTime || (this.previousCallTime = now);
1026-    remaining = wait - (now - this.previousCallTime);
1027-    if ((0 < remaining && remaining < wait)) {
1028-      this.previousCallTime = now;
1029-      this._stopDelayedCall();
1030-      return this.delayedCallTimeout = setTimeout((function(_this) {
1031-        return function() {
1032-          _this.previousCallTime = 0;
1033-          _this.delayedCallTimeout = null;
1034-          return _this._lookUp(query);
1035-        };
1036-      })(this), wait);
1037-    } else {
1038-      this._stopDelayedCall();
1039-      if (this.previousCallTime !== now) {
1040-        this.previousCallTime = 0;
1041-      }
1042-      return this._lookUp(query);
1043-    }
1044-  };
1045-
1046-  Controller.prototype._stopDelayedCall = function() {
1047-    if (this.delayedCallTimeout) {
1048-      clearTimeout(this.delayedCallTimeout);
1049-      return this.delayedCallTimeout = null;
1050-    }
1051-  };
1052-
1053-  Controller.prototype._generateQueryCBId = function() {
1054-    return {};
1055-  };
1056-
1057-  Controller.prototype._lookUp = function(query) {
1058-    var _callback;
1059-    _callback = function(queryCBId, data) {
1060-      if (queryCBId !== this.expectedQueryCBId) {
1061-        return;
1062-      }
1063-      if (data && data.length > 0) {
1064-        return this.renderView(this.constructor.arrayToDefaultHash(data));
1065-      } else {
1066-        return this.view.hide();
1067-      }
1068-    };
1069-    this.expectedQueryCBId = this._generateQueryCBId();
1070-    return this.model.query(query.text, $.proxy(_callback, this, this.expectedQueryCBId));
1071-  };
1072-
1073-  return Controller;
1074-
1075-})();
1076-
1077-var TextareaController,
1078-  extend = function(child, parent) { for (var key in parent) { if (hasProp.call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; },
1079-  hasProp = {}.hasOwnProperty;
1080-
1081-TextareaController = (function(superClass) {
1082-  extend(TextareaController, superClass);
1083-
1084-  function TextareaController() {
1085-    return TextareaController.__super__.constructor.apply(this, arguments);
1086-  }
1087-
1088-  TextareaController.prototype.catchQuery = function() {
1089-    var caretPos, content, end, isString, query, start, subtext;
1090-    content = this.$inputor.val();
1091-    caretPos = this.$inputor.caret('pos', {
1092-      iframe: this.app.iframe
1093-    });
1094-    subtext = content.slice(0, caretPos);
1095-    query = this.callbacks("matcher").call(this, this.at, subtext, this.getOpt('startWithSpace'), this.getOpt("acceptSpaceBar"));
1096-    isString = typeof query === 'string';
1097-    if (isString && query.length < this.getOpt('minLen', 0)) {
1098-      return;
1099-    }
1100-    if (isString && query.length <= this.getOpt('maxLen', 20)) {
1101-      start = caretPos - query.length;
1102-      end = start + query.length;
1103-      this.pos = start;
1104-      query = {
1105-        'text': query,
1106-        'headPos': start,
1107-        'endPos': end
1108-      };
1109-      this.trigger("matched", [this.at, query.text]);
1110-    } else {
1111-      query = null;
1112-      this.view.hide();
1113-    }
1114-    return this.query = query;
1115-  };
1116-
1117-  TextareaController.prototype.rect = function() {
1118-    var c, iframeOffset, scaleBottom;
1119-    if (!(c = this.$inputor.caret('offset', this.pos - 1, {
1120-      iframe: this.app.iframe
1121-    }))) {
1122-      return;
1123-    }
1124-    if (this.app.iframe && !this.app.iframeAsRoot) {
1125-      iframeOffset = $(this.app.iframe).offset();
1126-      c.left += iframeOffset.left;
1127-      c.top += iframeOffset.top;
1128-    }
1129-    scaleBottom = this.app.document.selection ? 0 : 2;
1130-    return {
1131-      left: c.left,
1132-      top: c.top,
1133-      bottom: c.top + c.height + scaleBottom
1134-    };
1135-  };
1136-
1137-  TextareaController.prototype.insert = function(content, $li) {
1138-    var $inputor, source, startStr, suffix, text;
1139-    $inputor = this.$inputor;
1140-    source = $inputor.val();
1141-    startStr = source.slice(0, Math.max(this.query.headPos - this.at.length, 0));
1142-    suffix = (suffix = this.getOpt('suffix')) === "" ? suffix : suffix || " ";
1143-    content += suffix;
1144-    text = "" + startStr + content + (source.slice(this.query['endPos'] || 0));
1145-    $inputor.val(text);
1146-    $inputor.caret('pos', startStr.length + content.length, {
1147-      iframe: this.app.iframe
1148-    });
1149-    if (!$inputor.is(':focus')) {
1150-      $inputor.focus();
1151-    }
1152-    return $inputor.change();
1153-  };
1154-
1155-  return TextareaController;
1156-
1157-})(Controller);
1158-
1159-var EditableController,
1160-  extend = function(child, parent) { for (var key in parent) { if (hasProp.call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; },
1161-  hasProp = {}.hasOwnProperty;
1162-
1163-EditableController = (function(superClass) {
1164-  extend(EditableController, superClass);
1165-
1166-  function EditableController() {
1167-    return EditableController.__super__.constructor.apply(this, arguments);
1168-  }
1169-
1170-  EditableController.prototype._getRange = function() {
1171-    var sel;
1172-    sel = this.app.window.getSelection();
1173-    if (sel.rangeCount > 0) {
1174-      return sel.getRangeAt(0);
1175-    }
1176-  };
1177-
1178-  EditableController.prototype._setRange = function(position, node, range) {
1179-    if (range == null) {
1180-      range = this._getRange();
1181-    }
1182-    if (!(range && node)) {
1183-      return;
1184-    }
1185-    node = $(node)[0];
1186-    if (position === 'after') {
1187-      range.setEndAfter(node);
1188-      range.setStartAfter(node);
1189-    } else {
1190-      range.setEndBefore(node);
1191-      range.setStartBefore(node);
1192-    }
1193-    range.collapse(false);
1194-    return this._clearRange(range);
1195-  };
1196-
1197-  EditableController.prototype._clearRange = function(range) {
1198-    var sel;
1199-    if (range == null) {
1200-      range = this._getRange();
1201-    }
1202-    sel = this.app.window.getSelection();
1203-    if (this.ctrl_a_pressed == null) {
1204-      sel.removeAllRanges();
1205-      return sel.addRange(range);
1206-    }
1207-  };
1208-
1209-  EditableController.prototype._movingEvent = function(e) {
1210-    var ref;
1211-    return e.type === 'click' || ((ref = e.which) === KEY_CODE.RIGHT || ref === KEY_CODE.LEFT || ref === KEY_CODE.UP || ref === KEY_CODE.DOWN);
1212-  };
1213-
1214-  EditableController.prototype._unwrap = function(node) {
1215-    var next;
1216-    node = $(node).unwrap().get(0);
1217-    if ((next = node.nextSibling) && next.nodeValue) {
1218-      node.nodeValue += next.nodeValue;
1219-      $(next).remove();
1220-    }
1221-    return node;
1222-  };
1223-
1224-  EditableController.prototype.catchQuery = function(e) {
1225-    var $inserted, $query, _range, index, inserted, isString, lastNode, matched, offset, query, query_content, range;
1226-    if (!(range = this._getRange())) {
1227-      return;
1228-    }
1229-    if (!range.collapsed) {
1230-      return;
1231-    }
1232-    if (e.which === KEY_CODE.ENTER) {
1233-      ($query = $(range.startContainer).closest('.atwho-query')).contents().unwrap();
1234-      if ($query.is(':empty')) {
1235-        $query.remove();
1236-      }
1237-      ($query = $(".atwho-query", this.app.document)).text($query.text()).contents().last().unwrap();
1238-      this._clearRange();
1239-      return;
1240-    }
1241-    if (/firefox/i.test(navigator.userAgent)) {
1242-      if ($(range.startContainer).is(this.$inputor)) {
1243-        this._clearRange();
1244-        return;
1245-      }
1246-      if (e.which === KEY_CODE.BACKSPACE && range.startContainer.nodeType === document.ELEMENT_NODE && (offset = range.startOffset - 1) >= 0) {
1247-        _range = range.cloneRange();
1248-        _range.setStart(range.startContainer, offset);
1249-        if ($(_range.cloneContents()).contents().last().is('.atwho-inserted')) {
1250-          inserted = $(range.startContainer).contents().get(offset);
1251-          this._setRange('after', $(inserted).contents().last());
1252-        }
1253-      } else if (e.which === KEY_CODE.LEFT && range.startContainer.nodeType === document.TEXT_NODE) {
1254-        $inserted = $(range.startContainer.previousSibling);
1255-        if ($inserted.is('.atwho-inserted') && range.startOffset === 0) {
1256-          this._setRange('after', $inserted.contents().last());
1257-        }
1258-      }
1259-    }
1260-    $(range.startContainer).closest('.atwho-inserted').addClass('atwho-query').siblings().removeClass('atwho-query');
1261-    if (($query = $(".atwho-query", this.app.document)).length > 0 && $query.is(':empty') && $query.text().length === 0) {
1262-      $query.remove();
1263-    }
1264-    if (!this._movingEvent(e)) {
1265-      $query.removeClass('atwho-inserted');
1266-    }
1267-    if ($query.length > 0) {
1268-      switch (e.which) {
1269-        case KEY_CODE.LEFT:
1270-          this._setRange('before', $query.get(0), range);
1271-          $query.removeClass('atwho-query');
1272-          return;
1273-        case KEY_CODE.RIGHT:
1274-          this._setRange('after', $query.get(0).nextSibling, range);
1275-          $query.removeClass('atwho-query');
1276-          return;
1277-      }
1278-    }
1279-    if ($query.length > 0 && (query_content = $query.attr('data-atwho-at-query'))) {
1280-      $query.empty().html(query_content).attr('data-atwho-at-query', null);
1281-      this._setRange('after', $query.get(0), range);
1282-    }
1283-    _range = range.cloneRange();
1284-    _range.setStart(range.startContainer, 0);
1285-    matched = this.callbacks("matcher").call(this, this.at, _range.toString(), this.getOpt('startWithSpace'), this.getOpt("acceptSpaceBar"));
1286-    isString = typeof matched === 'string';
1287-    if ($query.length === 0 && isString && (index = range.startOffset - this.at.length - matched.length) >= 0) {
1288-      range.setStart(range.startContainer, index);
1289-      $query = $('<span/>', this.app.document).attr(this.getOpt("editableAtwhoQueryAttrs")).addClass('atwho-query');
1290-      range.surroundContents($query.get(0));
1291-      lastNode = $query.contents().last().get(0);
1292-      if (lastNode) {
1293-        if (/firefox/i.test(navigator.userAgent)) {
1294-          range.setStart(lastNode, lastNode.length);
1295-          range.setEnd(lastNode, lastNode.length);
1296-          this._clearRange(range);
1297-        } else {
1298-          this._setRange('after', lastNode, range);
1299-        }
1300-      }
1301-    }
1302-    if (isString && matched.length < this.getOpt('minLen', 0)) {
1303-      return;
1304-    }
1305-    if (isString && matched.length <= this.getOpt('maxLen', 20)) {
1306-      query = {
1307-        text: matched,
1308-        el: $query
1309-      };
1310-      this.trigger("matched", [this.at, query.text]);
1311-      return this.query = query;
1312-    } else {
1313-      this.view.hide();
1314-      this.query = {
1315-        el: $query
1316-      };
1317-      if ($query.text().indexOf(this.at) >= 0) {
1318-        if (this._movingEvent(e) && $query.hasClass('atwho-inserted')) {
1319-          $query.removeClass('atwho-query');
1320-        } else if (false !== this.callbacks('afterMatchFailed').call(this, this.at, $query)) {
1321-          this._setRange("after", this._unwrap($query.text($query.text()).contents().first()));
1322-        }
1323-      }
1324-      return null;
1325-    }
1326-  };
1327-
1328-  EditableController.prototype.rect = function() {
1329-    var $iframe, iframeOffset, rect;
1330-    rect = this.query.el.offset();
1331-    if (!(rect && this.query.el[0].getClientRects().length)) {
1332-      return;
1333-    }
1334-    if (this.app.iframe && !this.app.iframeAsRoot) {
1335-      iframeOffset = ($iframe = $(this.app.iframe)).offset();
1336-      rect.left += iframeOffset.left - this.$inputor.scrollLeft();
1337-      rect.top += iframeOffset.top - this.$inputor.scrollTop();
1338-    }
1339-    rect.bottom = rect.top + this.query.el.height();
1340-    return rect;
1341-  };
1342-
1343-  EditableController.prototype.insert = function(content, $li) {
1344-    var data, overrides, range, suffix, suffixNode;
1345-    if (!this.$inputor.is(':focus')) {
1346-      this.$inputor.focus();
1347-    }
1348-    overrides = this.getOpt('functionOverrides');
1349-    if (overrides.insert) {
1350-      return overrides.insert.call(this, content, $li);
1351-    }
1352-    suffix = (suffix = this.getOpt('suffix')) === "" ? suffix : suffix || "\u00A0";
1353-    data = $li.data('item-data');
1354-    this.query.el.removeClass('atwho-query').addClass('atwho-inserted').html(content).attr('data-atwho-at-query', "" + data['atwho-at'] + this.query.text).attr('contenteditable', "false");
1355-    if (range = this._getRange()) {
1356-      if (this.query.el.length) {
1357-        range.setEndAfter(this.query.el[0]);
1358-      }
1359-      range.collapse(false);
1360-      range.insertNode(suffixNode = this.app.document.createTextNode("" + suffix));
1361-      this._setRange('after', suffixNode, range);
1362-    }
1363-    if (!this.$inputor.is(':focus')) {
1364-      this.$inputor.focus();
1365-    }
1366-    return this.$inputor.change();
1367-  };
1368-
1369-  return EditableController;
1370-
1371-})(Controller);
1372-
1373-var Model;
1374-
1375-Model = (function() {
1376-  function Model(context) {
1377-    this.context = context;
1378-    this.at = this.context.at;
1379-    this.storage = this.context.$inputor;
1380-  }
1381-
1382-  Model.prototype.destroy = function() {
1383-    return this.storage.data(this.at, null);
1384-  };
1385-
1386-  Model.prototype.saved = function() {
1387-    return this.fetch() > 0;
1388-  };
1389-
1390-  Model.prototype.query = function(query, callback) {
1391-    var _remoteFilter, data, searchKey;
1392-    data = this.fetch();
1393-    searchKey = this.context.getOpt("searchKey");
1394-    data = this.context.callbacks('filter').call(this.context, query, data, searchKey) || [];
1395-    _remoteFilter = this.context.callbacks('remoteFilter');
1396-    if (data.length > 0 || (!_remoteFilter && data.length === 0)) {
1397-      return callback(data);
1398-    } else {
1399-      return _remoteFilter.call(this.context, query, callback);
1400-    }
1401-  };
1402-
1403-  Model.prototype.fetch = function() {
1404-    return this.storage.data(this.at) || [];
1405-  };
1406-
1407-  Model.prototype.save = function(data) {
1408-    return this.storage.data(this.at, this.context.callbacks("beforeSave").call(this.context, data || []));
1409-  };
1410-
1411-  Model.prototype.load = function(data) {
1412-    if (!(this.saved() || !data)) {
1413-      return this._load(data);
1414-    }
1415-  };
1416-
1417-  Model.prototype.reload = function(data) {
1418-    return this._load(data);
1419-  };
1420-
1421-  Model.prototype._load = function(data) {
1422-    if (typeof data === "string") {
1423-      return $.ajax(data, {
1424-        dataType: "json"
1425-      }).done((function(_this) {
1426-        return function(data) {
1427-          return _this.save(data);
1428-        };
1429-      })(this));
1430-    } else {
1431-      return this.save(data);
1432-    }
1433-  };
1434-
1435-  return Model;
1436-
1437-})();
1438-
1439-var View;
1440-
1441-View = (function() {
1442-  function View(context) {
1443-    this.context = context;
1444-    this.$el = $("<div class='atwho-view'><ul class='atwho-view-ul'></ul></div>");
1445-    this.$elUl = this.$el.children();
1446-    this.timeoutID = null;
1447-    this.context.$el.append(this.$el);
1448-    this.bindEvent();
1449-  }
1450-
1451-  View.prototype.init = function() {
1452-    var header_tpl, id;
1453-    id = this.context.getOpt("alias") || this.context.at.charCodeAt(0);
1454-    header_tpl = this.context.getOpt("headerTpl");
1455-    if (header_tpl && this.$el.children().length === 1) {
1456-      this.$el.prepend(header_tpl);
1457-    }
1458-    return this.$el.attr({
1459-      'id': "at-view-" + id
1460-    });
1461-  };
1462-
1463-  View.prototype.destroy = function() {
1464-    return this.$el.remove();
1465-  };
1466-
1467-  View.prototype.bindEvent = function() {
1468-    var $menu, lastCoordX, lastCoordY;
1469-    $menu = this.$el.find('ul');
1470-    lastCoordX = 0;
1471-    lastCoordY = 0;
1472-    return $menu.on('mousemove.atwho-view', 'li', (function(_this) {
1473-      return function(e) {
1474-        var $cur;
1475-        if (lastCoordX === e.clientX && lastCoordY === e.clientY) {
1476-          return;
1477-        }
1478-        lastCoordX = e.clientX;
1479-        lastCoordY = e.clientY;
1480-        $cur = $(e.currentTarget);
1481-        if ($cur.hasClass('cur')) {
1482-          return;
1483-        }
1484-        $menu.find('.cur').removeClass('cur');
1485-        return $cur.addClass('cur');
1486-      };
1487-    })(this)).on('click.atwho-view', 'li', (function(_this) {
1488-      return function(e) {
1489-        $menu.find('.cur').removeClass('cur');
1490-        $(e.currentTarget).addClass('cur');
1491-        _this.choose(e);
1492-        return e.preventDefault();
1493-      };
1494-    })(this));
1495-  };
1496-
1497-  View.prototype.visible = function() {
1498-    return $.expr.filters.visible(this.$el[0]);
1499-  };
1500-
1501-  View.prototype.highlighted = function() {
1502-    return this.$el.find(".cur").length > 0;
1503-  };
1504-
1505-  View.prototype.choose = function(e) {
1506-    var $li, content;
1507-    if (($li = this.$el.find(".cur")).length) {
1508-      content = this.context.insertContentFor($li);
1509-      this.context._stopDelayedCall();
1510-      this.context.insert(this.context.callbacks("beforeInsert").call(this.context, content, $li, e), $li);
1511-      this.context.trigger("inserted", [$li, e]);
1512-      this.hide(e);
1513-    }
1514-    if (this.context.getOpt("hideWithoutSuffix")) {
1515-      return this.stopShowing = true;
1516-    }
1517-  };
1518-
1519-  View.prototype.reposition = function(rect) {
1520-    var _window, offset, overflowOffset, ref;
1521-    _window = this.context.app.iframeAsRoot ? this.context.app.window : window;
1522-    if (rect.bottom + this.$el.height() - $(_window).scrollTop() > $(_window).height()) {
1523-      rect.bottom = rect.top - this.$el.height();
1524-    }
1525-    if (rect.left > (overflowOffset = $(_window).width() - this.$el.width() - 5)) {
1526-      rect.left = overflowOffset;
1527-    }
1528-    offset = {
1529-      left: rect.left,
1530-      top: rect.bottom
1531-    };
1532-    if ((ref = this.context.callbacks("beforeReposition")) != null) {
1533-      ref.call(this.context, offset);
1534-    }
1535-    this.$el.offset(offset);
1536-    return this.context.trigger("reposition", [offset]);
1537-  };
1538-
1539-  View.prototype.next = function() {
1540-    var cur, next, nextEl, offset;
1541-    cur = this.$el.find('.cur').removeClass('cur');
1542-    next = cur.next();
1543-    if (!next.length) {
1544-      next = this.$el.find('li:first');
1545-    }
1546-    next.addClass('cur');
1547-    nextEl = next[0];
1548-    offset = nextEl.offsetTop + nextEl.offsetHeight + (nextEl.nextSibling ? nextEl.nextSibling.offsetHeight : 0);
1549-    return this.scrollTop(Math.max(0, offset - this.$el.height()));
1550-  };
1551-
1552-  View.prototype.prev = function() {
1553-    var cur, offset, prev, prevEl;
1554-    cur = this.$el.find('.cur').removeClass('cur');
1555-    prev = cur.prev();
1556-    if (!prev.length) {
1557-      prev = this.$el.find('li:last');
1558-    }
1559-    prev.addClass('cur');
1560-    prevEl = prev[0];
1561-    offset = prevEl.offsetTop + prevEl.offsetHeight + (prevEl.nextSibling ? prevEl.nextSibling.offsetHeight : 0);
1562-    return this.scrollTop(Math.max(0, offset - this.$el.height()));
1563-  };
1564-
1565-  View.prototype.scrollTop = function(scrollTop) {
1566-    var scrollDuration;
1567-    scrollDuration = this.context.getOpt('scrollDuration');
1568-    if (scrollDuration) {
1569-      return this.$elUl.animate({
1570-        scrollTop: scrollTop
1571-      }, scrollDuration);
1572-    } else {
1573-      return this.$elUl.scrollTop(scrollTop);
1574-    }
1575-  };
1576-
1577-  View.prototype.show = function() {
1578-    var rect;
1579-    if (this.stopShowing) {
1580-      this.stopShowing = false;
1581-      return;
1582-    }
1583-    if (!this.visible()) {
1584-      this.$el.show();
1585-      this.$el.scrollTop(0);
1586-      this.context.trigger('shown');
1587-    }
1588-    if (rect = this.context.rect()) {
1589-      return this.reposition(rect);
1590-    }
1591-  };
1592-
1593-  View.prototype.hide = function(e, time) {
1594-    var callback;
1595-    if (!this.visible()) {
1596-      return;
1597-    }
1598-    if (isNaN(time)) {
1599-      this.$el.hide();
1600-      return this.context.trigger('hidden', [e]);
1601-    } else {
1602-      callback = (function(_this) {
1603-        return function() {
1604-          return _this.hide();
1605-        };
1606-      })(this);
1607-      clearTimeout(this.timeoutID);
1608-      return this.timeoutID = setTimeout(callback, time);
1609-    }
1610-  };
1611-
1612-  View.prototype.render = function(list) {
1613-    var $li, $ul, i, item, len, li, tpl;
1614-    if (!($.isArray(list) && list.length > 0)) {
1615-      this.hide();
1616-      return;
1617-    }
1618-    this.$el.find('ul').empty();
1619-    $ul = this.$el.find('ul');
1620-    tpl = this.context.getOpt('displayTpl');
1621-    for (i = 0, len = list.length; i < len; i++) {
1622-      item = list[i];
1623-      item = $.extend({}, item, {
1624-        'atwho-at': this.context.at
1625-      });
1626-      li = this.context.callbacks("tplEval").call(this.context, tpl, item, "onDisplay");
1627-      $li = $(this.context.callbacks("highlighter").call(this.context, li, this.context.query.text));
1628-      $li.data("item-data", item);
1629-      $ul.append($li);
1630-    }
1631-    this.show();
1632-    if (this.context.getOpt('highlightFirst')) {
1633-      return $ul.find("li:first").addClass("cur");
1634-    }
1635-  };
1636-
1637-  return View;
1638-
1639-})();
1640-
1641-var Api;
1642-
1643-Api = {
1644-  load: function(at, data) {
1645-    var c;
1646-    if (c = this.controller(at)) {
1647-      return c.model.load(data);
1648-    }
1649-  },
1650-  isSelecting: function() {
1651-    var ref;
1652-    return !!((ref = this.controller()) != null ? ref.view.visible() : void 0);
1653-  },
1654-  hide: function() {
1655-    var ref;
1656-    return (ref = this.controller()) != null ? ref.view.hide() : void 0;
1657-  },
1658-  reposition: function() {
1659-    var c;
1660-    if (c = this.controller()) {
1661-      return c.view.reposition(c.rect());
1662-    }
1663-  },
1664-  setIframe: function(iframe, asRoot) {
1665-    this.setupRootElement(iframe, asRoot);
1666-    return null;
1667-  },
1668-  run: function() {
1669-    return this.dispatch();
1670-  },
1671-  destroy: function() {
1672-    this.shutdown();
1673-    return this.$inputor.data('atwho', null);
1674-  }
1675-};
1676-
1677-$.fn.atwho = function(method) {
1678-  var _args, result;
1679-  _args = arguments;
1680-  result = null;
1681-  this.filter('textarea, input, [contenteditable=""], [contenteditable=true]').each(function() {
1682-    var $this, app;
1683-    if (!(app = ($this = $(this)).data("atwho"))) {
1684-      $this.data('atwho', (app = new App(this)));
1685-    }
1686-    if (typeof method === 'object' || !method) {
1687-      return app.reg(method.at, method);
1688-    } else if (Api[method] && app) {
1689-      return result = Api[method].apply(app, Array.prototype.slice.call(_args, 1));
1690-    } else {
1691-      return $.error("Method " + method + " does not exist on jQuery.atwho");
1692-    }
1693-  });
1694-  if (result != null) {
1695-    return result;
1696-  } else {
1697-    return this;
1698-  }
1699-};
1700-
1701-$.fn.atwho["default"] = {
1702-  at: void 0,
1703-  alias: void 0,
1704-  data: null,
1705-  displayTpl: "<li>${name}</li>",
1706-  insertTpl: "${atwho-at}${name}",
1707-  headerTpl: null,
1708-  callbacks: DEFAULT_CALLBACKS,
1709-  functionOverrides: {},
1710-  searchKey: "name",
1711-  suffix: void 0,
1712-  hideWithoutSuffix: false,
1713-  startWithSpace: true,
1714-  acceptSpaceBar: false,
1715-  highlightFirst: true,
1716-  limit: 5,
1717-  maxLen: 20,
1718-  minLen: 0,
1719-  displayTimeout: 300,
1720-  delay: null,
1721-  spaceSelectsMatch: false,
1722-  tabSelectsMatch: true,
1723-  editableAtwhoQueryAttrs: {},
1724-  scrollDuration: 150,
1725-  suspendOnComposing: true,
1726-  lookUpOnClick: true
1727-};
1728-
1729-$.fn.atwho.debug = false;
1730-
1731-}));
1732diff --git src/bp-core/js/vendor/jquery.atwho.txt src/bp-core/js/vendor/jquery.atwho.txt
1733deleted file mode 100644
1734index 36cd1c122..000000000
1735--- src/bp-core/js/vendor/jquery.atwho.txt
1736+++ /dev/null
1737@@ -1,22 +0,0 @@
1738-Copyright (c) 2013 chord.luo@gmail.com
1739-
1740-Permission is hereby granted, free of charge, to any person
1741-obtaining a copy of this software and associated documentation
1742-files (the "Software"), to deal in the Software without
1743-restriction, including without limitation the rights to use,
1744-copy, modify, merge, publish, distribute, sublicense, and/or sell
1745-copies of the Software, and to permit persons to whom the
1746-Software is furnished to do so, subject to the following
1747-conditions:
1748-
1749-The above copyright notice and this permission notice shall be
1750-included in all copies or substantial portions of the Software.
1751-
1752-THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
1753-EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
1754-OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
1755-NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
1756-HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
1757-WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
1758-FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
1759-OTHER DEALINGS IN THE SOFTWARE.
1760diff --git src/bp-core/js/vendor/jquery.caret.js src/bp-core/js/vendor/jquery.caret.js
1761deleted file mode 100755
1762index 811ec63ee..000000000
1763--- src/bp-core/js/vendor/jquery.caret.js
1764+++ /dev/null
1765@@ -1,436 +0,0 @@
1766-(function (root, factory) {
1767-  if (typeof define === 'function' && define.amd) {
1768-    // AMD. Register as an anonymous module.
1769-    define(["jquery"], function ($) {
1770-      return (root.returnExportsGlobal = factory($));
1771-    });
1772-  } else if (typeof exports === 'object') {
1773-    // Node. Does not work with strict CommonJS, but
1774-    // only CommonJS-like enviroments that support module.exports,
1775-    // like Node.
1776-    module.exports = factory(require("jquery"));
1777-  } else {
1778-    factory(jQuery);
1779-  }
1780-}(this, function ($) {
1781-
1782-/*
1783-  Implement Github like autocomplete mentions
1784-  http://ichord.github.com/At.js
1785-
1786-  Copyright (c) 2013 chord.luo@gmail.com
1787-  Licensed under the MIT license.
1788-*/
1789-
1790-/*
1791-本插件操䜜 textarea 或者 input 内的插入笊
1792-只实现了获埗插入笊圚文本框䞭的䜍眮我讟眮
1793-插入笊的䜍眮.
1794-*/
1795-
1796-"use strict";
1797-var EditableCaret, InputCaret, Mirror, Utils, discoveryIframeOf, methods, oDocument, oFrame, oWindow, pluginName, setContextBy;
1798-
1799-pluginName = 'caret';
1800-
1801-EditableCaret = (function() {
1802-  function EditableCaret($inputor) {
1803-    this.$inputor = $inputor;
1804-    this.domInputor = this.$inputor[0];
1805-  }
1806-
1807-  EditableCaret.prototype.setPos = function(pos) {
1808-    var fn, found, offset, sel;
1809-    if (sel = oWindow.getSelection()) {
1810-      offset = 0;
1811-      found = false;
1812-      (fn = function(pos, parent) {
1813-        var node, range, _i, _len, _ref, _results;
1814-        _ref = parent.childNodes;
1815-        _results = [];
1816-        for (_i = 0, _len = _ref.length; _i < _len; _i++) {
1817-          node = _ref[_i];
1818-          if (found) {
1819-            break;
1820-          }
1821-          if (node.nodeType === 3) {
1822-            if (offset + node.length >= pos) {
1823-              found = true;
1824-              range = oDocument.createRange();
1825-              range.setStart(node, pos - offset);
1826-              sel.removeAllRanges();
1827-              sel.addRange(range);
1828-              break;
1829-            } else {
1830-              _results.push(offset += node.length);
1831-            }
1832-          } else {
1833-            _results.push(fn(pos, node));
1834-          }
1835-        }
1836-        return _results;
1837-      })(pos, this.domInputor);
1838-    }
1839-    return this.domInputor;
1840-  };
1841-
1842-  EditableCaret.prototype.getIEPosition = function() {
1843-    return this.getPosition();
1844-  };
1845-
1846-  EditableCaret.prototype.getPosition = function() {
1847-    var inputor_offset, offset;
1848-    offset = this.getOffset();
1849-    inputor_offset = this.$inputor.offset();
1850-    offset.left -= inputor_offset.left;
1851-    offset.top -= inputor_offset.top;
1852-    return offset;
1853-  };
1854-
1855-  EditableCaret.prototype.getOldIEPos = function() {
1856-    var preCaretTextRange, textRange;
1857-    textRange = oDocument.selection.createRange();
1858-    preCaretTextRange = oDocument.body.createTextRange();
1859-    preCaretTextRange.moveToElementText(this.domInputor);
1860-    preCaretTextRange.setEndPoint("EndToEnd", textRange);
1861-    return preCaretTextRange.text.length;
1862-  };
1863-
1864-  EditableCaret.prototype.getPos = function() {
1865-    var clonedRange, pos, range;
1866-    if (range = this.range()) {
1867-      clonedRange = range.cloneRange();
1868-      clonedRange.selectNodeContents(this.domInputor);
1869-      clonedRange.setEnd(range.endContainer, range.endOffset);
1870-      pos = clonedRange.toString().length;
1871-      clonedRange.detach();
1872-      return pos;
1873-    } else if (oDocument.selection) {
1874-      return this.getOldIEPos();
1875-    }
1876-  };
1877-
1878-  EditableCaret.prototype.getOldIEOffset = function() {
1879-    var range, rect;
1880-    range = oDocument.selection.createRange().duplicate();
1881-    range.moveStart("character", -1);
1882-    rect = range.getBoundingClientRect();
1883-    return {
1884-      height: rect.bottom - rect.top,
1885-      left: rect.left,
1886-      top: rect.top
1887-    };
1888-  };
1889-
1890-  EditableCaret.prototype.getOffset = function(pos) {
1891-    var clonedRange, offset, range, rect, shadowCaret;
1892-    if (oWindow.getSelection && (range = this.range())) {
1893-      if (range.endOffset - 1 > 0 && range.endContainer !== this.domInputor) {
1894-        clonedRange = range.cloneRange();
1895-        clonedRange.setStart(range.endContainer, range.endOffset - 1);
1896-        clonedRange.setEnd(range.endContainer, range.endOffset);
1897-        rect = clonedRange.getBoundingClientRect();
1898-        offset = {
1899-          height: rect.height,
1900-          left: rect.left + rect.width,
1901-          top: rect.top
1902-        };
1903-        clonedRange.detach();
1904-      }
1905-      if (!offset || (offset != null ? offset.height : void 0) === 0) {
1906-        clonedRange = range.cloneRange();
1907-        shadowCaret = $(oDocument.createTextNode("|"));
1908-        clonedRange.insertNode(shadowCaret[0]);
1909-        clonedRange.selectNode(shadowCaret[0]);
1910-        rect = clonedRange.getBoundingClientRect();
1911-        offset = {
1912-          height: rect.height,
1913-          left: rect.left,
1914-          top: rect.top
1915-        };
1916-        shadowCaret.remove();
1917-        clonedRange.detach();
1918-      }
1919-    } else if (oDocument.selection) {
1920-      offset = this.getOldIEOffset();
1921-    }
1922-    if (offset) {
1923-      offset.top += $(oWindow).scrollTop();
1924-      offset.left += $(oWindow).scrollLeft();
1925-    }
1926-    return offset;
1927-  };
1928-
1929-  EditableCaret.prototype.range = function() {
1930-    var sel;
1931-    if (!oWindow.getSelection) {
1932-      return;
1933-    }
1934-    sel = oWindow.getSelection();
1935-    if (sel.rangeCount > 0) {
1936-      return sel.getRangeAt(0);
1937-    } else {
1938-      return null;
1939-    }
1940-  };
1941-
1942-  return EditableCaret;
1943-
1944-})();
1945-
1946-InputCaret = (function() {
1947-  function InputCaret($inputor) {
1948-    this.$inputor = $inputor;
1949-    this.domInputor = this.$inputor[0];
1950-  }
1951-
1952-  InputCaret.prototype.getIEPos = function() {
1953-    var endRange, inputor, len, normalizedValue, pos, range, textInputRange;
1954-    inputor = this.domInputor;
1955-    range = oDocument.selection.createRange();
1956-    pos = 0;
1957-    if (range && range.parentElement() === inputor) {
1958-      normalizedValue = inputor.value.replace(/\r\n/g, "\n");
1959-      len = normalizedValue.length;
1960-      textInputRange = inputor.createTextRange();
1961-      textInputRange.moveToBookmark(range.getBookmark());
1962-      endRange = inputor.createTextRange();
1963-      endRange.collapse(false);
1964-      if (textInputRange.compareEndPoints("StartToEnd", endRange) > -1) {
1965-        pos = len;
1966-      } else {
1967-        pos = -textInputRange.moveStart("character", -len);
1968-      }
1969-    }
1970-    return pos;
1971-  };
1972-
1973-  InputCaret.prototype.getPos = function() {
1974-    if (oDocument.selection) {
1975-      return this.getIEPos();
1976-    } else {
1977-      return this.domInputor.selectionStart;
1978-    }
1979-  };
1980-
1981-  InputCaret.prototype.setPos = function(pos) {
1982-    var inputor, range;
1983-    inputor = this.domInputor;
1984-    if (oDocument.selection) {
1985-      range = inputor.createTextRange();
1986-      range.move("character", pos);
1987-      range.select();
1988-    } else if (inputor.setSelectionRange) {
1989-      inputor.setSelectionRange(pos, pos);
1990-    }
1991-    return inputor;
1992-  };
1993-
1994-  InputCaret.prototype.getIEOffset = function(pos) {
1995-    var h, textRange, x, y;
1996-    textRange = this.domInputor.createTextRange();
1997-    pos || (pos = this.getPos());
1998-    textRange.move('character', pos);
1999-    x = textRange.boundingLeft;
2000-    y = textRange.boundingTop;
2001-    h = textRange.boundingHeight;
2002-    return {
2003-      left: x,
2004-      top: y,
2005-      height: h
2006-    };
2007-  };
2008-
2009-  InputCaret.prototype.getOffset = function(pos) {
2010-    var $inputor, offset, position;
2011-    $inputor = this.$inputor;
2012-    if (oDocument.selection) {
2013-      offset = this.getIEOffset(pos);
2014-      offset.top += $(oWindow).scrollTop() + $inputor.scrollTop();
2015-      offset.left += $(oWindow).scrollLeft() + $inputor.scrollLeft();
2016-      return offset;
2017-    } else {
2018-      offset = $inputor.offset();
2019-      position = this.getPosition(pos);
2020-      return offset = {
2021-        left: offset.left + position.left - $inputor.scrollLeft(),
2022-        top: offset.top + position.top - $inputor.scrollTop(),
2023-        height: position.height
2024-      };
2025-    }
2026-  };
2027-
2028-  InputCaret.prototype.getPosition = function(pos) {
2029-    var $inputor, at_rect, end_range, format, html, mirror, start_range;
2030-    $inputor = this.$inputor;
2031-    format = function(value) {
2032-      value = value.replace(/<|>|`|"|&/g, '?').replace(/\r\n|\r|\n/g, "<br/>");
2033-      if (/firefox/i.test(navigator.userAgent)) {
2034-        value = value.replace(/\s/g, '&nbsp;');
2035-      }
2036-      return value;
2037-    };
2038-    if (pos === void 0) {
2039-      pos = this.getPos();
2040-    }
2041-    start_range = $inputor.val().slice(0, pos);
2042-    end_range = $inputor.val().slice(pos);
2043-    html = "<span style='position: relative; display: inline;'>" + format(start_range) + "</span>";
2044-    html += "<span id='caret' style='position: relative; display: inline;'>|</span>";
2045-    html += "<span style='position: relative; display: inline;'>" + format(end_range) + "</span>";
2046-    mirror = new Mirror($inputor);
2047-    return at_rect = mirror.create(html).rect();
2048-  };
2049-
2050-  InputCaret.prototype.getIEPosition = function(pos) {
2051-    var h, inputorOffset, offset, x, y;
2052-    offset = this.getIEOffset(pos);
2053-    inputorOffset = this.$inputor.offset();
2054-    x = offset.left - inputorOffset.left;
2055-    y = offset.top - inputorOffset.top;
2056-    h = offset.height;
2057-    return {
2058-      left: x,
2059-      top: y,
2060-      height: h
2061-    };
2062-  };
2063-
2064-  return InputCaret;
2065-
2066-})();
2067-
2068-Mirror = (function() {
2069-  Mirror.prototype.css_attr = ["borderBottomWidth", "borderLeftWidth", "borderRightWidth", "borderTopStyle", "borderRightStyle", "borderBottomStyle", "borderLeftStyle", "borderTopWidth", "boxSizing", "fontFamily", "fontSize", "fontWeight", "height", "letterSpacing", "lineHeight", "marginBottom", "marginLeft", "marginRight", "marginTop", "outlineWidth", "overflow", "overflowX", "overflowY", "paddingBottom", "paddingLeft", "paddingRight", "paddingTop", "textAlign", "textOverflow", "textTransform", "whiteSpace", "wordBreak", "wordWrap"];
2070-
2071-  function Mirror($inputor) {
2072-    this.$inputor = $inputor;
2073-  }
2074-
2075-  Mirror.prototype.mirrorCss = function() {
2076-    var css,
2077-      _this = this;
2078-    css = {
2079-      position: 'absolute',
2080-      left: -9999,
2081-      top: 0,
2082-      zIndex: -20000
2083-    };
2084-    if (this.$inputor.prop('tagName') === 'TEXTAREA') {
2085-      this.css_attr.push('width');
2086-    }
2087-    $.each(this.css_attr, function(i, p) {
2088-      return css[p] = _this.$inputor.css(p);
2089-    });
2090-    return css;
2091-  };
2092-
2093-  Mirror.prototype.create = function(html) {
2094-    this.$mirror = $('<div></div>');
2095-    this.$mirror.css(this.mirrorCss());
2096-    this.$mirror.html(html);
2097-    this.$inputor.after(this.$mirror);
2098-    return this;
2099-  };
2100-
2101-  Mirror.prototype.rect = function() {
2102-    var $flag, pos, rect;
2103-    $flag = this.$mirror.find("#caret");
2104-    pos = $flag.position();
2105-    rect = {
2106-      left: pos.left,
2107-      top: pos.top,
2108-      height: $flag.height()
2109-    };
2110-    this.$mirror.remove();
2111-    return rect;
2112-  };
2113-
2114-  return Mirror;
2115-
2116-})();
2117-
2118-Utils = {
2119-  contentEditable: function($inputor) {
2120-    return !!($inputor[0].contentEditable && $inputor[0].contentEditable === 'true');
2121-  }
2122-};
2123-
2124-methods = {
2125-  pos: function(pos) {
2126-    if (pos || pos === 0) {
2127-      return this.setPos(pos);
2128-    } else {
2129-      return this.getPos();
2130-    }
2131-  },
2132-  position: function(pos) {
2133-    if (oDocument.selection) {
2134-      return this.getIEPosition(pos);
2135-    } else {
2136-      return this.getPosition(pos);
2137-    }
2138-  },
2139-  offset: function(pos) {
2140-    var offset;
2141-    offset = this.getOffset(pos);
2142-    return offset;
2143-  }
2144-};
2145-
2146-oDocument = null;
2147-
2148-oWindow = null;
2149-
2150-oFrame = null;
2151-
2152-setContextBy = function(settings) {
2153-  var iframe;
2154-  if (iframe = settings != null ? settings.iframe : void 0) {
2155-    oFrame = iframe;
2156-    oWindow = iframe.contentWindow;
2157-    return oDocument = iframe.contentDocument || oWindow.document;
2158-  } else {
2159-    oFrame = void 0;
2160-    oWindow = window;
2161-    return oDocument = document;
2162-  }
2163-};
2164-
2165-discoveryIframeOf = function($dom) {
2166-  var error;
2167-  oDocument = $dom[0].ownerDocument;
2168-  oWindow = oDocument.defaultView || oDocument.parentWindow;
2169-  try {
2170-    return oFrame = oWindow.frameElement;
2171-  } catch (_error) {
2172-    error = _error;
2173-  }
2174-};
2175-
2176-$.fn.caret = function(method, value, settings) {
2177-  var caret;
2178-  if (methods[method]) {
2179-    if ($.isPlainObject(value)) {
2180-      setContextBy(value);
2181-      value = void 0;
2182-    } else {
2183-      setContextBy(settings);
2184-    }
2185-    caret = Utils.contentEditable(this) ? new EditableCaret(this) : new InputCaret(this);
2186-    return methods[method].apply(caret, [value]);
2187-  } else {
2188-    return $.error("Method " + method + " does not exist on jQuery.caret");
2189-  }
2190-};
2191-
2192-$.fn.caret.EditableCaret = EditableCaret;
2193-
2194-$.fn.caret.InputCaret = InputCaret;
2195-
2196-$.fn.caret.Utils = Utils;
2197-
2198-$.fn.caret.apis = methods;
2199-
2200-
2201-}));
2202diff --git src/bp-core/js/vendor/jquery.caret.txt src/bp-core/js/vendor/jquery.caret.txt
2203deleted file mode 100644
2204index 36cd1c122..000000000
2205--- src/bp-core/js/vendor/jquery.caret.txt
2206+++ /dev/null
2207@@ -1,22 +0,0 @@
2208-Copyright (c) 2013 chord.luo@gmail.com
2209-
2210-Permission is hereby granted, free of charge, to any person
2211-obtaining a copy of this software and associated documentation
2212-files (the "Software"), to deal in the Software without
2213-restriction, including without limitation the rights to use,
2214-copy, modify, merge, publish, distribute, sublicense, and/or sell
2215-copies of the Software, and to permit persons to whom the
2216-Software is furnished to do so, subject to the following
2217-conditions:
2218-
2219-The above copyright notice and this permission notice shall be
2220-included in all copies or substantial portions of the Software.
2221-
2222-THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
2223-EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
2224-OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
2225-NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
2226-HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
2227-WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
2228-FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
2229-OTHER DEALINGS IN THE SOFTWARE.
2230diff --git src/bp-core/js/vendor/tribute.js src/bp-core/js/vendor/tribute.js
2231new file mode 100644
2232index 000000000..bd029c928
2233--- /dev/null
2234+++ src/bp-core/js/vendor/tribute.js
2235@@ -0,0 +1,1798 @@
2236+(function(f){if(typeof exports==="object"&&typeof module!=="undefined"){module.exports=f()}else if(typeof define==="function"&&define.amd){define([],f)}else{var g;if(typeof window!=="undefined"){g=window}else if(typeof global!=="undefined"){g=global}else if(typeof self!=="undefined"){g=self}else{g=this}g.Tribute = f()}})(function(){var define,module,exports;return (function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c="function"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error("Cannot find module '"+i+"'");throw a.code="MODULE_NOT_FOUND",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u="function"==typeof require&&require,i=0;i<t.length;i++)o(t[i]);return o}return r})()({1:[function(require,module,exports){
2237+"use strict";
2238+
2239+Object.defineProperty(exports, "__esModule", {
2240+    value: true
2241+});
2242+
2243+var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
2244+
2245+var _utils = require("./utils");
2246+
2247+var _utils2 = _interopRequireDefault(_utils);
2248+
2249+var _TributeEvents = require("./TributeEvents");
2250+
2251+var _TributeEvents2 = _interopRequireDefault(_TributeEvents);
2252+
2253+var _TributeMenuEvents = require("./TributeMenuEvents");
2254+
2255+var _TributeMenuEvents2 = _interopRequireDefault(_TributeMenuEvents);
2256+
2257+var _TributeRange = require("./TributeRange");
2258+
2259+var _TributeRange2 = _interopRequireDefault(_TributeRange);
2260+
2261+var _TributeSearch = require("./TributeSearch");
2262+
2263+var _TributeSearch2 = _interopRequireDefault(_TributeSearch);
2264+
2265+function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
2266+
2267+function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
2268+
2269+var Tribute = function () {
2270+    function Tribute(_ref) {
2271+        var _this = this;
2272+
2273+        var _ref$values = _ref.values,
2274+            values = _ref$values === undefined ? null : _ref$values,
2275+            _ref$iframe = _ref.iframe,
2276+            iframe = _ref$iframe === undefined ? null : _ref$iframe,
2277+            _ref$selectClass = _ref.selectClass,
2278+            selectClass = _ref$selectClass === undefined ? 'highlight' : _ref$selectClass,
2279+            _ref$trigger = _ref.trigger,
2280+            trigger = _ref$trigger === undefined ? '@' : _ref$trigger,
2281+            _ref$selectTemplate = _ref.selectTemplate,
2282+            selectTemplate = _ref$selectTemplate === undefined ? null : _ref$selectTemplate,
2283+            _ref$menuItemTemplate = _ref.menuItemTemplate,
2284+            menuItemTemplate = _ref$menuItemTemplate === undefined ? null : _ref$menuItemTemplate,
2285+            _ref$lookup = _ref.lookup,
2286+            lookup = _ref$lookup === undefined ? 'key' : _ref$lookup,
2287+            _ref$fillAttr = _ref.fillAttr,
2288+            fillAttr = _ref$fillAttr === undefined ? 'value' : _ref$fillAttr,
2289+            _ref$collection = _ref.collection,
2290+            collection = _ref$collection === undefined ? null : _ref$collection,
2291+            _ref$menuContainer = _ref.menuContainer,
2292+            menuContainer = _ref$menuContainer === undefined ? null : _ref$menuContainer,
2293+            _ref$noMatchTemplate = _ref.noMatchTemplate,
2294+            noMatchTemplate = _ref$noMatchTemplate === undefined ? null : _ref$noMatchTemplate,
2295+            _ref$requireLeadingSp = _ref.requireLeadingSpace,
2296+            requireLeadingSpace = _ref$requireLeadingSp === undefined ? true : _ref$requireLeadingSp,
2297+            _ref$allowSpaces = _ref.allowSpaces,
2298+            allowSpaces = _ref$allowSpaces === undefined ? false : _ref$allowSpaces,
2299+            _ref$replaceTextSuffi = _ref.replaceTextSuffix,
2300+            replaceTextSuffix = _ref$replaceTextSuffi === undefined ? null : _ref$replaceTextSuffi,
2301+            _ref$positionMenu = _ref.positionMenu,
2302+            positionMenu = _ref$positionMenu === undefined ? true : _ref$positionMenu,
2303+            _ref$spaceSelectsMatc = _ref.spaceSelectsMatch,
2304+            spaceSelectsMatch = _ref$spaceSelectsMatc === undefined ? false : _ref$spaceSelectsMatc;
2305+
2306+        _classCallCheck(this, Tribute);
2307+
2308+        this.menuSelected = 0;
2309+        this.current = {};
2310+        this.inputEvent = false;
2311+        this.isActive = false;
2312+        this.menuContainer = menuContainer;
2313+        this.allowSpaces = allowSpaces;
2314+        this.replaceTextSuffix = replaceTextSuffix;
2315+        this.positionMenu = positionMenu;
2316+        this.hasTrailingSpace = false;
2317+        this.spaceSelectsMatch = spaceSelectsMatch;
2318+
2319+        if (values) {
2320+            this.collection = [{
2321+                // symbol that starts the lookup
2322+                trigger: trigger,
2323+
2324+                // is it wrapped in an iframe
2325+                iframe: iframe,
2326+
2327+                // class applied to selected item
2328+                selectClass: selectClass,
2329+
2330+                // function called on select that retuns the content to insert
2331+                selectTemplate: (selectTemplate || Tribute.defaultSelectTemplate).bind(this),
2332+
2333+                // function called that returns content for an item
2334+                menuItemTemplate: (menuItemTemplate || Tribute.defaultMenuItemTemplate).bind(this),
2335+
2336+                // function called when menu is empty, disables hiding of menu.
2337+                noMatchTemplate: function (t) {
2338+                    if (typeof t === 'function') {
2339+                        return t.bind(_this);
2340+                    }
2341+
2342+                    return noMatchTemplate;
2343+                }(noMatchTemplate),
2344+
2345+                // column to search against in the object
2346+                lookup: lookup,
2347+
2348+                // column that contains the content to insert by default
2349+                fillAttr: fillAttr,
2350+
2351+                // array of objects or a function returning an array of objects
2352+                values: values,
2353+
2354+                requireLeadingSpace: requireLeadingSpace
2355+            }];
2356+        } else if (collection) {
2357+            this.collection = collection.map(function (item) {
2358+                return {
2359+                    trigger: item.trigger || trigger,
2360+                    iframe: item.iframe || iframe,
2361+                    selectClass: item.selectClass || selectClass,
2362+                    selectTemplate: (item.selectTemplate || Tribute.defaultSelectTemplate).bind(_this),
2363+                    menuItemTemplate: (item.menuItemTemplate || Tribute.defaultMenuItemTemplate).bind(_this),
2364+                    // function called when menu is empty, disables hiding of menu.
2365+                    noMatchTemplate: function (t) {
2366+                        if (typeof t === 'function') {
2367+                            return t.bind(_this);
2368+                        }
2369+
2370+                        return null;
2371+                    }(noMatchTemplate),
2372+                    lookup: item.lookup || lookup,
2373+                    fillAttr: item.fillAttr || fillAttr,
2374+                    values: item.values,
2375+                    requireLeadingSpace: item.requireLeadingSpace
2376+                };
2377+            });
2378+        } else {
2379+            throw new Error('[Tribute] No collection specified.');
2380+        }
2381+
2382+        new _TributeRange2.default(this);
2383+        new _TributeEvents2.default(this);
2384+        new _TributeMenuEvents2.default(this);
2385+        new _TributeSearch2.default(this);
2386+    }
2387+
2388+    _createClass(Tribute, [{
2389+        key: "triggers",
2390+        value: function triggers() {
2391+            return this.collection.map(function (config) {
2392+                return config.trigger;
2393+            });
2394+        }
2395+    }, {
2396+        key: "attach",
2397+        value: function attach(el) {
2398+            if (!el) {
2399+                throw new Error('[Tribute] Must pass in a DOM node or NodeList.');
2400+            }
2401+
2402+            // Check if it is a jQuery collection
2403+            if (typeof jQuery !== 'undefined' && el instanceof jQuery) {
2404+                el = el.get();
2405+            }
2406+
2407+            // Is el an Array/Array-like object?
2408+            if (el.constructor === NodeList || el.constructor === HTMLCollection || el.constructor === Array) {
2409+                var length = el.length;
2410+                for (var i = 0; i < length; ++i) {
2411+                    this._attach(el[i]);
2412+                }
2413+            } else {
2414+                this._attach(el);
2415+            }
2416+        }
2417+    }, {
2418+        key: "_attach",
2419+        value: function _attach(el) {
2420+            if (el.hasAttribute('data-tribute')) {
2421+                console.warn('Tribute was already bound to ' + el.nodeName);
2422+            }
2423+
2424+            this.ensureEditable(el);
2425+            this.events.bind(el);
2426+            el.setAttribute('data-tribute', true);
2427+        }
2428+    }, {
2429+        key: "ensureEditable",
2430+        value: function ensureEditable(element) {
2431+            if (Tribute.inputTypes().indexOf(element.nodeName) === -1) {
2432+                if (element.contentEditable) {
2433+                    element.contentEditable = true;
2434+                } else {
2435+                    throw new Error('[Tribute] Cannot bind to ' + element.nodeName);
2436+                }
2437+            }
2438+        }
2439+    }, {
2440+        key: "createMenu",
2441+        value: function createMenu() {
2442+            var wrapper = this.range.getDocument().createElement('div'),
2443+                ul = this.range.getDocument().createElement('ul');
2444+
2445+            wrapper.className = 'tribute-container';
2446+            wrapper.appendChild(ul);
2447+
2448+            if (this.menuContainer) {
2449+                return this.menuContainer.appendChild(wrapper);
2450+            }
2451+
2452+            return this.range.getDocument().body.appendChild(wrapper);
2453+        }
2454+    }, {
2455+        key: "showMenuFor",
2456+        value: function showMenuFor(element, scrollTo) {
2457+            var _this2 = this;
2458+
2459+            // Only proceed if menu isn't already shown for the current element & mentionText
2460+            if (this.isActive && this.current.element === element && this.current.mentionText === this.currentMentionTextSnapshot) {
2461+                return;
2462+            }
2463+            this.currentMentionTextSnapshot = this.current.mentionText;
2464+
2465+            // create the menu if it doesn't exist.
2466+            if (!this.menu) {
2467+                this.menu = this.createMenu();
2468+                element.tributeMenu = this.menu;
2469+                this.menuEvents.bind(this.menu);
2470+            }
2471+
2472+            this.isActive = true;
2473+            this.menuSelected = 0;
2474+
2475+            if (!this.current.mentionText) {
2476+                this.current.mentionText = '';
2477+            }
2478+
2479+            var processValues = function processValues(values) {
2480+                // Tribute may not be active any more by the time the value callback returns
2481+                if (!_this2.isActive) {
2482+                    return;
2483+                }
2484+
2485+                var items = _this2.search.filter(_this2.current.mentionText, values, {
2486+                    pre: '<span>',
2487+                    post: '</span>',
2488+                    extract: function extract(el) {
2489+                        if (typeof _this2.current.collection.lookup === 'string') {
2490+                            return el[_this2.current.collection.lookup];
2491+                        } else if (typeof _this2.current.collection.lookup === 'function') {
2492+                            return _this2.current.collection.lookup(el, _this2.current.mentionText);
2493+                        } else {
2494+                            throw new Error('Invalid lookup attribute, lookup must be string or function.');
2495+                        }
2496+                    }
2497+                });
2498+
2499+                _this2.current.filteredItems = items;
2500+
2501+                var ul = _this2.menu.querySelector('ul');
2502+
2503+                _this2.range.positionMenuAtCaret(scrollTo);
2504+
2505+                if (!items.length) {
2506+                    var noMatchEvent = new CustomEvent('tribute-no-match', { detail: _this2.menu });
2507+                    _this2.current.element.dispatchEvent(noMatchEvent);
2508+                    if (!_this2.current.collection.noMatchTemplate) {
2509+                        _this2.hideMenu();
2510+                    } else {
2511+                        ul.innerHTML = _this2.current.collection.noMatchTemplate();
2512+                    }
2513+
2514+                    return;
2515+                }
2516+
2517+                ul.innerHTML = '';
2518+
2519+                items.forEach(function (item, index) {
2520+                    var li = _this2.range.getDocument().createElement('li');
2521+                    li.setAttribute('data-index', index);
2522+                    li.addEventListener('mouseenter', function (e) {
2523+                        var li = e.target;
2524+                        var index = li.getAttribute('data-index');
2525+                        _this2.events.setActiveLi(index);
2526+                    });
2527+                    if (_this2.menuSelected === index) {
2528+                        li.className = _this2.current.collection.selectClass;
2529+                    }
2530+                    li.innerHTML = _this2.current.collection.menuItemTemplate(item);
2531+                    ul.appendChild(li);
2532+                });
2533+            };
2534+
2535+            if (typeof this.current.collection.values === 'function') {
2536+                this.current.collection.values(this.current.mentionText, processValues);
2537+            } else {
2538+                processValues(this.current.collection.values);
2539+            }
2540+        }
2541+    }, {
2542+        key: "showMenuForCollection",
2543+        value: function showMenuForCollection(element, collectionIndex) {
2544+            if (element !== document.activeElement) {
2545+                this.placeCaretAtEnd(element);
2546+            }
2547+
2548+            this.current.collection = this.collection[collectionIndex || 0];
2549+            this.current.externalTrigger = true;
2550+            this.current.element = element;
2551+
2552+            if (element.isContentEditable) this.insertTextAtCursor(this.current.collection.trigger);else this.insertAtCaret(element, this.current.collection.trigger);
2553+
2554+            this.showMenuFor(element);
2555+        }
2556+
2557+        // TODO: make sure this works for inputs/textareas
2558+
2559+    }, {
2560+        key: "placeCaretAtEnd",
2561+        value: function placeCaretAtEnd(el) {
2562+            el.focus();
2563+            if (typeof window.getSelection != "undefined" && typeof document.createRange != "undefined") {
2564+                var range = document.createRange();
2565+                range.selectNodeContents(el);
2566+                range.collapse(false);
2567+                var sel = window.getSelection();
2568+                sel.removeAllRanges();
2569+                sel.addRange(range);
2570+            } else if (typeof document.body.createTextRange != "undefined") {
2571+                var textRange = document.body.createTextRange();
2572+                textRange.moveToElementText(el);
2573+                textRange.collapse(false);
2574+                textRange.select();
2575+            }
2576+        }
2577+
2578+        // for contenteditable
2579+
2580+    }, {
2581+        key: "insertTextAtCursor",
2582+        value: function insertTextAtCursor(text) {
2583+            var sel, range, html;
2584+            sel = window.getSelection();
2585+            range = sel.getRangeAt(0);
2586+            range.deleteContents();
2587+            var textNode = document.createTextNode(text);
2588+            range.insertNode(textNode);
2589+            range.selectNodeContents(textNode);
2590+            range.collapse(false);
2591+            sel.removeAllRanges();
2592+            sel.addRange(range);
2593+        }
2594+
2595+        // for regular inputs
2596+
2597+    }, {
2598+        key: "insertAtCaret",
2599+        value: function insertAtCaret(textarea, text) {
2600+            var scrollPos = textarea.scrollTop;
2601+            var caretPos = textarea.selectionStart;
2602+
2603+            var front = textarea.value.substring(0, caretPos);
2604+            var back = textarea.value.substring(textarea.selectionEnd, textarea.value.length);
2605+            textarea.value = front + text + back;
2606+            caretPos = caretPos + text.length;
2607+            textarea.selectionStart = caretPos;
2608+            textarea.selectionEnd = caretPos;
2609+            textarea.focus();
2610+            textarea.scrollTop = scrollPos;
2611+        }
2612+    }, {
2613+        key: "hideMenu",
2614+        value: function hideMenu() {
2615+            if (this.menu) {
2616+                this.menu.style.cssText = 'display: none;';
2617+                this.isActive = false;
2618+                this.menuSelected = 0;
2619+                this.current = {};
2620+            }
2621+        }
2622+    }, {
2623+        key: "selectItemAtIndex",
2624+        value: function selectItemAtIndex(index, originalEvent) {
2625+            index = parseInt(index);
2626+            if (typeof index !== 'number') return;
2627+            var item = this.current.filteredItems[index];
2628+            var content = this.current.collection.selectTemplate(item);
2629+            if (content !== null) this.replaceText(content, originalEvent, item);
2630+        }
2631+    }, {
2632+        key: "replaceText",
2633+        value: function replaceText(content, originalEvent, item) {
2634+            this.range.replaceTriggerText(content, true, true, originalEvent, item);
2635+        }
2636+    }, {
2637+        key: "_append",
2638+        value: function _append(collection, newValues, replace) {
2639+            if (typeof collection.values === 'function') {
2640+                throw new Error('Unable to append to values, as it is a function.');
2641+            } else if (!replace) {
2642+                collection.values = collection.values.concat(newValues);
2643+            } else {
2644+                collection.values = newValues;
2645+            }
2646+        }
2647+    }, {
2648+        key: "append",
2649+        value: function append(collectionIndex, newValues, replace) {
2650+            var index = parseInt(collectionIndex);
2651+            if (typeof index !== 'number') throw new Error('please provide an index for the collection to update.');
2652+
2653+            var collection = this.collection[index];
2654+
2655+            this._append(collection, newValues, replace);
2656+        }
2657+    }, {
2658+        key: "appendCurrent",
2659+        value: function appendCurrent(newValues, replace) {
2660+            if (this.isActive) {
2661+                this._append(this.current.collection, newValues, replace);
2662+            } else {
2663+                throw new Error('No active state. Please use append instead and pass an index.');
2664+            }
2665+        }
2666+    }, {
2667+        key: "detach",
2668+        value: function detach(el) {
2669+            if (!el) {
2670+                throw new Error('[Tribute] Must pass in a DOM node or NodeList.');
2671+            }
2672+
2673+            // Check if it is a jQuery collection
2674+            if (typeof jQuery !== 'undefined' && el instanceof jQuery) {
2675+                el = el.get();
2676+            }
2677+
2678+            // Is el an Array/Array-like object?
2679+            if (el.constructor === NodeList || el.constructor === HTMLCollection || el.constructor === Array) {
2680+                var length = el.length;
2681+                for (var i = 0; i < length; ++i) {
2682+                    this._detach(el[i]);
2683+                }
2684+            } else {
2685+                this._detach(el);
2686+            }
2687+        }
2688+    }, {
2689+        key: "_detach",
2690+        value: function _detach(el) {
2691+            var _this3 = this;
2692+
2693+            this.events.unbind(el);
2694+            if (el.tributeMenu) {
2695+                this.menuEvents.unbind(el.tributeMenu);
2696+            }
2697+
2698+            setTimeout(function () {
2699+                el.removeAttribute('data-tribute');
2700+                _this3.isActive = false;
2701+                if (el.tributeMenu) {
2702+                    el.tributeMenu.remove();
2703+                }
2704+            });
2705+        }
2706+    }], [{
2707+        key: "defaultSelectTemplate",
2708+        value: function defaultSelectTemplate(item) {
2709+            if (typeof item === 'undefined') return null;
2710+            if (this.range.isContentEditable(this.current.element)) {
2711+                return '<span class="tribute-mention">' + (this.current.collection.trigger + item.original[this.current.collection.fillAttr]) + '</span>';
2712+            }
2713+
2714+            return this.current.collection.trigger + item.original[this.current.collection.fillAttr];
2715+        }
2716+    }, {
2717+        key: "defaultMenuItemTemplate",
2718+        value: function defaultMenuItemTemplate(matchItem) {
2719+            return matchItem.string;
2720+        }
2721+    }, {
2722+        key: "inputTypes",
2723+        value: function inputTypes() {
2724+            return ['TEXTAREA', 'INPUT'];
2725+        }
2726+    }]);
2727+
2728+    return Tribute;
2729+}();
2730+
2731+exports.default = Tribute;
2732+module.exports = exports["default"];
2733+
2734+},{"./TributeEvents":2,"./TributeMenuEvents":3,"./TributeRange":4,"./TributeSearch":5,"./utils":7}],2:[function(require,module,exports){
2735+'use strict';
2736+
2737+Object.defineProperty(exports, "__esModule", {
2738+    value: true
2739+});
2740+
2741+var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
2742+
2743+function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
2744+
2745+var TributeEvents = function () {
2746+    function TributeEvents(tribute) {
2747+        _classCallCheck(this, TributeEvents);
2748+
2749+        this.tribute = tribute;
2750+        this.tribute.events = this;
2751+    }
2752+
2753+    _createClass(TributeEvents, [{
2754+        key: 'bind',
2755+        value: function bind(element) {
2756+            element.boundKeydown = this.keydown.bind(element, this);
2757+            element.boundKeyup = this.keyup.bind(element, this);
2758+            element.boundInput = this.input.bind(element, this);
2759+
2760+            element.addEventListener('keydown', element.boundKeydown, false);
2761+            element.addEventListener('keyup', element.boundKeyup, false);
2762+            element.addEventListener('input', element.boundInput, false);
2763+        }
2764+    }, {
2765+        key: 'unbind',
2766+        value: function unbind(element) {
2767+            element.removeEventListener('keydown', element.boundKeydown, false);
2768+            element.removeEventListener('keyup', element.boundKeyup, false);
2769+            element.removeEventListener('input', element.boundInput, false);
2770+
2771+            delete element.boundKeydown;
2772+            delete element.boundKeyup;
2773+            delete element.boundInput;
2774+        }
2775+    }, {
2776+        key: 'keydown',
2777+        value: function keydown(instance, event) {
2778+            if (instance.shouldDeactivate(event)) {
2779+                instance.tribute.isActive = false;
2780+                instance.tribute.hideMenu();
2781+            }
2782+
2783+            var element = this;
2784+            instance.commandEvent = false;
2785+
2786+            TributeEvents.keys().forEach(function (o) {
2787+                if (o.key === event.keyCode) {
2788+                    instance.commandEvent = true;
2789+                    instance.callbacks()[o.value.toLowerCase()](event, element);
2790+                }
2791+            });
2792+        }
2793+    }, {
2794+        key: 'input',
2795+        value: function input(instance, event) {
2796+            instance.inputEvent = true;
2797+            instance.keyup.call(this, instance, event);
2798+        }
2799+    }, {
2800+        key: 'click',
2801+        value: function click(instance, event) {
2802+            var tribute = instance.tribute;
2803+            if (tribute.menu && tribute.menu.contains(event.target)) {
2804+                var li = event.target;
2805+                event.preventDefault();
2806+                event.stopPropagation();
2807+                while (li.nodeName.toLowerCase() !== 'li') {
2808+                    li = li.parentNode;
2809+                    if (!li || li === tribute.menu) {
2810+                        throw new Error('cannot find the <li> container for the click');
2811+                    }
2812+                }
2813+                tribute.selectItemAtIndex(li.getAttribute('data-index'), event);
2814+                tribute.hideMenu();
2815+
2816+                // TODO: should fire with externalTrigger and target is outside of menu
2817+            } else if (tribute.current.element && !tribute.current.externalTrigger) {
2818+                tribute.current.externalTrigger = false;
2819+                setTimeout(function () {
2820+                    return tribute.hideMenu();
2821+                });
2822+            }
2823+        }
2824+    }, {
2825+        key: 'keyup',
2826+        value: function keyup(instance, event) {
2827+            if (instance.inputEvent) {
2828+                instance.inputEvent = false;
2829+            }
2830+            instance.updateSelection(this);
2831+
2832+            if (event.keyCode === 27) return;
2833+
2834+            if (!instance.tribute.allowSpaces && instance.tribute.hasTrailingSpace) {
2835+                instance.tribute.hasTrailingSpace = false;
2836+                instance.commandEvent = true;
2837+                instance.callbacks()["space"](event, this);
2838+                return;
2839+            }
2840+
2841+            if (!instance.tribute.isActive) {
2842+                var keyCode = instance.getKeyCode(instance, this, event);
2843+
2844+                if (isNaN(keyCode) || !keyCode) return;
2845+
2846+                var trigger = instance.tribute.triggers().find(function (trigger) {
2847+                    return trigger.charCodeAt(0) === keyCode;
2848+                });
2849+
2850+                if (typeof trigger !== 'undefined') {
2851+                    instance.callbacks().triggerChar(event, this, trigger);
2852+                }
2853+            }
2854+
2855+            if (instance.tribute.current.trigger && instance.commandEvent === false || instance.tribute.isActive && event.keyCode === 8) {
2856+                instance.tribute.showMenuFor(this, true);
2857+            }
2858+        }
2859+    }, {
2860+        key: 'shouldDeactivate',
2861+        value: function shouldDeactivate(event) {
2862+            if (!this.tribute.isActive) return false;
2863+
2864+            if (this.tribute.current.mentionText.length === 0) {
2865+                var eventKeyPressed = false;
2866+                TributeEvents.keys().forEach(function (o) {
2867+                    if (event.keyCode === o.key) eventKeyPressed = true;
2868+                });
2869+
2870+                return !eventKeyPressed;
2871+            }
2872+
2873+            return false;
2874+        }
2875+    }, {
2876+        key: 'getKeyCode',
2877+        value: function getKeyCode(instance, el, event) {
2878+            var char = void 0;
2879+            var tribute = instance.tribute;
2880+            var info = tribute.range.getTriggerInfo(false, tribute.hasTrailingSpace, true, tribute.allowSpaces);
2881+
2882+            if (info) {
2883+                return info.mentionTriggerChar.charCodeAt(0);
2884+            } else {
2885+                return false;
2886+            }
2887+        }
2888+    }, {
2889+        key: 'updateSelection',
2890+        value: function updateSelection(el) {
2891+            this.tribute.current.element = el;
2892+            var info = this.tribute.range.getTriggerInfo(false, this.tribute.hasTrailingSpace, true, this.tribute.allowSpaces);
2893+
2894+            if (info) {
2895+                this.tribute.current.selectedPath = info.mentionSelectedPath;
2896+                this.tribute.current.mentionText = info.mentionText;
2897+                this.tribute.current.selectedOffset = info.mentionSelectedOffset;
2898+            }
2899+        }
2900+    }, {
2901+        key: 'callbacks',
2902+        value: function callbacks() {
2903+            var _this = this;
2904+
2905+            return {
2906+                triggerChar: function triggerChar(e, el, trigger) {
2907+                    var tribute = _this.tribute;
2908+                    tribute.current.trigger = trigger;
2909+
2910+                    var collectionItem = tribute.collection.find(function (item) {
2911+                        return item.trigger === trigger;
2912+                    });
2913+
2914+                    tribute.current.collection = collectionItem;
2915+                    if (tribute.inputEvent) tribute.showMenuFor(el, true);
2916+                },
2917+                enter: function enter(e, el) {
2918+                    // choose selection
2919+                    if (_this.tribute.isActive) {
2920+                        e.preventDefault();
2921+                        e.stopPropagation();
2922+                        setTimeout(function () {
2923+                            _this.tribute.selectItemAtIndex(_this.tribute.menuSelected, e);
2924+                            _this.tribute.hideMenu();
2925+                        }, 0);
2926+                    }
2927+                },
2928+                escape: function escape(e, el) {
2929+                    if (_this.tribute.isActive) {
2930+                        e.preventDefault();
2931+                        e.stopPropagation();
2932+                        _this.tribute.isActive = false;
2933+                        _this.tribute.hideMenu();
2934+                    }
2935+                },
2936+                tab: function tab(e, el) {
2937+                    // choose first match
2938+                    _this.callbacks().enter(e, el);
2939+                },
2940+                space: function space(e, el) {
2941+                    if (_this.tribute.isActive) {
2942+                        if (_this.tribute.spaceSelectsMatch) {
2943+                            _this.callbacks().enter(e, el);
2944+                        } else if (!_this.tribute.allowSpaces) {
2945+                            e.stopPropagation();
2946+                            setTimeout(function () {
2947+                                _this.tribute.hideMenu();
2948+                                _this.tribute.isActive = false;
2949+                            }, 0);
2950+                        }
2951+                    }
2952+                },
2953+                up: function up(e, el) {
2954+                    // navigate up ul
2955+                    if (_this.tribute.isActive) {
2956+                        e.preventDefault();
2957+                        e.stopPropagation();
2958+                        var count = _this.tribute.current.filteredItems.length,
2959+                            selected = _this.tribute.menuSelected;
2960+
2961+                        if (count > selected && selected > 0) {
2962+                            _this.tribute.menuSelected--;
2963+                            _this.setActiveLi();
2964+                        } else if (selected === 0) {
2965+                            _this.tribute.menuSelected = count - 1;
2966+                            _this.setActiveLi();
2967+                            _this.tribute.menu.scrollTop = _this.tribute.menu.scrollHeight;
2968+                        }
2969+                    }
2970+                },
2971+                down: function down(e, el) {
2972+                    // navigate down ul
2973+                    if (_this.tribute.isActive) {
2974+                        e.preventDefault();
2975+                        e.stopPropagation();
2976+                        var count = _this.tribute.current.filteredItems.length - 1,
2977+                            selected = _this.tribute.menuSelected;
2978+
2979+                        if (count > selected) {
2980+                            _this.tribute.menuSelected++;
2981+                            _this.setActiveLi();
2982+                        } else if (count === selected) {
2983+                            _this.tribute.menuSelected = 0;
2984+                            _this.setActiveLi();
2985+                            _this.tribute.menu.scrollTop = 0;
2986+                        }
2987+                    }
2988+                },
2989+                delete: function _delete(e, el) {
2990+                    if (_this.tribute.isActive && _this.tribute.current.mentionText.length < 1) {
2991+                        _this.tribute.hideMenu();
2992+                    } else if (_this.tribute.isActive) {
2993+                        _this.tribute.showMenuFor(el);
2994+                    }
2995+                }
2996+            };
2997+        }
2998+    }, {
2999+        key: 'setActiveLi',
3000+        value: function setActiveLi(index) {
3001+            var lis = this.tribute.menu.querySelectorAll('li'),
3002+                length = lis.length >>> 0;
3003+
3004+            // get heights
3005+            var menuFullHeight = this.getFullHeight(this.tribute.menu),
3006+                liHeight = this.getFullHeight(lis[0]);
3007+
3008+            if (index) this.tribute.menuSelected = index;
3009+
3010+            for (var i = 0; i < length; i++) {
3011+                var li = lis[i];
3012+                if (i === this.tribute.menuSelected) {
3013+                    var offset = liHeight * (i + 1);
3014+                    var scrollTop = this.tribute.menu.scrollTop;
3015+                    var totalScroll = scrollTop + menuFullHeight;
3016+
3017+                    if (offset > totalScroll) {
3018+                        this.tribute.menu.scrollTop += liHeight;
3019+                    } else if (offset < totalScroll) {
3020+                        this.tribute.menu.scrollTop -= liHeight;
3021+                    }
3022+
3023+                    li.className = this.tribute.current.collection.selectClass;
3024+                } else {
3025+                    li.className = '';
3026+                }
3027+            }
3028+        }
3029+    }, {
3030+        key: 'getFullHeight',
3031+        value: function getFullHeight(elem, includeMargin) {
3032+            var height = elem.getBoundingClientRect().height;
3033+
3034+            if (includeMargin) {
3035+                var style = elem.currentStyle || window.getComputedStyle(elem);
3036+                return height + parseFloat(style.marginTop) + parseFloat(style.marginBottom);
3037+            }
3038+
3039+            return height;
3040+        }
3041+    }], [{
3042+        key: 'keys',
3043+        value: function keys() {
3044+            return [{
3045+                key: 9,
3046+                value: 'TAB'
3047+            }, {
3048+                key: 8,
3049+                value: 'DELETE'
3050+            }, {
3051+                key: 13,
3052+                value: 'ENTER'
3053+            }, {
3054+                key: 27,
3055+                value: 'ESCAPE'
3056+            }, {
3057+                key: 32,
3058+                value: 'SPACE'
3059+            }, {
3060+                key: 38,
3061+                value: 'UP'
3062+            }, {
3063+                key: 40,
3064+                value: 'DOWN'
3065+            }];
3066+        }
3067+    }]);
3068+
3069+    return TributeEvents;
3070+}();
3071+
3072+exports.default = TributeEvents;
3073+module.exports = exports['default'];
3074+
3075+},{}],3:[function(require,module,exports){
3076+'use strict';
3077+
3078+Object.defineProperty(exports, "__esModule", {
3079+    value: true
3080+});
3081+
3082+var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
3083+
3084+function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
3085+
3086+var TributeMenuEvents = function () {
3087+    function TributeMenuEvents(tribute) {
3088+        _classCallCheck(this, TributeMenuEvents);
3089+
3090+        this.tribute = tribute;
3091+        this.tribute.menuEvents = this;
3092+        this.menu = this.tribute.menu;
3093+    }
3094+
3095+    _createClass(TributeMenuEvents, [{
3096+        key: 'bind',
3097+        value: function bind(menu) {
3098+            var _this = this;
3099+
3100+            menu.menuKeydownEvent = this.tribute.events.keydown.bind(this.menu, this);
3101+            this.menuClickEvent = this.tribute.events.click.bind(null, this);
3102+            this.menuContainerScrollEvent = this.debounce(function () {
3103+                if (_this.tribute.isActive) {
3104+                    _this.tribute.showMenuFor(_this.tribute.current.element, false);
3105+                }
3106+            }, 300, false);
3107+            this.windowResizeEvent = this.debounce(function () {
3108+                if (_this.tribute.isActive) {
3109+                    _this.tribute.range.positionMenuAtCaret(true);
3110+                }
3111+            }, 300, false);
3112+
3113+            // fixes IE11 issues with mouseup
3114+            this.tribute.range.getDocument().addEventListener('MSPointerUp', this.menuClickEvent, false);
3115+            menu.addEventListener('keydown', this.menuKeydownEvent, false);
3116+            this.tribute.range.getDocument().addEventListener('mouseup', this.menuClickEvent, false);
3117+            window.addEventListener('resize', this.windowResizeEvent);
3118+
3119+            if (this.menuContainer) {
3120+                this.menuContainer.addEventListener('scroll', this.menuContainerScrollEvent, false);
3121+            } else {
3122+                window.addEventListener('scroll', this.menuContainerScrollEvent);
3123+            }
3124+        }
3125+    }, {
3126+        key: 'unbind',
3127+        value: function unbind(menu) {
3128+            menu.removeEventListener('keydown', menu.menuKeydownEvent, false);
3129+            delete menu.menuKeydownEvent;
3130+            this.tribute.range.getDocument().removeEventListener('mouseup', this.menuClickEvent, false);
3131+            this.tribute.range.getDocument().removeEventListener('MSPointerUp', this.menuClickEvent, false);
3132+            window.removeEventListener('resize', this.windowResizeEvent);
3133+
3134+            if (this.menuContainer) {
3135+                this.menuContainer.removeEventListener('scroll', this.menuContainerScrollEvent, false);
3136+            } else {
3137+                window.removeEventListener('scroll', this.menuContainerScrollEvent);
3138+            }
3139+        }
3140+    }, {
3141+        key: 'debounce',
3142+        value: function debounce(func, wait, immediate) {
3143+            var _this2 = this,
3144+                _arguments = arguments;
3145+
3146+            var timeout;
3147+            return function () {
3148+                var context = _this2,
3149+                    args = _arguments;
3150+                var later = function later() {
3151+                    timeout = null;
3152+                    if (!immediate) func.apply(context, args);
3153+                };
3154+                var callNow = immediate && !timeout;
3155+                clearTimeout(timeout);
3156+                timeout = setTimeout(later, wait);
3157+                if (callNow) func.apply(context, args);
3158+            };
3159+        }
3160+    }]);
3161+
3162+    return TributeMenuEvents;
3163+}();
3164+
3165+exports.default = TributeMenuEvents;
3166+module.exports = exports['default'];
3167+
3168+},{}],4:[function(require,module,exports){
3169+'use strict';
3170+
3171+Object.defineProperty(exports, "__esModule", {
3172+    value: true
3173+});
3174+
3175+var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
3176+
3177+function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
3178+
3179+// Thanks to https://github.com/jeff-collins/ment.io
3180+var TributeRange = function () {
3181+    function TributeRange(tribute) {
3182+        _classCallCheck(this, TributeRange);
3183+
3184+        this.tribute = tribute;
3185+        this.tribute.range = this;
3186+    }
3187+
3188+    _createClass(TributeRange, [{
3189+        key: 'getDocument',
3190+        value: function getDocument() {
3191+            var iframe = void 0;
3192+            if (this.tribute.current.collection) {
3193+                iframe = this.tribute.current.collection.iframe;
3194+            }
3195+
3196+            if (!iframe) {
3197+                return document;
3198+            }
3199+
3200+            return iframe.contentWindow.document;
3201+        }
3202+    }, {
3203+        key: 'positionMenuAtCaret',
3204+        value: function positionMenuAtCaret(scrollTo) {
3205+            var _this = this;
3206+
3207+            var context = this.tribute.current,
3208+                coordinates = void 0;
3209+
3210+            var info = this.getTriggerInfo(false, this.tribute.hasTrailingSpace, true, this.tribute.allowSpaces);
3211+
3212+            if (typeof info !== 'undefined') {
3213+
3214+                if (!this.tribute.positionMenu) {
3215+                    this.tribute.menu.style.cssText = 'display: block;';
3216+                    return;
3217+                }
3218+
3219+                if (!this.isContentEditable(context.element)) {
3220+                    coordinates = this.getTextAreaOrInputUnderlinePosition(this.tribute.current.element, info.mentionPosition);
3221+                } else {
3222+                    coordinates = this.getContentEditableCaretPosition(info.mentionPosition);
3223+                }
3224+
3225+                this.tribute.menu.style.cssText = 'top: ' + coordinates.top + 'px;\n                                     left: ' + coordinates.left + 'px;\n                                     right: ' + coordinates.right + 'px;\n                                     bottom: ' + coordinates.bottom + 'px;\n                                     position: absolute;\n                                     zIndex: 10000;\n                                     display: block;';
3226+
3227+                if (coordinates.left === 'auto') {
3228+                    this.tribute.menu.style.left = 'auto';
3229+                }
3230+
3231+                if (coordinates.top === 'auto') {
3232+                    this.tribute.menu.style.top = 'auto';
3233+                }
3234+
3235+                if (scrollTo) this.scrollIntoView();
3236+
3237+                window.setTimeout(function () {
3238+                    var menuDimensions = {
3239+                        width: _this.tribute.menu.offsetWidth,
3240+                        height: _this.tribute.menu.offsetHeight
3241+                    };
3242+                    var menuIsOffScreen = _this.isMenuOffScreen(coordinates, menuDimensions);
3243+
3244+                    var menuIsOffScreenHorizontally = window.innerWidth > menuDimensions.width && (menuIsOffScreen.left || menuIsOffScreen.right);
3245+                    var menuIsOffScreenVertically = window.innerHeight > menuDimensions.height && (menuIsOffScreen.top || menuIsOffScreen.bottom);
3246+                    if (menuIsOffScreenHorizontally || menuIsOffScreenVertically) {
3247+                        _this.tribute.menu.style.cssText = 'display: none';
3248+                        _this.positionMenuAtCaret(scrollTo);
3249+                    }
3250+                }, 0);
3251+            } else {
3252+                this.tribute.menu.style.cssText = 'display: none';
3253+            }
3254+        }
3255+    }, {
3256+        key: 'selectElement',
3257+        value: function selectElement(targetElement, path, offset) {
3258+            var range = void 0;
3259+            var elem = targetElement;
3260+
3261+            if (path) {
3262+                for (var i = 0; i < path.length; i++) {
3263+                    elem = elem.childNodes[path[i]];
3264+                    if (elem === undefined) {
3265+                        return;
3266+                    }
3267+                    while (elem.length < offset) {
3268+                        offset -= elem.length;
3269+                        elem = elem.nextSibling;
3270+                    }
3271+                    if (elem.childNodes.length === 0 && !elem.length) {
3272+                        elem = elem.previousSibling;
3273+                    }
3274+                }
3275+            }
3276+            var sel = this.getWindowSelection();
3277+
3278+            range = this.getDocument().createRange();
3279+            range.setStart(elem, offset);
3280+            range.setEnd(elem, offset);
3281+            range.collapse(true);
3282+
3283+            try {
3284+                sel.removeAllRanges();
3285+            } catch (error) {}
3286+
3287+            sel.addRange(range);
3288+            targetElement.focus();
3289+        }
3290+    }, {
3291+        key: 'resetSelection',
3292+        value: function resetSelection(targetElement, path, offset) {
3293+            if (!this.isContentEditable(targetElement)) {
3294+                if (targetElement !== this.tribute.current.element) {
3295+                    targetElement.focus();
3296+                }
3297+            } else {
3298+                this.selectElement(targetElement, path, offset);
3299+            }
3300+        }
3301+    }, {
3302+        key: 'replaceTriggerText',
3303+        value: function replaceTriggerText(text, requireLeadingSpace, hasTrailingSpace, originalEvent, item) {
3304+            var context = this.tribute.current;
3305+            this.resetSelection(context.element, context.selectedPath, context.selectedOffset);
3306+
3307+            var info = this.getTriggerInfo(true, hasTrailingSpace, requireLeadingSpace, this.tribute.allowSpaces);
3308+
3309+            // Create the event
3310+            var replaceEvent = new CustomEvent('tribute-replaced', {
3311+                detail: {
3312+                    item: item,
3313+                    event: originalEvent
3314+                }
3315+            });
3316+
3317+            if (info !== undefined) {
3318+                if (!this.isContentEditable(context.element)) {
3319+                    var myField = this.tribute.current.element;
3320+                    var textSuffix = typeof this.tribute.replaceTextSuffix == 'string' ? this.tribute.replaceTextSuffix : ' ';
3321+                    text += textSuffix;
3322+                    var startPos = info.mentionPosition;
3323+                    var endPos = info.mentionPosition + info.mentionText.length + textSuffix.length;
3324+                    myField.value = myField.value.substring(0, startPos) + text + myField.value.substring(endPos, myField.value.length);
3325+                    myField.selectionStart = startPos + text.length;
3326+                    myField.selectionEnd = startPos + text.length;
3327+                } else {
3328+                    // add a space to the end of the pasted text
3329+                    var _textSuffix = typeof this.tribute.replaceTextSuffix == 'string' ? this.tribute.replaceTextSuffix : '\xA0';
3330+                    text += _textSuffix;
3331+                    this.pasteHtml(text, info.mentionPosition, info.mentionPosition + info.mentionText.length + 1);
3332+                }
3333+
3334+                context.element.dispatchEvent(replaceEvent);
3335+            }
3336+        }
3337+    }, {
3338+        key: 'pasteHtml',
3339+        value: function pasteHtml(html, startPos, endPos) {
3340+            var range = void 0,
3341+                sel = void 0;
3342+            sel = this.getWindowSelection();
3343+            range = this.getDocument().createRange();
3344+            range.setStart(sel.anchorNode, startPos);
3345+            range.setEnd(sel.anchorNode, endPos);
3346+            range.deleteContents();
3347+
3348+            var el = this.getDocument().createElement('div');
3349+            el.innerHTML = html;
3350+            var frag = this.getDocument().createDocumentFragment(),
3351+                node = void 0,
3352+                lastNode = void 0;
3353+            while (node = el.firstChild) {
3354+                lastNode = frag.appendChild(node);
3355+            }
3356+            range.insertNode(frag);
3357+
3358+            // Preserve the selection
3359+            if (lastNode) {
3360+                range = range.cloneRange();
3361+                range.setStartAfter(lastNode);
3362+                range.collapse(true);
3363+                sel.removeAllRanges();
3364+                sel.addRange(range);
3365+            }
3366+        }
3367+    }, {
3368+        key: 'getWindowSelection',
3369+        value: function getWindowSelection() {
3370+            if (this.tribute.collection.iframe) {
3371+                return this.tribute.collection.iframe.contentWindow.getSelection();
3372+            }
3373+
3374+            return window.getSelection();
3375+        }
3376+    }, {
3377+        key: 'getNodePositionInParent',
3378+        value: function getNodePositionInParent(element) {
3379+            if (element.parentNode === null) {
3380+                return 0;
3381+            }
3382+
3383+            for (var i = 0; i < element.parentNode.childNodes.length; i++) {
3384+                var node = element.parentNode.childNodes[i];
3385+
3386+                if (node === element) {
3387+                    return i;
3388+                }
3389+            }
3390+        }
3391+    }, {
3392+        key: 'getContentEditableSelectedPath',
3393+        value: function getContentEditableSelectedPath(ctx) {
3394+            var sel = this.getWindowSelection();
3395+            var selected = sel.anchorNode;
3396+            var path = [];
3397+            var offset = void 0;
3398+
3399+            if (selected != null) {
3400+                var i = void 0;
3401+                var ce = selected.contentEditable;
3402+                while (selected !== null && ce !== 'true') {
3403+                    i = this.getNodePositionInParent(selected);
3404+                    path.push(i);
3405+                    selected = selected.parentNode;
3406+                    if (selected !== null) {
3407+                        ce = selected.contentEditable;
3408+                    }
3409+                }
3410+                path.reverse();
3411+
3412+                // getRangeAt may not exist, need alternative
3413+                offset = sel.getRangeAt(0).startOffset;
3414+
3415+                return {
3416+                    selected: selected,
3417+                    path: path,
3418+                    offset: offset
3419+                };
3420+            }
3421+        }
3422+    }, {
3423+        key: 'getTextPrecedingCurrentSelection',
3424+        value: function getTextPrecedingCurrentSelection() {
3425+            var context = this.tribute.current,
3426+                text = '';
3427+
3428+            if (!this.isContentEditable(context.element)) {
3429+                var textComponent = this.tribute.current.element;
3430+                if (textComponent) {
3431+                    var startPos = textComponent.selectionStart;
3432+                    if (textComponent.value && startPos >= 0) {
3433+                        text = textComponent.value.substring(0, startPos);
3434+                    }
3435+                }
3436+            } else {
3437+                var selectedElem = this.getWindowSelection().anchorNode;
3438+
3439+                if (selectedElem != null) {
3440+                    var workingNodeContent = selectedElem.textContent;
3441+                    var selectStartOffset = this.getWindowSelection().getRangeAt(0).startOffset;
3442+
3443+                    if (workingNodeContent && selectStartOffset >= 0) {
3444+                        text = workingNodeContent.substring(0, selectStartOffset);
3445+                    }
3446+                }
3447+            }
3448+
3449+            return text;
3450+        }
3451+    }, {
3452+        key: 'getTriggerInfo',
3453+        value: function getTriggerInfo(menuAlreadyActive, hasTrailingSpace, requireLeadingSpace, allowSpaces) {
3454+            var _this2 = this;
3455+
3456+            var ctx = this.tribute.current;
3457+            var selected = void 0,
3458+                path = void 0,
3459+                offset = void 0;
3460+
3461+            if (!this.isContentEditable(ctx.element)) {
3462+                selected = this.tribute.current.element;
3463+            } else {
3464+                var selectionInfo = this.getContentEditableSelectedPath(ctx);
3465+
3466+                if (selectionInfo) {
3467+                    selected = selectionInfo.selected;
3468+                    path = selectionInfo.path;
3469+                    offset = selectionInfo.offset;
3470+                }
3471+            }
3472+
3473+            var effectiveRange = this.getTextPrecedingCurrentSelection();
3474+
3475+            if (effectiveRange !== undefined && effectiveRange !== null) {
3476+                var mostRecentTriggerCharPos = -1;
3477+                var triggerChar = void 0;
3478+
3479+                this.tribute.collection.forEach(function (config) {
3480+                    var c = config.trigger;
3481+                    var idx = config.requireLeadingSpace ? _this2.lastIndexWithLeadingSpace(effectiveRange, c) : effectiveRange.lastIndexOf(c);
3482+
3483+                    if (idx > mostRecentTriggerCharPos) {
3484+                        mostRecentTriggerCharPos = idx;
3485+                        triggerChar = c;
3486+                        requireLeadingSpace = config.requireLeadingSpace;
3487+                    }
3488+                });
3489+
3490+                if (mostRecentTriggerCharPos >= 0 && (mostRecentTriggerCharPos === 0 || !requireLeadingSpace || /[\xA0\s]/g.test(effectiveRange.substring(mostRecentTriggerCharPos - 1, mostRecentTriggerCharPos)))) {
3491+                    var currentTriggerSnippet = effectiveRange.substring(mostRecentTriggerCharPos + 1, effectiveRange.length);
3492+
3493+                    triggerChar = effectiveRange.substring(mostRecentTriggerCharPos, mostRecentTriggerCharPos + 1);
3494+                    var firstSnippetChar = currentTriggerSnippet.substring(0, 1);
3495+                    var leadingSpace = currentTriggerSnippet.length > 0 && (firstSnippetChar === ' ' || firstSnippetChar === '\xA0');
3496+                    if (hasTrailingSpace) {
3497+                        currentTriggerSnippet = currentTriggerSnippet.trim();
3498+                    }
3499+
3500+                    var regex = allowSpaces ? /[^\S ]/g : /[\xA0\s]/g;
3501+
3502+                    this.tribute.hasTrailingSpace = regex.test(currentTriggerSnippet);
3503+
3504+                    if (!leadingSpace && (menuAlreadyActive || !regex.test(currentTriggerSnippet))) {
3505+                        return {
3506+                            mentionPosition: mostRecentTriggerCharPos,
3507+                            mentionText: currentTriggerSnippet,
3508+                            mentionSelectedElement: selected,
3509+                            mentionSelectedPath: path,
3510+                            mentionSelectedOffset: offset,
3511+                            mentionTriggerChar: triggerChar
3512+                        };
3513+                    }
3514+                }
3515+            }
3516+        }
3517+    }, {
3518+        key: 'lastIndexWithLeadingSpace',
3519+        value: function lastIndexWithLeadingSpace(str, char) {
3520+            var reversedStr = str.split('').reverse().join('');
3521+            var index = -1;
3522+
3523+            for (var cidx = 0, len = str.length; cidx < len; cidx++) {
3524+                var firstChar = cidx === str.length - 1;
3525+                var leadingSpace = /\s/.test(reversedStr[cidx + 1]);
3526+                var match = char === reversedStr[cidx];
3527+
3528+                if (match && (firstChar || leadingSpace)) {
3529+                    index = str.length - 1 - cidx;
3530+                    break;
3531+                }
3532+            }
3533+
3534+            return index;
3535+        }
3536+    }, {
3537+        key: 'isContentEditable',
3538+        value: function isContentEditable(element) {
3539+            return element.nodeName !== 'INPUT' && element.nodeName !== 'TEXTAREA';
3540+        }
3541+    }, {
3542+        key: 'isMenuOffScreen',
3543+        value: function isMenuOffScreen(coordinates, menuDimensions) {
3544+            var windowWidth = window.innerWidth;
3545+            var windowHeight = window.innerHeight;
3546+            var doc = document.documentElement;
3547+            var windowLeft = (window.pageXOffset || doc.scrollLeft) - (doc.clientLeft || 0);
3548+            var windowTop = (window.pageYOffset || doc.scrollTop) - (doc.clientTop || 0);
3549+
3550+            var menuTop = typeof coordinates.top === 'number' ? coordinates.top : windowTop + windowHeight - coordinates.bottom - menuDimensions.height;
3551+            var menuRight = typeof coordinates.right === 'number' ? coordinates.right : coordinates.left + menuDimensions.width;
3552+            var menuBottom = typeof coordinates.bottom === 'number' ? coordinates.bottom : coordinates.top + menuDimensions.height;
3553+            var menuLeft = typeof coordinates.left === 'number' ? coordinates.left : windowLeft + windowWidth - coordinates.right - menuDimensions.width;
3554+
3555+            return {
3556+                top: menuTop < Math.floor(windowTop),
3557+                right: menuRight > Math.ceil(windowLeft + windowWidth),
3558+                bottom: menuBottom > Math.ceil(windowTop + windowHeight),
3559+                left: menuLeft < Math.floor(windowLeft)
3560+            };
3561+        }
3562+    }, {
3563+        key: 'getMenuDimensions',
3564+        value: function getMenuDimensions() {
3565+            // Width of the menu depends of its contents and position
3566+            // We must check what its width would be without any obstruction
3567+            // This way, we can achieve good positioning for flipping the menu
3568+            var dimensions = {
3569+                width: null,
3570+                height: null
3571+            };
3572+
3573+            this.tribute.menu.style.cssText = 'top: 0px;\n                                 left: 0px;\n                                 position: fixed;\n                                 zIndex: 10000;\n                                 display: block;\n                                 visibility; hidden;';
3574+            dimensions.width = this.tribute.menu.offsetWidth;
3575+            dimensions.height = this.tribute.menu.offsetHeight;
3576+
3577+            this.tribute.menu.style.cssText = 'display: none;';
3578+
3579+            return dimensions;
3580+        }
3581+    }, {
3582+        key: 'getTextAreaOrInputUnderlinePosition',
3583+        value: function getTextAreaOrInputUnderlinePosition(element, position, flipped) {
3584+            var properties = ['direction', 'boxSizing', 'width', 'height', 'overflowX', 'overflowY', 'borderTopWidth', 'borderRightWidth', 'borderBottomWidth', 'borderLeftWidth', 'paddingTop', 'paddingRight', 'paddingBottom', 'paddingLeft', 'fontStyle', 'fontVariant', 'fontWeight', 'fontStretch', 'fontSize', 'fontSizeAdjust', 'lineHeight', 'fontFamily', 'textAlign', 'textTransform', 'textIndent', 'textDecoration', 'letterSpacing', 'wordSpacing'];
3585+
3586+            var isFirefox = window.mozInnerScreenX !== null;
3587+
3588+            var div = this.getDocument().createElement('div');
3589+            div.id = 'input-textarea-caret-position-mirror-div';
3590+            this.getDocument().body.appendChild(div);
3591+
3592+            var style = div.style;
3593+            var computed = window.getComputedStyle ? getComputedStyle(element) : element.currentStyle;
3594+
3595+            style.whiteSpace = 'pre-wrap';
3596+            if (element.nodeName !== 'INPUT') {
3597+                style.wordWrap = 'break-word';
3598+            }
3599+
3600+            // position off-screen
3601+            style.position = 'absolute';
3602+            style.visibility = 'hidden';
3603+
3604+            // transfer the element's properties to the div
3605+            properties.forEach(function (prop) {
3606+                style[prop] = computed[prop];
3607+            });
3608+
3609+            if (isFirefox) {
3610+                style.width = parseInt(computed.width) - 2 + 'px';
3611+                if (element.scrollHeight > parseInt(computed.height)) style.overflowY = 'scroll';
3612+            } else {
3613+                style.overflow = 'hidden';
3614+            }
3615+
3616+            div.textContent = element.value.substring(0, position);
3617+
3618+            if (element.nodeName === 'INPUT') {
3619+                div.textContent = div.textContent.replace(/\s/g, ' ');
3620+            }
3621+
3622+            var span = this.getDocument().createElement('span');
3623+            span.textContent = element.value.substring(position) || '.';
3624+            div.appendChild(span);
3625+
3626+            var rect = element.getBoundingClientRect();
3627+            var doc = document.documentElement;
3628+            var windowLeft = (window.pageXOffset || doc.scrollLeft) - (doc.clientLeft || 0);
3629+            var windowTop = (window.pageYOffset || doc.scrollTop) - (doc.clientTop || 0);
3630+
3631+            var coordinates = {
3632+                top: rect.top + windowTop + span.offsetTop + parseInt(computed.borderTopWidth) + parseInt(computed.fontSize) - element.scrollTop,
3633+                left: rect.left + windowLeft + span.offsetLeft + parseInt(computed.borderLeftWidth)
3634+            };
3635+
3636+            var windowWidth = window.innerWidth;
3637+            var windowHeight = window.innerHeight;
3638+
3639+            var menuDimensions = this.getMenuDimensions();
3640+            var menuIsOffScreen = this.isMenuOffScreen(coordinates, menuDimensions);
3641+
3642+            if (menuIsOffScreen.right) {
3643+                coordinates.right = windowWidth - coordinates.left;
3644+                coordinates.left = 'auto';
3645+            }
3646+
3647+            var parentHeight = this.tribute.menuContainer ? this.tribute.menuContainer.offsetHeight : this.getDocument().body.offsetHeight;
3648+
3649+            if (menuIsOffScreen.bottom) {
3650+                var parentRect = this.tribute.menuContainer ? this.tribute.menuContainer.getBoundingClientRect() : this.getDocument().body.getBoundingClientRect();
3651+                var scrollStillAvailable = parentHeight - (windowHeight - parentRect.top);
3652+
3653+                coordinates.bottom = scrollStillAvailable + (windowHeight - rect.top - span.offsetTop);
3654+                coordinates.top = 'auto';
3655+            }
3656+
3657+            menuIsOffScreen = this.isMenuOffScreen(coordinates, menuDimensions);
3658+            if (menuIsOffScreen.left) {
3659+                coordinates.left = windowWidth > menuDimensions.width ? windowLeft + windowWidth - menuDimensions.width : windowLeft;
3660+                delete coordinates.right;
3661+            }
3662+            if (menuIsOffScreen.top) {
3663+                coordinates.top = windowHeight > menuDimensions.height ? windowTop + windowHeight - menuDimensions.height : windowTop;
3664+                delete coordinates.bottom;
3665+            }
3666+
3667+            this.getDocument().body.removeChild(div);
3668+            return coordinates;
3669+        }
3670+    }, {
3671+        key: 'getContentEditableCaretPosition',
3672+        value: function getContentEditableCaretPosition(selectedNodePosition) {
3673+            var markerTextChar = '';
3674+            var markerEl = void 0,
3675+                markerId = 'sel_' + new Date().getTime() + '_' + Math.random().toString().substr(2);
3676+            var range = void 0;
3677+            var sel = this.getWindowSelection();
3678+            var prevRange = sel.getRangeAt(0);
3679+
3680+            range = this.getDocument().createRange();
3681+            range.setStart(sel.anchorNode, selectedNodePosition);
3682+            range.setEnd(sel.anchorNode, selectedNodePosition);
3683+
3684+            range.collapse(false);
3685+
3686+            // Create the marker element containing a single invisible character using DOM methods and insert it
3687+            markerEl = this.getDocument().createElement('span');
3688+            markerEl.id = markerId;
3689+
3690+            markerEl.appendChild(this.getDocument().createTextNode(markerTextChar));
3691+            range.insertNode(markerEl);
3692+            sel.removeAllRanges();
3693+            sel.addRange(prevRange);
3694+
3695+            var rect = markerEl.getBoundingClientRect();
3696+            var doc = document.documentElement;
3697+            var windowLeft = (window.pageXOffset || doc.scrollLeft) - (doc.clientLeft || 0);
3698+            var windowTop = (window.pageYOffset || doc.scrollTop) - (doc.clientTop || 0);
3699+            var coordinates = {
3700+                left: rect.left + windowLeft,
3701+                top: rect.top + markerEl.offsetHeight + windowTop
3702+            };
3703+            var windowWidth = window.innerWidth;
3704+            var windowHeight = window.innerHeight;
3705+
3706+            var menuDimensions = this.getMenuDimensions();
3707+            var menuIsOffScreen = this.isMenuOffScreen(coordinates, menuDimensions);
3708+
3709+            if (menuIsOffScreen.right) {
3710+                coordinates.left = 'auto';
3711+                coordinates.right = windowWidth - rect.left - windowLeft;
3712+            }
3713+
3714+            var parentHeight = this.tribute.menuContainer ? this.tribute.menuContainer.offsetHeight : this.getDocument().body.offsetHeight;
3715+
3716+            if (menuIsOffScreen.bottom) {
3717+                var parentRect = this.tribute.menuContainer ? this.tribute.menuContainer.getBoundingClientRect() : this.getDocument().body.getBoundingClientRect();
3718+                var scrollStillAvailable = parentHeight - (windowHeight - parentRect.top);
3719+
3720+                coordinates.top = 'auto';
3721+                coordinates.bottom = scrollStillAvailable + (windowHeight - rect.top);
3722+            }
3723+
3724+            menuIsOffScreen = this.isMenuOffScreen(coordinates, menuDimensions);
3725+            if (menuIsOffScreen.left) {
3726+                coordinates.left = windowWidth > menuDimensions.width ? windowLeft + windowWidth - menuDimensions.width : windowLeft;
3727+                delete coordinates.right;
3728+            }
3729+            if (menuIsOffScreen.top) {
3730+                coordinates.top = windowHeight > menuDimensions.height ? windowTop + windowHeight - menuDimensions.height : windowTop;
3731+                delete coordinates.bottom;
3732+            }
3733+
3734+            markerEl.parentNode.removeChild(markerEl);
3735+            return coordinates;
3736+        }
3737+    }, {
3738+        key: 'scrollIntoView',
3739+        value: function scrollIntoView(elem) {
3740+            var reasonableBuffer = 20,
3741+                clientRect = void 0;
3742+            var maxScrollDisplacement = 100;
3743+            var e = this.menu;
3744+
3745+            if (typeof e === 'undefined') return;
3746+
3747+            while (clientRect === undefined || clientRect.height === 0) {
3748+                clientRect = e.getBoundingClientRect();
3749+
3750+                if (clientRect.height === 0) {
3751+                    e = e.childNodes[0];
3752+                    if (e === undefined || !e.getBoundingClientRect) {
3753+                        return;
3754+                    }
3755+                }
3756+            }
3757+
3758+            var elemTop = clientRect.top;
3759+            var elemBottom = elemTop + clientRect.height;
3760+
3761+            if (elemTop < 0) {
3762+                window.scrollTo(0, window.pageYOffset + clientRect.top - reasonableBuffer);
3763+            } else if (elemBottom > window.innerHeight) {
3764+                var maxY = window.pageYOffset + clientRect.top - reasonableBuffer;
3765+
3766+                if (maxY - window.pageYOffset > maxScrollDisplacement) {
3767+                    maxY = window.pageYOffset + maxScrollDisplacement;
3768+                }
3769+
3770+                var targetY = window.pageYOffset - (window.innerHeight - elemBottom);
3771+
3772+                if (targetY > maxY) {
3773+                    targetY = maxY;
3774+                }
3775+
3776+                window.scrollTo(0, targetY);
3777+            }
3778+        }
3779+    }]);
3780+
3781+    return TributeRange;
3782+}();
3783+
3784+exports.default = TributeRange;
3785+module.exports = exports['default'];
3786+
3787+},{}],5:[function(require,module,exports){
3788+'use strict';
3789+
3790+Object.defineProperty(exports, "__esModule", {
3791+    value: true
3792+});
3793+
3794+var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
3795+
3796+function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
3797+
3798+// Thanks to https://github.com/mattyork/fuzzy
3799+var TributeSearch = function () {
3800+    function TributeSearch(tribute) {
3801+        _classCallCheck(this, TributeSearch);
3802+
3803+        this.tribute = tribute;
3804+        this.tribute.search = this;
3805+    }
3806+
3807+    _createClass(TributeSearch, [{
3808+        key: 'simpleFilter',
3809+        value: function simpleFilter(pattern, array) {
3810+            var _this = this;
3811+
3812+            return array.filter(function (string) {
3813+                return _this.test(pattern, string);
3814+            });
3815+        }
3816+    }, {
3817+        key: 'test',
3818+        value: function test(pattern, string) {
3819+            return this.match(pattern, string) !== null;
3820+        }
3821+    }, {
3822+        key: 'match',
3823+        value: function match(pattern, string, opts) {
3824+            opts = opts || {};
3825+            var patternIdx = 0,
3826+                result = [],
3827+                len = string.length,
3828+                totalScore = 0,
3829+                currScore = 0,
3830+                pre = opts.pre || '',
3831+                post = opts.post || '',
3832+                compareString = opts.caseSensitive && string || string.toLowerCase(),
3833+                ch = void 0,
3834+                compareChar = void 0;
3835+
3836+            pattern = opts.caseSensitive && pattern || pattern.toLowerCase();
3837+
3838+            var patternCache = this.traverse(compareString, pattern, 0, 0, []);
3839+            if (!patternCache) {
3840+                return null;
3841+            }
3842+
3843+            return {
3844+                rendered: this.render(string, patternCache.cache, pre, post),
3845+                score: patternCache.score
3846+            };
3847+        }
3848+    }, {
3849+        key: 'traverse',
3850+        value: function traverse(string, pattern, stringIndex, patternIndex, patternCache) {
3851+            // if the pattern search at end
3852+            if (pattern.length === patternIndex) {
3853+
3854+                // calculate score and copy the cache containing the indices where it's found
3855+                return {
3856+                    score: this.calculateScore(patternCache),
3857+                    cache: patternCache.slice()
3858+                };
3859+            }
3860+
3861+            // if string at end or remaining pattern > remaining string
3862+            if (string.length === stringIndex || pattern.length - patternIndex > string.length - stringIndex) {
3863+                return undefined;
3864+            }
3865+
3866+            var c = pattern[patternIndex];
3867+            var index = string.indexOf(c, stringIndex);
3868+            var best = void 0,
3869+                temp = void 0;
3870+
3871+            while (index > -1) {
3872+                patternCache.push(index);
3873+                temp = this.traverse(string, pattern, index + 1, patternIndex + 1, patternCache);
3874+                patternCache.pop();
3875+
3876+                // if downstream traversal failed, return best answer so far
3877+                if (!temp) {
3878+                    return best;
3879+                }
3880+
3881+                if (!best || best.score < temp.score) {
3882+                    best = temp;
3883+                }
3884+
3885+                index = string.indexOf(c, index + 1);
3886+            }
3887+
3888+            return best;
3889+        }
3890+    }, {
3891+        key: 'calculateScore',
3892+        value: function calculateScore(patternCache) {
3893+            var score = 0;
3894+            var temp = 1;
3895+
3896+            patternCache.forEach(function (index, i) {
3897+                if (i > 0) {
3898+                    if (patternCache[i - 1] + 1 === index) {
3899+                        temp += temp + 1;
3900+                    } else {
3901+                        temp = 1;
3902+                    }
3903+                }
3904+
3905+                score += temp;
3906+            });
3907+
3908+            return score;
3909+        }
3910+    }, {
3911+        key: 'render',
3912+        value: function render(string, indices, pre, post) {
3913+            var rendered = string.substring(0, indices[0]);
3914+
3915+            indices.forEach(function (index, i) {
3916+                rendered += pre + string[index] + post + string.substring(index + 1, indices[i + 1] ? indices[i + 1] : string.length);
3917+            });
3918+
3919+            return rendered;
3920+        }
3921+    }, {
3922+        key: 'filter',
3923+        value: function filter(pattern, arr, opts) {
3924+            var _this2 = this;
3925+
3926+            opts = opts || {};
3927+            return arr.reduce(function (prev, element, idx, arr) {
3928+                var str = element;
3929+
3930+                if (opts.extract) {
3931+                    str = opts.extract(element);
3932+
3933+                    if (!str) {
3934+                        // take care of undefineds / nulls / etc.
3935+                        str = '';
3936+                    }
3937+                }
3938+
3939+                var rendered = _this2.match(pattern, str, opts);
3940+
3941+                if (rendered != null) {
3942+                    prev[prev.length] = {
3943+                        string: rendered.rendered,
3944+                        score: rendered.score,
3945+                        index: idx,
3946+                        original: element
3947+                    };
3948+                }
3949+
3950+                return prev;
3951+            }, []).sort(function (a, b) {
3952+                var compare = b.score - a.score;
3953+                if (compare) return compare;
3954+                return a.index - b.index;
3955+            });
3956+        }
3957+    }]);
3958+
3959+    return TributeSearch;
3960+}();
3961+
3962+exports.default = TributeSearch;
3963+module.exports = exports['default'];
3964+
3965+},{}],6:[function(require,module,exports){
3966+"use strict";
3967+
3968+Object.defineProperty(exports, "__esModule", {
3969+  value: true
3970+});
3971+
3972+var _Tribute = require("./Tribute");
3973+
3974+var _Tribute2 = _interopRequireDefault(_Tribute);
3975+
3976+function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
3977+
3978+exports.default = _Tribute2.default; /**
3979+                                     * Tribute.js
3980+                                     * Native ES6 JavaScript @mention Plugin
3981+                                     **/
3982+
3983+module.exports = exports["default"];
3984+
3985+},{"./Tribute":1}],7:[function(require,module,exports){
3986+'use strict';
3987+
3988+if (!Array.prototype.find) {
3989+    Array.prototype.find = function (predicate) {
3990+        if (this === null) {
3991+            throw new TypeError('Array.prototype.find called on null or undefined');
3992+        }
3993+        if (typeof predicate !== 'function') {
3994+            throw new TypeError('predicate must be a function');
3995+        }
3996+        var list = Object(this);
3997+        var length = list.length >>> 0;
3998+        var thisArg = arguments[1];
3999+        var value;
4000+
4001+        for (var i = 0; i < length; i++) {
4002+            value = list[i];
4003+            if (predicate.call(thisArg, value, i, list)) {
4004+                return value;
4005+            }
4006+        }
4007+        return undefined;
4008+    };
4009+}
4010+
4011+if (window && typeof window.CustomEvent !== "function") {
4012+    var CustomEvent = function CustomEvent(event, params) {
4013+        params = params || {
4014+            bubbles: false,
4015+            cancelable: false,
4016+            detail: undefined
4017+        };
4018+        var evt = document.createEvent('CustomEvent');
4019+        evt.initCustomEvent(event, params.bubbles, params.cancelable, params.detail);
4020+        return evt;
4021+    };
4022+
4023+    if (typeof window.Event !== 'undefined') {
4024+        CustomEvent.prototype = window.Event.prototype;
4025+    }
4026+
4027+    window.CustomEvent = CustomEvent;
4028+}
4029+
4030+},{}]},{},[6])(6)
4031+});
4032+
4033+//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","src/Tribute.js","src/TributeEvents.js","src/TributeMenuEvents.js","src/TributeRange.js","src/TributeSearch.js","src/index.js","src/utils.js"],"names":[],"mappings":"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","file":"generated.js","sourceRoot":"","sourcesContent":["(function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c=\"function\"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error(\"Cannot find module '\"+i+\"'\");throw a.code=\"MODULE_NOT_FOUND\",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u=\"function\"==typeof require&&require,i=0;i<t.length;i++)o(t[i]);return o}return r})()","import TributeUtils from \"./utils\";\nimport TributeEvents from \"./TributeEvents\";\nimport TributeMenuEvents from \"./TributeMenuEvents\";\nimport TributeRange from \"./TributeRange\";\nimport TributeSearch from \"./TributeSearch\";\n\nclass Tribute {\n    constructor({\n        values = null,\n        iframe = null,\n        selectClass = 'highlight',\n        trigger = '@',\n        selectTemplate = null,\n        menuItemTemplate = null,\n        lookup = 'key',\n        fillAttr = 'value',\n        collection = null,\n        menuContainer = null,\n        noMatchTemplate = null,\n        requireLeadingSpace = true,\n        allowSpaces = false,\n        replaceTextSuffix = null,\n        positionMenu = true,\n        spaceSelectsMatch = false,\n    }) {\n\n        this.menuSelected = 0\n        this.current = {}\n        this.inputEvent = false\n        this.isActive = false\n        this.menuContainer = menuContainer\n        this.allowSpaces = allowSpaces\n        this.replaceTextSuffix = replaceTextSuffix\n        this.positionMenu = positionMenu\n        this.hasTrailingSpace = false;\n        this.spaceSelectsMatch = spaceSelectsMatch;\n\n        if (values) {\n            this.collection = [{\n                // symbol that starts the lookup\n                trigger: trigger,\n\n                // is it wrapped in an iframe\n                iframe: iframe,\n\n                // class applied to selected item\n                selectClass: selectClass,\n\n                // function called on select that retuns the content to insert\n                selectTemplate: (selectTemplate || Tribute.defaultSelectTemplate).bind(this),\n\n                // function called that returns content for an item\n                menuItemTemplate: (menuItemTemplate || Tribute.defaultMenuItemTemplate).bind(this),\n\n                // function called when menu is empty, disables hiding of menu.\n                noMatchTemplate: (t => {\n                    if (typeof t === 'function') {\n                        return t.bind(this)\n                    }\n\n                    return noMatchTemplate\n                })(noMatchTemplate),\n\n                // column to search against in the object\n                lookup: lookup,\n\n                // column that contains the content to insert by default\n                fillAttr: fillAttr,\n\n                // array of objects or a function returning an array of objects\n                values: values,\n\n                requireLeadingSpace: requireLeadingSpace\n            }]\n        }\n        else if (collection) {\n            this.collection = collection.map(item => {\n                return {\n                    trigger: item.trigger || trigger,\n                    iframe: item.iframe || iframe,\n                    selectClass: item.selectClass || selectClass,\n                    selectTemplate: (item.selectTemplate || Tribute.defaultSelectTemplate).bind(this),\n                    menuItemTemplate: (item.menuItemTemplate || Tribute.defaultMenuItemTemplate).bind(this),\n                    // function called when menu is empty, disables hiding of menu.\n                    noMatchTemplate: (t => {\n                        if (typeof t === 'function') {\n                            return t.bind(this)\n                        }\n\n                        return null\n                    })(noMatchTemplate),\n                    lookup: item.lookup || lookup,\n                    fillAttr: item.fillAttr || fillAttr,\n                    values: item.values,\n                    requireLeadingSpace: item.requireLeadingSpace\n                }\n            })\n        }\n        else {\n            throw new Error('[Tribute] No collection specified.')\n        }\n\n        new TributeRange(this)\n        new TributeEvents(this)\n        new TributeMenuEvents(this)\n        new TributeSearch(this)\n    }\n\n    static defaultSelectTemplate(item) {\n      if (typeof item === 'undefined') return null;\n      if (this.range.isContentEditable(this.current.element)) {\n          return '<span class=\"tribute-mention\">' + (this.current.collection.trigger + item.original[this.current.collection.fillAttr]) + '</span>';\n      }\n\n      return this.current.collection.trigger + item.original[this.current.collection.fillAttr];\n    }\n\n    static defaultMenuItemTemplate(matchItem) {\n        return matchItem.string\n    }\n\n    static inputTypes() {\n        return ['TEXTAREA', 'INPUT']\n    }\n\n    triggers() {\n        return this.collection.map(config => {\n            return config.trigger\n        })\n    }\n\n    attach(el) {\n        if (!el) {\n            throw new Error('[Tribute] Must pass in a DOM node or NodeList.')\n        }\n\n        // Check if it is a jQuery collection\n        if (typeof jQuery !== 'undefined' && el instanceof jQuery) {\n            el = el.get()\n        }\n\n        // Is el an Array/Array-like object?\n        if (el.constructor === NodeList || el.constructor === HTMLCollection || el.constructor === Array) {\n            let length = el.length\n            for (var i = 0; i < length; ++i) {\n                this._attach(el[i])\n            }\n        } else {\n            this._attach(el)\n        }\n    }\n\n    _attach(el) {\n        if (el.hasAttribute('data-tribute')) {\n            console.warn('Tribute was already bound to ' + el.nodeName)\n        }\n\n        this.ensureEditable(el)\n        this.events.bind(el)\n        el.setAttribute('data-tribute', true)\n    }\n\n    ensureEditable(element) {\n        if (Tribute.inputTypes().indexOf(element.nodeName) === -1) {\n            if (element.contentEditable) {\n                element.contentEditable = true\n            } else {\n                throw new Error('[Tribute] Cannot bind to ' + element.nodeName)\n            }\n        }\n    }\n\n    createMenu() {\n        let wrapper = this.range.getDocument().createElement('div'),\n            ul = this.range.getDocument().createElement('ul')\n\n        wrapper.className = 'tribute-container'\n        wrapper.appendChild(ul)\n\n        if (this.menuContainer) {\n            return this.menuContainer.appendChild(wrapper)\n        }\n\n        return this.range.getDocument().body.appendChild(wrapper)\n    }\n\n    showMenuFor(element, scrollTo) {\n        // Only proceed if menu isn't already shown for the current element & mentionText\n        if (this.isActive && this.current.element === element && this.current.mentionText === this.currentMentionTextSnapshot) {\n          return\n        }\n        this.currentMentionTextSnapshot = this.current.mentionText\n\n        // create the menu if it doesn't exist.\n        if (!this.menu) {\n            this.menu = this.createMenu()\n            element.tributeMenu = this.menu\n            this.menuEvents.bind(this.menu)\n        }\n\n        this.isActive = true\n        this.menuSelected = 0\n\n        if (!this.current.mentionText) {\n            this.current.mentionText = ''\n        }\n\n        const processValues = (values) => {\n            // Tribute may not be active any more by the time the value callback returns\n            if (!this.isActive) {\n                return\n            }\n\n            let items = this.search.filter(this.current.mentionText, values, {\n                pre: '<span>',\n                post: '</span>',\n                extract: (el) => {\n                    if (typeof this.current.collection.lookup === 'string') {\n                        return el[this.current.collection.lookup]\n                    } else if (typeof this.current.collection.lookup === 'function') {\n                        return this.current.collection.lookup(el, this.current.mentionText)\n                    } else {\n                        throw new Error('Invalid lookup attribute, lookup must be string or function.')\n                    }\n                }\n            })\n\n            this.current.filteredItems = items\n\n\n            let ul = this.menu.querySelector('ul')\n\n            this.range.positionMenuAtCaret(scrollTo)\n\n            if (!items.length) {\n                let noMatchEvent = new CustomEvent('tribute-no-match', { detail: this.menu })\n                this.current.element.dispatchEvent(noMatchEvent)\n                if (!this.current.collection.noMatchTemplate) {\n                    this.hideMenu()\n                } else {\n                    ul.innerHTML = this.current.collection.noMatchTemplate()\n                }\n\n                return\n            }\n\n            ul.innerHTML = ''\n\n            items.forEach((item, index) => {\n                let li = this.range.getDocument().createElement('li')\n                li.setAttribute('data-index', index)\n                li.addEventListener('mouseenter', (e) => {\n                  let li = e.target;\n                  let index = li.getAttribute('data-index')\n                  this.events.setActiveLi(index)\n                })\n                if (this.menuSelected === index) {\n                    li.className = this.current.collection.selectClass\n                }\n                li.innerHTML = this.current.collection.menuItemTemplate(item)\n                ul.appendChild(li)\n            })\n        }\n\n        if (typeof this.current.collection.values === 'function') {\n            this.current.collection.values(this.current.mentionText, processValues)\n        } else {\n            processValues(this.current.collection.values)\n        }\n    }\n\n    showMenuForCollection(element, collectionIndex) {\n        if (element !== document.activeElement) {\n            this.placeCaretAtEnd(element)\n        }\n\n        this.current.collection = this.collection[collectionIndex || 0]\n        this.current.externalTrigger = true\n        this.current.element = element\n\n        if (element.isContentEditable)\n            this.insertTextAtCursor(this.current.collection.trigger)\n        else\n            this.insertAtCaret(element, this.current.collection.trigger)\n\n        this.showMenuFor(element)\n    }\n\n    // TODO: make sure this works for inputs/textareas\n    placeCaretAtEnd(el) {\n        el.focus();\n        if (typeof window.getSelection != \"undefined\"\n                && typeof document.createRange != \"undefined\") {\n            var range = document.createRange();\n            range.selectNodeContents(el);\n            range.collapse(false);\n            var sel = window.getSelection();\n            sel.removeAllRanges();\n            sel.addRange(range);\n        } else if (typeof document.body.createTextRange != \"undefined\") {\n            var textRange = document.body.createTextRange();\n            textRange.moveToElementText(el);\n            textRange.collapse(false);\n            textRange.select();\n        }\n    }\n\n    // for contenteditable\n    insertTextAtCursor(text) {\n        var sel, range, html;\n        sel = window.getSelection();\n        range = sel.getRangeAt(0);\n        range.deleteContents();\n        var textNode = document.createTextNode(text);\n        range.insertNode(textNode);\n        range.selectNodeContents(textNode)\n        range.collapse(false)\n        sel.removeAllRanges()\n        sel.addRange(range)\n    }\n\n    // for regular inputs\n    insertAtCaret(textarea, text) {\n        var scrollPos = textarea.scrollTop;\n        var caretPos = textarea.selectionStart;\n\n        var front = (textarea.value).substring(0, caretPos);\n        var back = (textarea.value).substring(textarea.selectionEnd, textarea.value.length);\n        textarea.value = front + text + back;\n        caretPos = caretPos + text.length;\n        textarea.selectionStart = caretPos;\n        textarea.selectionEnd = caretPos;\n        textarea.focus();\n        textarea.scrollTop = scrollPos;\n    }\n\n    hideMenu() {\n        if (this.menu) {\n            this.menu.style.cssText = 'display: none;'\n            this.isActive = false\n            this.menuSelected = 0\n            this.current = {}\n        }\n    }\n\n    selectItemAtIndex(index, originalEvent) {\n        index = parseInt(index)\n        if (typeof index !== 'number') return\n        let item = this.current.filteredItems[index]\n        let content = this.current.collection.selectTemplate(item)\n        if (content !== null) this.replaceText(content, originalEvent, item)\n    }\n\n    replaceText(content, originalEvent, item) {\n        this.range.replaceTriggerText(content, true, true, originalEvent, item)\n    }\n\n    _append(collection, newValues, replace) {\n        if (typeof collection.values === 'function') {\n            throw new Error('Unable to append to values, as it is a function.')\n        } else if (!replace) {\n            collection.values = collection.values.concat(newValues)\n        } else {\n            collection.values = newValues\n        }\n    }\n\n    append(collectionIndex, newValues, replace) {\n        let index = parseInt(collectionIndex)\n        if (typeof index !== 'number') throw new Error('please provide an index for the collection to update.')\n\n        let collection = this.collection[index]\n\n        this._append(collection, newValues, replace)\n    }\n\n    appendCurrent(newValues, replace) {\n        if (this.isActive) {\n            this._append(this.current.collection, newValues, replace)\n        } else {\n            throw new Error('No active state. Please use append instead and pass an index.')\n        }\n    }\n\n    detach(el) {\n        if (!el) {\n            throw new Error('[Tribute] Must pass in a DOM node or NodeList.')\n        }\n\n        // Check if it is a jQuery collection\n        if (typeof jQuery !== 'undefined' && el instanceof jQuery) {\n            el = el.get()\n        }\n\n        // Is el an Array/Array-like object?\n        if (el.constructor === NodeList || el.constructor === HTMLCollection || el.constructor === Array) {\n            let length = el.length\n            for (var i = 0; i < length; ++i) {\n                this._detach(el[i])\n            }\n        } else {\n            this._detach(el)\n        }\n    }\n\n    _detach(el) {\n        this.events.unbind(el)\n        if (el.tributeMenu) {\n            this.menuEvents.unbind(el.tributeMenu)\n        }\n\n        setTimeout(() => {\n            el.removeAttribute('data-tribute')\n            this.isActive = false\n            if (el.tributeMenu) {\n                el.tributeMenu.remove()\n            }\n        })\n    }\n}\n\nexport default Tribute;\n","class TributeEvents {\n    constructor(tribute) {\n        this.tribute = tribute\n        this.tribute.events = this\n    }\n\n    static keys() {\n        return [{\n            key: 9,\n            value: 'TAB'\n        }, {\n            key: 8,\n            value: 'DELETE'\n        }, {\n            key: 13,\n            value: 'ENTER'\n        }, {\n            key: 27,\n            value: 'ESCAPE'\n        }, {\n            key: 32,\n            value: 'SPACE'\n        }, {\n            key: 38,\n            value: 'UP'\n        }, {\n            key: 40,\n            value: 'DOWN'\n        }]\n    }\n\n    bind(element) {\n        element.boundKeydown = this.keydown.bind(element, this);\n        element.boundKeyup = this.keyup.bind(element, this);\n        element.boundInput = this.input.bind(element, this);\n\n        element.addEventListener('keydown',\n            element.boundKeydown, false)\n        element.addEventListener('keyup',\n            element.boundKeyup, false)\n        element.addEventListener('input',\n            element.boundInput, false)\n    }\n\n    unbind(element) {\n        element.removeEventListener('keydown',\n            element.boundKeydown, false)\n        element.removeEventListener('keyup',\n            element.boundKeyup, false)\n        element.removeEventListener('input',\n            element.boundInput, false)\n\n        delete element.boundKeydown\n        delete element.boundKeyup\n        delete element.boundInput\n    }\n\n    keydown(instance, event) {\n        if (instance.shouldDeactivate(event)) {\n            instance.tribute.isActive = false\n            instance.tribute.hideMenu()\n        }\n\n        let element = this\n        instance.commandEvent = false\n\n        TributeEvents.keys().forEach(o => {\n            if (o.key === event.keyCode) {\n                instance.commandEvent = true\n                instance.callbacks()[o.value.toLowerCase()](event, element)\n            }\n        })\n    }\n\n    input(instance, event) {\n        instance.inputEvent = true\n        instance.keyup.call(this, instance, event)\n    }\n\n    click(instance, event) {\n        let tribute = instance.tribute\n        if (tribute.menu && tribute.menu.contains(event.target)) {\n            let li = event.target\n            event.preventDefault()\n            event.stopPropagation()\n            while (li.nodeName.toLowerCase() !== 'li') {\n                li = li.parentNode\n                if (!li || li === tribute.menu) {\n                    throw new Error('cannot find the <li> container for the click')\n                }\n            }\n            tribute.selectItemAtIndex(li.getAttribute('data-index'), event)\n            tribute.hideMenu()\n\n        // TODO: should fire with externalTrigger and target is outside of menu\n        } else if (tribute.current.element && !tribute.current.externalTrigger) {\n            tribute.current.externalTrigger = false\n            setTimeout(() => tribute.hideMenu())\n        }\n    }\n\n    keyup(instance, event) {\n        if (instance.inputEvent) {\n            instance.inputEvent = false\n        }\n        instance.updateSelection(this)\n\n        if (event.keyCode === 27) return\n\n        if (!instance.tribute.allowSpaces && instance.tribute.hasTrailingSpace) {\n            instance.tribute.hasTrailingSpace = false;\n            instance.commandEvent = true;\n            instance.callbacks()[\"space\"](event, this);\n            return\n        }\n\n        if (!instance.tribute.isActive) {\n            let keyCode = instance.getKeyCode(instance, this, event)\n\n            if (isNaN(keyCode) || !keyCode) return\n\n            let trigger = instance.tribute.triggers().find(trigger => {\n                return trigger.charCodeAt(0) === keyCode\n            })\n\n            if (typeof trigger !== 'undefined') {\n                instance.callbacks().triggerChar(event, this, trigger)\n            }\n        }\n\n        if (instance.tribute.current.trigger && instance.commandEvent === false\n            || instance.tribute.isActive && event.keyCode === 8) {\n          instance.tribute.showMenuFor(this, true)\n        }\n    }\n\n    shouldDeactivate(event) {\n        if (!this.tribute.isActive) return false\n\n        if (this.tribute.current.mentionText.length === 0) {\n            let eventKeyPressed = false\n            TributeEvents.keys().forEach(o => {\n                if (event.keyCode === o.key) eventKeyPressed = true\n            })\n\n            return !eventKeyPressed\n        }\n\n        return false\n    }\n\n    getKeyCode(instance, el, event) {\n        let char\n        let tribute = instance.tribute\n        let info = tribute.range.getTriggerInfo(false, tribute.hasTrailingSpace, true, tribute.allowSpaces)\n\n        if (info) {\n            return info.mentionTriggerChar.charCodeAt(0)\n        } else {\n            return false\n        }\n    }\n\n    updateSelection(el) {\n        this.tribute.current.element = el\n        let info = this.tribute.range.getTriggerInfo(false, this.tribute.hasTrailingSpace, true, this.tribute.allowSpaces)\n\n        if (info) {\n            this.tribute.current.selectedPath = info.mentionSelectedPath\n            this.tribute.current.mentionText = info.mentionText\n            this.tribute.current.selectedOffset = info.mentionSelectedOffset\n        }\n    }\n\n    callbacks() {\n        return {\n            triggerChar: (e, el, trigger) => {\n                let tribute = this.tribute\n                tribute.current.trigger = trigger\n\n                let collectionItem = tribute.collection.find(item => {\n                    return item.trigger === trigger\n                })\n\n                tribute.current.collection = collectionItem\n                if (tribute.inputEvent) tribute.showMenuFor(el, true)\n            },\n            enter: (e, el) => {\n                // choose selection\n                if (this.tribute.isActive) {\n                    e.preventDefault()\n                    e.stopPropagation()\n                    setTimeout(() => {\n                        this.tribute.selectItemAtIndex(this.tribute.menuSelected, e)\n                        this.tribute.hideMenu()\n                    }, 0)\n                }\n            },\n            escape: (e, el) => {\n                if (this.tribute.isActive) {\n                    e.preventDefault()\n                    e.stopPropagation()\n                    this.tribute.isActive = false\n                    this.tribute.hideMenu()\n                }\n            },\n            tab: (e, el) => {\n                // choose first match\n                this.callbacks().enter(e, el)\n            },\n            space: (e, el) => {\n                if (this.tribute.isActive) {\n                    if (this.tribute.spaceSelectsMatch) {\n                        this.callbacks().enter(e, el)\n                    } else if (!this.tribute.allowSpaces) {\n                        e.stopPropagation();\n                        setTimeout(() => {\n                            this.tribute.hideMenu();\n                            this.tribute.isActive = false;\n                        }, 0);\n                    }\n                }\n            },\n            up: (e, el) => {\n                // navigate up ul\n                if (this.tribute.isActive) {\n                    e.preventDefault()\n                    e.stopPropagation()\n                    let count = this.tribute.current.filteredItems.length,\n                        selected = this.tribute.menuSelected\n\n                    if (count > selected && selected > 0) {\n                        this.tribute.menuSelected--\n                        this.setActiveLi()\n                    } else if (selected === 0) {\n                      this.tribute.menuSelected = count - 1\n                      this.setActiveLi()\n                      this.tribute.menu.scrollTop = this.tribute.menu.scrollHeight\n                    }\n                }\n            },\n            down: (e, el) => {\n                // navigate down ul\n                if (this.tribute.isActive) {\n                    e.preventDefault()\n                    e.stopPropagation()\n                    let count = this.tribute.current.filteredItems.length - 1,\n                        selected = this.tribute.menuSelected\n\n                    if (count > selected) {\n                        this.tribute.menuSelected++\n                        this.setActiveLi()\n                    } else if (count === selected) {\n                        this.tribute.menuSelected = 0\n                        this.setActiveLi()\n                        this.tribute.menu.scrollTop = 0\n                    }\n                }\n            },\n            delete: (e, el) => {\n                if (this.tribute.isActive && this.tribute.current.mentionText.length < 1) {\n                    this.tribute.hideMenu()\n                } else if (this.tribute.isActive) {\n                    this.tribute.showMenuFor(el)\n                }\n            }\n        }\n    }\n\n    setActiveLi(index) {\n        let lis = this.tribute.menu.querySelectorAll('li'),\n            length = lis.length >>> 0\n\n        // get heights\n        let menuFullHeight = this.getFullHeight(this.tribute.menu),\n            liHeight = this.getFullHeight(lis[0])\n\n        if (index) this.tribute.menuSelected = index;\n\n        for (let i = 0; i < length; i++) {\n            let li = lis[i]\n            if (i === this.tribute.menuSelected) {\n                let offset = liHeight * (i+1)\n                let scrollTop = this.tribute.menu.scrollTop\n                let totalScroll = scrollTop + menuFullHeight\n\n                if (offset > totalScroll) {\n                  this.tribute.menu.scrollTop += liHeight\n                } else if (offset < totalScroll) {\n                  this.tribute.menu.scrollTop -= liHeight\n                }\n\n                li.className = this.tribute.current.collection.selectClass\n            } else {\n                li.className = ''\n            }\n        }\n    }\n\n    getFullHeight(elem, includeMargin) {\n      let height = elem.getBoundingClientRect().height\n\n      if (includeMargin) {\n        let style = elem.currentStyle || window.getComputedStyle(elem)\n        return height + parseFloat(style.marginTop) + parseFloat(style.marginBottom)\n      }\n\n      return height\n    }\n\n}\n\nexport default TributeEvents;\n","class TributeMenuEvents {\n    constructor(tribute) {\n        this.tribute = tribute\n        this.tribute.menuEvents = this\n        this.menu = this.tribute.menu\n    }\n\n    bind(menu) {\n        menu.menuKeydownEvent = this.tribute.events.keydown.bind(this.menu, this)\n        this.menuClickEvent = this.tribute.events.click.bind(null, this)\n        this.menuContainerScrollEvent = this.debounce(() => {\n            if (this.tribute.isActive) {\n                this.tribute.showMenuFor(this.tribute.current.element, false)\n            }\n        }, 300, false)\n        this.windowResizeEvent = this.debounce(() => {\n            if (this.tribute.isActive) {\n                this.tribute.range.positionMenuAtCaret(true)\n            }\n        }, 300, false)\n\n        // fixes IE11 issues with mouseup\n        this.tribute.range.getDocument().addEventListener('MSPointerUp',\n            this.menuClickEvent, false)\n        menu.addEventListener('keydown',\n            this.menuKeydownEvent, false)\n        this.tribute.range.getDocument().addEventListener('mouseup',\n            this.menuClickEvent, false)\n        window.addEventListener('resize', this.windowResizeEvent)\n\n        if (this.menuContainer) {\n            this.menuContainer.addEventListener('scroll', this.menuContainerScrollEvent, false)\n        } else {\n            window.addEventListener('scroll', this.menuContainerScrollEvent)\n        }\n\n    }\n\n    unbind(menu) {\n        menu.removeEventListener('keydown',\n            menu.menuKeydownEvent, false)\n        delete menu.menuKeydownEvent\n        this.tribute.range.getDocument().removeEventListener('mouseup',\n            this.menuClickEvent, false)\n        this.tribute.range.getDocument().removeEventListener('MSPointerUp',\n            this.menuClickEvent, false)\n        window.removeEventListener('resize', this.windowResizeEvent)\n\n        if (this.menuContainer) {\n            this.menuContainer.removeEventListener('scroll', this.menuContainerScrollEvent, false)\n        } else {\n            window.removeEventListener('scroll', this.menuContainerScrollEvent)\n        }\n    }\n\n    debounce(func, wait, immediate) {\n        var timeout\n        return () => {\n            var context = this,\n                args = arguments\n            var later = () => {\n                timeout = null\n                if (!immediate) func.apply(context, args)\n            }\n            var callNow = immediate && !timeout\n            clearTimeout(timeout)\n            timeout = setTimeout(later, wait)\n            if (callNow) func.apply(context, args)\n        }\n    }\n}\n\n\nexport default TributeMenuEvents;\n","// Thanks to https://github.com/jeff-collins/ment.io\nclass TributeRange {\n    constructor(tribute) {\n        this.tribute = tribute\n        this.tribute.range = this\n    }\n\n    getDocument() {\n        let iframe\n        if (this.tribute.current.collection) {\n            iframe = this.tribute.current.collection.iframe\n        }\n\n        if (!iframe) {\n            return document\n        }\n\n        return iframe.contentWindow.document\n    }\n\n    positionMenuAtCaret(scrollTo) {\n        let context = this.tribute.current,\n            coordinates\n\n        let info = this.getTriggerInfo(false, this.tribute.hasTrailingSpace, true, this.tribute.allowSpaces)\n\n        if (typeof info !== 'undefined') {\n\n            if(!this.tribute.positionMenu){\n                this.tribute.menu.style.cssText = `display: block;`\n                return\n            }\n\n            if (!this.isContentEditable(context.element)) {\n                coordinates = this.getTextAreaOrInputUnderlinePosition(this.tribute.current.element,\n                    info.mentionPosition)\n            }\n            else {\n                coordinates = this.getContentEditableCaretPosition(info.mentionPosition)\n            }\n\n\n            this.tribute.menu.style.cssText = `top: ${coordinates.top}px;\n                                     left: ${coordinates.left}px;\n                                     right: ${coordinates.right}px;\n                                     bottom: ${coordinates.bottom}px;\n                                     position: absolute;\n                                     zIndex: 10000;\n                                     display: block;`\n\n            if (coordinates.left === 'auto') {\n                this.tribute.menu.style.left = 'auto'\n            }\n\n            if (coordinates.top === 'auto') {\n                this.tribute.menu.style.top = 'auto'\n            }\n\n            if (scrollTo) this.scrollIntoView()\n\n            window.setTimeout(() => {\n                let menuDimensions = {\n                   width: this.tribute.menu.offsetWidth,\n                   height: this.tribute.menu.offsetHeight\n                }\n                let menuIsOffScreen = this.isMenuOffScreen(coordinates, menuDimensions)\n\n                let menuIsOffScreenHorizontally = window.innerWidth > menuDimensions.width && (menuIsOffScreen.left || menuIsOffScreen.right)\n                let menuIsOffScreenVertically = window.innerHeight > menuDimensions.height && (menuIsOffScreen.top || menuIsOffScreen.bottom)\n                if (menuIsOffScreenHorizontally || menuIsOffScreenVertically) {\n                    this.tribute.menu.style.cssText = 'display: none'\n                    this.positionMenuAtCaret(scrollTo)\n                }\n            }, 0)\n\n        } else {\n            this.tribute.menu.style.cssText = 'display: none'\n        }\n    }\n\n    selectElement(targetElement, path, offset) {\n        let range\n        let elem = targetElement\n\n        if (path) {\n            for (var i = 0; i < path.length; i++) {\n                elem = elem.childNodes[path[i]]\n                if (elem === undefined) {\n                    return\n                }\n                while (elem.length < offset) {\n                    offset -= elem.length\n                    elem = elem.nextSibling\n                }\n                if (elem.childNodes.length === 0 && !elem.length) {\n                    elem = elem.previousSibling\n                }\n            }\n        }\n        let sel = this.getWindowSelection()\n\n        range = this.getDocument().createRange()\n        range.setStart(elem, offset)\n        range.setEnd(elem, offset)\n        range.collapse(true)\n\n        try {\n            sel.removeAllRanges()\n        } catch (error) {}\n\n        sel.addRange(range)\n        targetElement.focus()\n    }\n\n    resetSelection(targetElement, path, offset) {\n        if (!this.isContentEditable(targetElement)) {\n            if (targetElement !== this.tribute.current.element) {\n                targetElement.focus()\n            }\n        } else {\n            this.selectElement(targetElement, path, offset)\n        }\n    }\n\n    replaceTriggerText(text, requireLeadingSpace, hasTrailingSpace, originalEvent, item) {\n        let context = this.tribute.current\n        this.resetSelection(context.element, context.selectedPath, context.selectedOffset)\n\n        let info = this.getTriggerInfo(true, hasTrailingSpace, requireLeadingSpace, this.tribute.allowSpaces)\n\n        // Create the event\n        let replaceEvent = new CustomEvent('tribute-replaced', {\n            detail: {\n                item: item,\n                event: originalEvent\n            }\n        })\n\n        if (info !== undefined) {\n            if (!this.isContentEditable(context.element)) {\n                let myField = this.tribute.current.element\n                let textSuffix = typeof this.tribute.replaceTextSuffix == 'string'\n                    ? this.tribute.replaceTextSuffix\n                    : ' '\n                text += textSuffix\n                let startPos = info.mentionPosition\n                let endPos = info.mentionPosition + info.mentionText.length + textSuffix.length\n                myField.value = myField.value.substring(0, startPos) + text +\n                    myField.value.substring(endPos, myField.value.length)\n                myField.selectionStart = startPos + text.length\n                myField.selectionEnd = startPos + text.length\n            } else {\n                // add a space to the end of the pasted text\n                let textSuffix = typeof this.tribute.replaceTextSuffix == 'string'\n                    ? this.tribute.replaceTextSuffix\n                    : '\\xA0'\n                text += textSuffix\n                this.pasteHtml(text, info.mentionPosition,\n                    info.mentionPosition + info.mentionText.length + 1)\n            }\n\n            context.element.dispatchEvent(replaceEvent)\n        }\n    }\n\n    pasteHtml(html, startPos, endPos) {\n        let range, sel\n        sel = this.getWindowSelection()\n        range = this.getDocument().createRange()\n        range.setStart(sel.anchorNode, startPos)\n        range.setEnd(sel.anchorNode, endPos)\n        range.deleteContents()\n\n        let el = this.getDocument().createElement('div')\n        el.innerHTML = html\n        let frag = this.getDocument().createDocumentFragment(),\n            node, lastNode\n        while ((node = el.firstChild)) {\n            lastNode = frag.appendChild(node)\n        }\n        range.insertNode(frag)\n\n        // Preserve the selection\n        if (lastNode) {\n            range = range.cloneRange()\n            range.setStartAfter(lastNode)\n            range.collapse(true)\n            sel.removeAllRanges()\n            sel.addRange(range)\n        }\n    }\n\n    getWindowSelection() {\n        if (this.tribute.collection.iframe) {\n            return this.tribute.collection.iframe.contentWindow.getSelection()\n        }\n\n        return window.getSelection()\n    }\n\n    getNodePositionInParent(element) {\n        if (element.parentNode === null) {\n            return 0\n        }\n\n        for (var i = 0; i < element.parentNode.childNodes.length; i++) {\n            let node = element.parentNode.childNodes[i]\n\n            if (node === element) {\n                return i\n            }\n        }\n    }\n\n    getContentEditableSelectedPath(ctx) {\n        let sel = this.getWindowSelection()\n        let selected = sel.anchorNode\n        let path = []\n        let offset\n\n        if (selected != null) {\n            let i\n            let ce = selected.contentEditable\n            while (selected !== null && ce !== 'true') {\n                i = this.getNodePositionInParent(selected)\n                path.push(i)\n                selected = selected.parentNode\n                if (selected !== null) {\n                    ce = selected.contentEditable\n                }\n            }\n            path.reverse()\n\n            // getRangeAt may not exist, need alternative\n            offset = sel.getRangeAt(0).startOffset\n\n            return {\n                selected: selected,\n                path: path,\n                offset: offset\n            }\n        }\n    }\n\n    getTextPrecedingCurrentSelection() {\n        let context = this.tribute.current,\n            text = ''\n\n        if (!this.isContentEditable(context.element)) {\n            let textComponent = this.tribute.current.element;\n            if (textComponent) {\n                let startPos = textComponent.selectionStart\n                if (textComponent.value && startPos >= 0) {\n                    text = textComponent.value.substring(0, startPos)\n                }\n            }\n\n        } else {\n            let selectedElem = this.getWindowSelection().anchorNode\n\n            if (selectedElem != null) {\n                let workingNodeContent = selectedElem.textContent\n                let selectStartOffset = this.getWindowSelection().getRangeAt(0).startOffset\n\n                if (workingNodeContent && selectStartOffset >= 0) {\n                    text = workingNodeContent.substring(0, selectStartOffset)\n                }\n            }\n        }\n\n        return text\n    }\n\n    getTriggerInfo(menuAlreadyActive, hasTrailingSpace, requireLeadingSpace, allowSpaces) {\n        let ctx = this.tribute.current\n        let selected, path, offset\n\n        if (!this.isContentEditable(ctx.element)) {\n            selected = this.tribute.current.element\n        } else {\n            let selectionInfo = this.getContentEditableSelectedPath(ctx)\n\n            if (selectionInfo) {\n                selected = selectionInfo.selected\n                path = selectionInfo.path\n                offset = selectionInfo.offset\n            }\n        }\n\n        let effectiveRange = this.getTextPrecedingCurrentSelection()\n\n        if (effectiveRange !== undefined && effectiveRange !== null) {\n            let mostRecentTriggerCharPos = -1\n            let triggerChar\n\n            this.tribute.collection.forEach(config => {\n                let c = config.trigger\n                let idx = config.requireLeadingSpace ?\n                    this.lastIndexWithLeadingSpace(effectiveRange, c) :\n                    effectiveRange.lastIndexOf(c)\n\n                if (idx > mostRecentTriggerCharPos) {\n                    mostRecentTriggerCharPos = idx\n                    triggerChar = c\n                    requireLeadingSpace = config.requireLeadingSpace\n                }\n            })\n\n            if (mostRecentTriggerCharPos >= 0 &&\n                (\n                    mostRecentTriggerCharPos === 0 ||\n                    !requireLeadingSpace ||\n                    /[\\xA0\\s]/g.test(\n                        effectiveRange.substring(\n                            mostRecentTriggerCharPos - 1,\n                            mostRecentTriggerCharPos)\n                    )\n                )\n            ) {\n                let currentTriggerSnippet = effectiveRange.substring(mostRecentTriggerCharPos + 1,\n                    effectiveRange.length)\n\n                triggerChar = effectiveRange.substring(mostRecentTriggerCharPos, mostRecentTriggerCharPos + 1)\n                let firstSnippetChar = currentTriggerSnippet.substring(0, 1)\n                let leadingSpace = currentTriggerSnippet.length > 0 &&\n                    (\n                        firstSnippetChar === ' ' ||\n                        firstSnippetChar === '\\xA0'\n                    )\n                if (hasTrailingSpace) {\n                    currentTriggerSnippet = currentTriggerSnippet.trim()\n                }\n\n                let regex = allowSpaces ? /[^\\S ]/g : /[\\xA0\\s]/g;\n\n                this.tribute.hasTrailingSpace = regex.test(currentTriggerSnippet);\n\n                if (!leadingSpace && (menuAlreadyActive || !(regex.test(currentTriggerSnippet)))) {\n                    return {\n                        mentionPosition: mostRecentTriggerCharPos,\n                        mentionText: currentTriggerSnippet,\n                        mentionSelectedElement: selected,\n                        mentionSelectedPath: path,\n                        mentionSelectedOffset: offset,\n                        mentionTriggerChar: triggerChar\n                    }\n                }\n            }\n        }\n    }\n\n    lastIndexWithLeadingSpace (str, char) {\n        let reversedStr = str.split('').reverse().join('')\n        let index = -1\n\n        for (let cidx = 0, len = str.length; cidx < len; cidx++) {\n            let firstChar = cidx === str.length - 1\n            let leadingSpace = /\\s/.test(reversedStr[cidx + 1])\n            let match = char === reversedStr[cidx]\n\n            if (match && (firstChar || leadingSpace)) {\n                index = str.length - 1 - cidx\n                break\n            }\n        }\n\n        return index\n    }\n\n    isContentEditable(element) {\n        return element.nodeName !== 'INPUT' && element.nodeName !== 'TEXTAREA'\n    }\n\n    isMenuOffScreen(coordinates, menuDimensions) {\n        let windowWidth = window.innerWidth\n        let windowHeight = window.innerHeight\n        let doc = document.documentElement\n        let windowLeft = (window.pageXOffset || doc.scrollLeft) - (doc.clientLeft || 0)\n        let windowTop = (window.pageYOffset || doc.scrollTop) - (doc.clientTop || 0)\n\n        let menuTop = typeof coordinates.top === 'number' ? coordinates.top : windowTop + windowHeight - coordinates.bottom - menuDimensions.height\n        let menuRight = typeof coordinates.right === 'number' ? coordinates.right : coordinates.left + menuDimensions.width\n        let menuBottom = typeof coordinates.bottom === 'number' ? coordinates.bottom : coordinates.top + menuDimensions.height\n        let menuLeft = typeof coordinates.left === 'number' ? coordinates.left : windowLeft + windowWidth - coordinates.right - menuDimensions.width\n\n        return {\n            top: menuTop < Math.floor(windowTop),\n            right: menuRight > Math.ceil(windowLeft + windowWidth),\n            bottom: menuBottom > Math.ceil(windowTop + windowHeight),\n            left: menuLeft < Math.floor(windowLeft)\n        }\n    }\n\n    getMenuDimensions() {\n        // Width of the menu depends of its contents and position\n        // We must check what its width would be without any obstruction\n        // This way, we can achieve good positioning for flipping the menu\n        let dimensions = {\n            width: null,\n            height: null\n        }\n\n        this.tribute.menu.style.cssText = `top: 0px;\n                                 left: 0px;\n                                 position: fixed;\n                                 zIndex: 10000;\n                                 display: block;\n                                 visibility; hidden;`\n       dimensions.width = this.tribute.menu.offsetWidth\n       dimensions.height = this.tribute.menu.offsetHeight\n\n       this.tribute.menu.style.cssText = `display: none;`\n\n       return dimensions\n    }\n\n    getTextAreaOrInputUnderlinePosition(element, position, flipped) {\n        let properties = ['direction', 'boxSizing', 'width', 'height', 'overflowX',\n            'overflowY', 'borderTopWidth', 'borderRightWidth',\n            'borderBottomWidth', 'borderLeftWidth', 'paddingTop',\n            'paddingRight', 'paddingBottom', 'paddingLeft',\n            'fontStyle', 'fontVariant', 'fontWeight', 'fontStretch',\n            'fontSize', 'fontSizeAdjust', 'lineHeight', 'fontFamily',\n            'textAlign', 'textTransform', 'textIndent',\n            'textDecoration', 'letterSpacing', 'wordSpacing'\n        ]\n\n        let isFirefox = (window.mozInnerScreenX !== null)\n\n        let div = this.getDocument().createElement('div')\n        div.id = 'input-textarea-caret-position-mirror-div'\n        this.getDocument().body.appendChild(div)\n\n        let style = div.style\n        let computed = window.getComputedStyle ? getComputedStyle(element) : element.currentStyle\n\n        style.whiteSpace = 'pre-wrap'\n        if (element.nodeName !== 'INPUT') {\n            style.wordWrap = 'break-word'\n        }\n\n        // position off-screen\n        style.position = 'absolute'\n        style.visibility = 'hidden'\n\n        // transfer the element's properties to the div\n        properties.forEach(prop => {\n            style[prop] = computed[prop]\n        })\n\n        if (isFirefox) {\n            style.width = `${(parseInt(computed.width) - 2)}px`\n            if (element.scrollHeight > parseInt(computed.height))\n                style.overflowY = 'scroll'\n        } else {\n            style.overflow = 'hidden'\n        }\n\n        div.textContent = element.value.substring(0, position)\n\n        if (element.nodeName === 'INPUT') {\n            div.textContent = div.textContent.replace(/\\s/g, ' ')\n        }\n\n        let span = this.getDocument().createElement('span')\n        span.textContent = element.value.substring(position) || '.'\n        div.appendChild(span)\n\n        let rect = element.getBoundingClientRect()\n        let doc = document.documentElement\n        let windowLeft = (window.pageXOffset || doc.scrollLeft) - (doc.clientLeft || 0)\n        let windowTop = (window.pageYOffset || doc.scrollTop) - (doc.clientTop || 0)\n\n        let coordinates = {\n            top: rect.top + windowTop + span.offsetTop + parseInt(computed.borderTopWidth) + parseInt(computed.fontSize) - element.scrollTop,\n            left: rect.left + windowLeft + span.offsetLeft + parseInt(computed.borderLeftWidth)\n        }\n\n        let windowWidth = window.innerWidth\n        let windowHeight = window.innerHeight\n\n        let menuDimensions = this.getMenuDimensions()\n        let menuIsOffScreen = this.isMenuOffScreen(coordinates, menuDimensions)\n\n        if (menuIsOffScreen.right) {\n            coordinates.right = windowWidth - coordinates.left\n            coordinates.left = 'auto'\n        }\n\n        let parentHeight = this.tribute.menuContainer\n            ? this.tribute.menuContainer.offsetHeight\n            : this.getDocument().body.offsetHeight\n\n        if (menuIsOffScreen.bottom) {\n            let parentRect = this.tribute.menuContainer\n                ? this.tribute.menuContainer.getBoundingClientRect()\n                : this.getDocument().body.getBoundingClientRect()\n            let scrollStillAvailable = parentHeight - (windowHeight - parentRect.top)\n\n            coordinates.bottom = scrollStillAvailable + (windowHeight - rect.top - span.offsetTop)\n            coordinates.top = 'auto'\n        }\n\n        menuIsOffScreen = this.isMenuOffScreen(coordinates, menuDimensions)\n        if (menuIsOffScreen.left) {\n            coordinates.left = windowWidth > menuDimensions.width\n                ? windowLeft + windowWidth - menuDimensions.width\n                : windowLeft\n            delete coordinates.right\n        }\n        if (menuIsOffScreen.top) {\n            coordinates.top = windowHeight > menuDimensions.height\n                ? windowTop + windowHeight - menuDimensions.height\n                : windowTop\n            delete coordinates.bottom\n        }\n\n        this.getDocument().body.removeChild(div)\n        return coordinates\n    }\n\n    getContentEditableCaretPosition(selectedNodePosition) {\n        let markerTextChar = '﻿'\n        let markerEl, markerId = `sel_${new Date().getTime()}_${Math.random().toString().substr(2)}`\n        let range\n        let sel = this.getWindowSelection()\n        let prevRange = sel.getRangeAt(0)\n\n        range = this.getDocument().createRange()\n        range.setStart(sel.anchorNode, selectedNodePosition)\n        range.setEnd(sel.anchorNode, selectedNodePosition)\n\n        range.collapse(false)\n\n        // Create the marker element containing a single invisible character using DOM methods and insert it\n        markerEl = this.getDocument().createElement('span')\n        markerEl.id = markerId\n\n        markerEl.appendChild(this.getDocument().createTextNode(markerTextChar))\n        range.insertNode(markerEl)\n        sel.removeAllRanges()\n        sel.addRange(prevRange)\n\n        let rect = markerEl.getBoundingClientRect()\n        let doc = document.documentElement\n        let windowLeft = (window.pageXOffset || doc.scrollLeft) - (doc.clientLeft || 0)\n        let windowTop = (window.pageYOffset || doc.scrollTop) - (doc.clientTop || 0)\n        let coordinates = {\n            left: rect.left + windowLeft,\n            top: rect.top + markerEl.offsetHeight + windowTop\n        }\n        let windowWidth = window.innerWidth\n        let windowHeight = window.innerHeight\n\n        let menuDimensions = this.getMenuDimensions()\n        let menuIsOffScreen = this.isMenuOffScreen(coordinates, menuDimensions)\n\n        if (menuIsOffScreen.right) {\n            coordinates.left = 'auto'\n            coordinates.right = windowWidth - rect.left - windowLeft\n        }\n\n        let parentHeight = this.tribute.menuContainer\n            ? this.tribute.menuContainer.offsetHeight\n            : this.getDocument().body.offsetHeight\n\n        if (menuIsOffScreen.bottom) {\n            let parentRect = this.tribute.menuContainer\n                ? this.tribute.menuContainer.getBoundingClientRect()\n                : this.getDocument().body.getBoundingClientRect()\n            let scrollStillAvailable = parentHeight - (windowHeight - parentRect.top)\n\n            coordinates.top = 'auto'\n            coordinates.bottom = scrollStillAvailable + (windowHeight - rect.top)\n        }\n\n        menuIsOffScreen = this.isMenuOffScreen(coordinates, menuDimensions)\n        if (menuIsOffScreen.left) {\n            coordinates.left = windowWidth > menuDimensions.width\n                ? windowLeft + windowWidth - menuDimensions.width\n                : windowLeft\n            delete coordinates.right\n        }\n        if (menuIsOffScreen.top) {\n            coordinates.top = windowHeight > menuDimensions.height\n                ? windowTop + windowHeight - menuDimensions.height\n                : windowTop\n            delete coordinates.bottom\n        }\n\n        markerEl.parentNode.removeChild(markerEl)\n        return coordinates\n    }\n\n    scrollIntoView(elem) {\n        let reasonableBuffer = 20,\n            clientRect\n        let maxScrollDisplacement = 100\n        let e = this.menu\n\n        if (typeof e === 'undefined') return;\n\n        while (clientRect === undefined || clientRect.height === 0) {\n            clientRect = e.getBoundingClientRect()\n\n            if (clientRect.height === 0) {\n                e = e.childNodes[0]\n                if (e === undefined || !e.getBoundingClientRect) {\n                    return\n                }\n            }\n        }\n\n        let elemTop = clientRect.top\n        let elemBottom = elemTop + clientRect.height\n\n        if (elemTop < 0) {\n            window.scrollTo(0, window.pageYOffset + clientRect.top - reasonableBuffer)\n        } else if (elemBottom > window.innerHeight) {\n            let maxY = window.pageYOffset + clientRect.top - reasonableBuffer\n\n            if (maxY - window.pageYOffset > maxScrollDisplacement) {\n                maxY = window.pageYOffset + maxScrollDisplacement\n            }\n\n            let targetY = window.pageYOffset - (window.innerHeight - elemBottom)\n\n            if (targetY > maxY) {\n                targetY = maxY\n            }\n\n            window.scrollTo(0, targetY)\n        }\n    }\n}\n\n\nexport default TributeRange;\n","// Thanks to https://github.com/mattyork/fuzzy\nclass TributeSearch {\n    constructor(tribute) {\n        this.tribute = tribute\n        this.tribute.search = this\n    }\n\n    simpleFilter(pattern, array) {\n        return array.filter(string => {\n            return this.test(pattern, string)\n        })\n    }\n\n    test(pattern, string) {\n        return this.match(pattern, string) !== null\n    }\n\n    match(pattern, string, opts) {\n        opts = opts || {}\n        let patternIdx = 0,\n            result = [],\n            len = string.length,\n            totalScore = 0,\n            currScore = 0,\n            pre = opts.pre || '',\n            post = opts.post || '',\n            compareString = opts.caseSensitive && string || string.toLowerCase(),\n            ch, compareChar\n\n        pattern = opts.caseSensitive && pattern || pattern.toLowerCase()\n\n        let patternCache = this.traverse(compareString, pattern, 0, 0, [])\n        if (!patternCache) {\n            return null\n        }\n\n        return {\n            rendered: this.render(string, patternCache.cache, pre, post),\n            score: patternCache.score\n        }\n    }\n\n    traverse(string, pattern, stringIndex, patternIndex, patternCache) {\n        // if the pattern search at end\n        if (pattern.length === patternIndex) {\n\n            // calculate score and copy the cache containing the indices where it's found\n            return {\n                score: this.calculateScore(patternCache),\n                cache: patternCache.slice()\n            }\n        }\n\n        // if string at end or remaining pattern > remaining string\n        if (string.length === stringIndex || pattern.length - patternIndex > string.length - stringIndex) {\n            return undefined\n        }\n\n        let c = pattern[patternIndex]\n        let index = string.indexOf(c, stringIndex)\n        let best, temp\n\n        while (index > -1) {\n            patternCache.push(index)\n            temp = this.traverse(string, pattern, index + 1, patternIndex + 1, patternCache)\n            patternCache.pop()\n\n            // if downstream traversal failed, return best answer so far\n            if (!temp) {\n                return best\n            }\n\n            if (!best || best.score < temp.score) {\n                best = temp\n            }\n\n            index = string.indexOf(c, index + 1)\n        }\n\n        return best\n    }\n\n    calculateScore(patternCache) {\n        let score = 0\n        let temp = 1\n\n        patternCache.forEach((index, i) => {\n            if (i > 0) {\n                if (patternCache[i - 1] + 1 === index) {\n                    temp += temp + 1\n                }\n                else {\n                    temp = 1\n                }\n            }\n\n            score += temp\n        })\n\n        return score\n    }\n\n    render(string, indices, pre, post) {\n        var rendered = string.substring(0, indices[0])\n\n        indices.forEach((index, i) => {\n            rendered += pre + string[index] + post +\n                string.substring(index + 1, (indices[i + 1]) ? indices[i + 1] : string.length)\n        })\n\n        return rendered\n    }\n\n    filter(pattern, arr, opts) {\n        opts = opts || {}\n        return arr\n            .reduce((prev, element, idx, arr) => {\n                let str = element\n\n                if (opts.extract) {\n                    str = opts.extract(element)\n\n                    if (!str) { // take care of undefineds / nulls / etc.\n                        str = ''\n                    }\n                }\n\n                let rendered = this.match(pattern, str, opts)\n\n                if (rendered != null) {\n                    prev[prev.length] = {\n                        string: rendered.rendered,\n                        score: rendered.score,\n                        index: idx,\n                        original: element\n                    }\n                }\n\n                return prev\n            }, [])\n\n        .sort((a, b) => {\n            let compare = b.score - a.score\n            if (compare) return compare\n            return a.index - b.index\n        })\n    }\n}\n\nexport default TributeSearch;\n","/**\n* Tribute.js\n* Native ES6 JavaScript @mention Plugin\n**/\n\nimport Tribute from \"./Tribute\";\n\nexport default Tribute;\n","if (!Array.prototype.find) {\n    Array.prototype.find = function(predicate) {\n        if (this === null) {\n            throw new TypeError('Array.prototype.find called on null or undefined')\n        }\n        if (typeof predicate !== 'function') {\n            throw new TypeError('predicate must be a function')\n        }\n        var list = Object(this)\n        var length = list.length >>> 0\n        var thisArg = arguments[1]\n        var value\n\n        for (var i = 0; i < length; i++) {\n            value = list[i]\n            if (predicate.call(thisArg, value, i, list)) {\n                return value\n            }\n        }\n        return undefined\n    }\n}\n\nif (window && typeof window.CustomEvent !== \"function\") {\n  function CustomEvent(event, params) {\n    params = params || {\n      bubbles: false,\n      cancelable: false,\n      detail: undefined\n    }\n    var evt = document.createEvent('CustomEvent')\n    evt.initCustomEvent(event, params.bubbles, params.cancelable, params.detail)\n    return evt\n  }\n\n if (typeof window.Event !== 'undefined') {\n   CustomEvent.prototype = window.Event.prototype\n }\n\n  window.CustomEvent = CustomEvent\n}"]}
4034diff --git src/bp-friends/bp-friends-functions.php src/bp-friends/bp-friends-functions.php
4035index 051c65018..cf4ae4c0f 100644
4036--- src/bp-friends/bp-friends-functions.php
4037+++ src/bp-friends/bp-friends-functions.php
4038@@ -804,12 +804,14 @@ function bp_friends_prime_mentions_results() {
4039                $result        = new stdClass();
4040                $result->ID    = $user->user_nicename;
4041                $result->image = bp_core_fetch_avatar( array( 'html' => false, 'item_id' => $user->ID ) );
4042+               $result->user_id = $user->ID;
4043 
4044                if ( ! empty( $user->display_name ) && ! bp_disable_profile_sync() ) {
4045                        $result->name = $user->display_name;
4046                } else {
4047                        $result->name = bp_core_get_user_displayname( $user->ID );
4048                }
4049+               $result->search = "{$result->ID} {$result->name}";
4050 
4051                $results[] = $result;
4052        }