2012-11-14 20:52:53 +01:00
|
|
|
var search = (function () {
|
|
|
|
|
|
|
|
var exports = {};
|
|
|
|
|
2012-10-26 16:59:38 +02:00
|
|
|
var cached_term = "";
|
|
|
|
var cached_matches = [];
|
|
|
|
var cached_index;
|
|
|
|
var cached_table = $('table.focused_table');
|
|
|
|
|
2012-11-15 03:44:50 +01:00
|
|
|
// Data storage for the typeahead -- to go from object to string representation and vice versa.
|
|
|
|
var labels = [];
|
|
|
|
var mapped = {};
|
|
|
|
|
2012-11-14 22:12:21 +01:00
|
|
|
function narrow_or_search_for_term(item) {
|
2012-11-15 03:44:50 +01:00
|
|
|
var obj = mapped[item];
|
|
|
|
if (obj.action === "search") {
|
|
|
|
// TODO: Should this actually be in a keyup handler? Otherwise we can't tab over and search-down that easily
|
|
|
|
exports.search_button_handler(true); // Effectively, click the button (since enter is no longer going to)
|
|
|
|
return obj.query;
|
|
|
|
} else if (obj.action === "stream") {
|
|
|
|
narrow.by_stream_name(obj.query);
|
|
|
|
// It's sort of annoying that this is not in a position to
|
|
|
|
// blur the search box, because it means that Esc won't
|
|
|
|
// unnarrow, it'll leave the searchbox.
|
|
|
|
return ""; // Keep the search box empty
|
|
|
|
} else if (obj.action === "private_message") {
|
|
|
|
narrow.by_private_message_partner(obj.query.full_name, obj.query.email);
|
|
|
|
return "";
|
|
|
|
}
|
2012-11-14 22:12:21 +01:00
|
|
|
return item;
|
|
|
|
}
|
|
|
|
|
2012-11-15 03:44:50 +01:00
|
|
|
function render_object(obj) {
|
|
|
|
if (obj.action === 'search') {
|
|
|
|
return "Search for " + obj.query;
|
|
|
|
} else if (obj.action === 'stream') {
|
|
|
|
return "Narrow to stream " + obj.query;
|
|
|
|
} else if (obj.action === 'private_message') {
|
|
|
|
return "Narrow to person " + obj.query.full_name + " <" + obj.query.email + ">";
|
|
|
|
}
|
|
|
|
return "Error";
|
|
|
|
}
|
|
|
|
|
|
|
|
// Borrowed from composebox_typeahead_highlighter in composebox_typeahead.js.
|
|
|
|
function searchbox_typeahead_highlighter(item) {
|
|
|
|
var query = this.query;
|
|
|
|
var string_item = render_object(mapped[item]);
|
|
|
|
query = query.replace(/[\-\[\]{}()*+?.,\\\^$|#\s]/g, '\\$&');
|
|
|
|
var regex = new RegExp('(' + query + ')', 'ig');
|
|
|
|
var pieces = string_item.split(regex);
|
|
|
|
var result = "";
|
|
|
|
$.each(pieces, function(idx, piece) {
|
|
|
|
if (piece.match(regex)) {
|
|
|
|
result += "<strong>" + Handlebars.Utils.escapeExpression(piece) + "</strong>";
|
|
|
|
} else {
|
|
|
|
result += Handlebars.Utils.escapeExpression(piece);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2012-11-14 22:12:21 +01:00
|
|
|
exports.initialize = function () {
|
|
|
|
$( "#search_query" ).typeahead({
|
|
|
|
source: function (query, process) {
|
2012-11-15 03:44:50 +01:00
|
|
|
var streams = $.map(stream_list, function(elt,idx) {
|
|
|
|
return {action: 'stream', query: elt};
|
|
|
|
});
|
|
|
|
var people = $.map(people_list, function(elt,idx) {
|
|
|
|
return {action: 'private_message', query: elt};
|
|
|
|
});
|
|
|
|
var options = streams.concat(people);
|
|
|
|
options.unshift({action: 'search', query: query});
|
|
|
|
|
|
|
|
mapped = {};
|
|
|
|
labels = [];
|
|
|
|
$.each(options, function (i, obj) {
|
|
|
|
var label = render_object(obj);
|
|
|
|
mapped[label] = obj;
|
|
|
|
labels.push(label);
|
|
|
|
});
|
|
|
|
return labels;
|
2012-11-14 22:12:21 +01:00
|
|
|
},
|
|
|
|
items: 3,
|
2012-11-15 03:44:50 +01:00
|
|
|
highlighter: searchbox_typeahead_highlighter,
|
|
|
|
matcher: function (item) {
|
|
|
|
var obj = mapped[item];
|
|
|
|
var actual_search_term = obj.query;
|
|
|
|
if (obj.action === 'private_message') {
|
|
|
|
actual_search_term = obj.query.full_name + ' <' + obj.query.email + '>';
|
|
|
|
}
|
|
|
|
// Case-insensitive (from Bootstrap's default matcher).
|
|
|
|
return (actual_search_term.toLowerCase().indexOf(this.query.toLowerCase()) !== -1);
|
|
|
|
},
|
2012-11-14 22:12:21 +01:00
|
|
|
updater: narrow_or_search_for_term
|
|
|
|
});
|
|
|
|
|
|
|
|
$("#searchbox_form").keydown(function (e) {
|
|
|
|
var code = e.which;
|
|
|
|
if (code === 13 && $("#search_query").data().typeahead.shown) {
|
|
|
|
// We pressed Enter and the typeahead is open;
|
|
|
|
// don't submit the form so that the typeahead
|
|
|
|
// can instead handle our Enter keypress.
|
|
|
|
e.preventDefault();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
});
|
2012-11-15 03:44:50 +01:00
|
|
|
// TODO: If we were to add a keyup handler, maybe we would blur
|
|
|
|
// the input after "Enter"? Otherwise, right now, when we narrow,
|
|
|
|
// our cursor remains in the searchbox, when probably it should be
|
|
|
|
// blurred or move to the button or something.
|
2012-11-14 22:12:21 +01:00
|
|
|
};
|
|
|
|
|
2012-10-26 16:59:38 +02:00
|
|
|
function match_on_visible_text(row, search_term) {
|
|
|
|
// You can't select on :visible, since that includes hidden elements that
|
|
|
|
// take up space.
|
2012-10-29 17:25:18 +01:00
|
|
|
return row.find(".message_content, .sender_name, .message_header, .message_time")
|
2012-10-27 04:14:17 +02:00
|
|
|
.text().toLowerCase().indexOf(search_term) !== -1;
|
2012-10-26 16:59:38 +02:00
|
|
|
}
|
|
|
|
|
2012-11-01 23:03:10 +01:00
|
|
|
function disable_search_arrows_if(condition, affected_arrows) {
|
|
|
|
var i, button;
|
|
|
|
|
|
|
|
for (i = 0; i < affected_arrows.length; i++) {
|
|
|
|
button = $("#search_" + affected_arrows[i]);
|
|
|
|
if (condition) {
|
|
|
|
button.attr("disabled", "disabled");
|
|
|
|
} else {
|
|
|
|
button.removeAttr("disabled");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-26 16:59:38 +02:00
|
|
|
function search(term, highlighted_message, reverse) {
|
|
|
|
// term: case-insensitive text to search for
|
|
|
|
// highlighted_message: the current location of the pointer. Ignored
|
|
|
|
// on cached queries
|
|
|
|
// reverse: boolean as to whether the search is forward or backwards
|
|
|
|
//
|
|
|
|
// returns a message object containing the search term.
|
|
|
|
var previous_header_matched = false;
|
|
|
|
|
|
|
|
var focused_table = $('table.focused_table');
|
|
|
|
if ((term !== cached_term) || (cached_table[0] !== focused_table[0])) {
|
|
|
|
cached_term = term;
|
|
|
|
cached_matches = [];
|
|
|
|
cached_index = null;
|
|
|
|
cached_table = focused_table;
|
2012-11-01 19:20:41 +01:00
|
|
|
var selected_zid = rows.id(highlighted_message);
|
2012-10-26 16:59:38 +02:00
|
|
|
|
2012-10-27 02:16:46 +02:00
|
|
|
focused_table.find('.message_row, .recipient_row').each(function (index, row) {
|
2012-10-26 16:59:38 +02:00
|
|
|
row = $(row);
|
|
|
|
if (previous_header_matched || (match_on_visible_text(row, term))) {
|
|
|
|
previous_header_matched = false;
|
|
|
|
|
|
|
|
if (row.hasClass("recipient_row")) {
|
|
|
|
previous_header_matched = true;
|
|
|
|
} else {
|
|
|
|
cached_matches.push(row);
|
2012-11-01 19:20:41 +01:00
|
|
|
var zid = rows.id(row);
|
2012-10-26 16:59:38 +02:00
|
|
|
if ((reverse && (zid <= selected_zid)) ||
|
|
|
|
(!reverse && (zid >= selected_zid) && !cached_index)) {
|
|
|
|
// Keep track of the closest match going up or down.
|
|
|
|
cached_index = cached_matches.length - 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2012-11-01 23:03:10 +01:00
|
|
|
disable_search_arrows_if(cached_matches.length === 0, ["up", "down"]);
|
|
|
|
|
2012-10-26 16:59:38 +02:00
|
|
|
return cached_matches[cached_index];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (reverse) {
|
|
|
|
if (cached_index > 0) {
|
|
|
|
cached_index--;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (cached_index < cached_matches.length - 1) {
|
|
|
|
cached_index++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-01 23:03:10 +01:00
|
|
|
disable_search_arrows_if(cached_matches.length === 0, ["up", "down"]);
|
|
|
|
disable_search_arrows_if(cached_index === 0, ["up"]);
|
|
|
|
disable_search_arrows_if(cached_index === cached_matches.length - 1, ["down"]);
|
|
|
|
|
2012-10-26 16:59:38 +02:00
|
|
|
return cached_matches[cached_index];
|
|
|
|
}
|
|
|
|
|
|
|
|
function highlight_match(row, search_term) {
|
|
|
|
$('table tr').removeHighlight();
|
|
|
|
row.highlight(search_term);
|
|
|
|
|
|
|
|
row = row.prev('.recipient_row');
|
|
|
|
if ((row.length !== 0) && (match_on_visible_text(row, search_term))) {
|
|
|
|
row.highlight(search_term);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-14 20:52:53 +01:00
|
|
|
exports.search_button_handler = function (reverse) {
|
2012-11-14 21:08:49 +01:00
|
|
|
var query = $('#search_query').val().toLowerCase();
|
2012-10-26 16:59:38 +02:00
|
|
|
var res = search(query, selected_message, reverse);
|
|
|
|
if (!res) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-11-01 19:23:15 +01:00
|
|
|
select_message(res);
|
2012-10-26 16:59:38 +02:00
|
|
|
highlight_match(res, query);
|
|
|
|
scroll_to_selected();
|
2012-11-14 20:52:53 +01:00
|
|
|
};
|
2012-10-26 16:59:38 +02:00
|
|
|
|
|
|
|
function clear_search_cache() {
|
|
|
|
cached_term = "";
|
|
|
|
}
|
|
|
|
|
2012-11-14 20:52:53 +01:00
|
|
|
exports.focus_search = function () {
|
2012-11-16 04:13:52 +01:00
|
|
|
if (!$('.search_button').is(':visible')) {
|
|
|
|
// Shrink the searchbox to make room for the buttons.
|
|
|
|
var search_query = $('#search_query');
|
|
|
|
var new_width = search_query.width() -
|
|
|
|
$('.search_button').outerWidth(true)*3;
|
|
|
|
search_query.width(new_width);
|
|
|
|
$("#search_arrows").addClass("input-append");
|
|
|
|
$('.search_button').show();
|
|
|
|
disable_search_arrows_if(false, ["up", "down"]);
|
|
|
|
}
|
2012-11-14 20:52:53 +01:00
|
|
|
};
|
2012-11-01 19:20:51 +01:00
|
|
|
|
2012-11-14 20:52:53 +01:00
|
|
|
exports.initiate_search = function () {
|
2012-11-14 21:08:49 +01:00
|
|
|
$('#search_query').val('').focus();
|
2012-11-14 20:52:53 +01:00
|
|
|
};
|
2012-11-01 17:14:33 +01:00
|
|
|
|
2012-11-14 20:52:53 +01:00
|
|
|
exports.clear_search = function () {
|
2012-10-26 16:59:38 +02:00
|
|
|
$('table tr').removeHighlight();
|
2012-11-15 21:40:05 +01:00
|
|
|
// Clear & reset searchbox to its normal size
|
|
|
|
$('#search_query').val('').width('');
|
2012-11-01 19:20:51 +01:00
|
|
|
$("#search_arrows").removeClass("input-append");
|
2012-11-01 23:03:10 +01:00
|
|
|
$("#search_up, #search_down").removeAttr("disabled");
|
2012-11-02 17:00:31 +01:00
|
|
|
$('.search_button').blur().hide();
|
2012-10-26 16:59:38 +02:00
|
|
|
clear_search_cache();
|
2012-11-14 20:52:53 +01:00
|
|
|
};
|
2012-10-26 16:59:38 +02:00
|
|
|
|
2012-11-14 20:52:53 +01:00
|
|
|
exports.something_is_highlighted = function () {
|
2012-10-26 16:59:38 +02:00
|
|
|
return $(".highlight").length > 0;
|
2012-11-14 20:52:53 +01:00
|
|
|
};
|
2012-10-26 16:59:38 +02:00
|
|
|
|
2012-11-14 20:52:53 +01:00
|
|
|
exports.update_highlight_on_narrow = function () {
|
2012-10-26 16:59:38 +02:00
|
|
|
highlight_match(selected_message, cached_term);
|
|
|
|
clear_search_cache();
|
2012-11-14 20:52:53 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
return exports;
|
|
|
|
|
|
|
|
}());
|