/**
* Copyright (c) 2017 Google Inc. All Rights Reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License"); you
* may not use this file except in compliance with the License. You may
* obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
* implied. See the License for the specific language governing
* permissions and limitations under the License.
*/
(function ($) {
var _inequalityRegex = '(^)(<|>|<=|>=|=)?[ ]*?[0-9]+$';
var _inequalityHint = 'e.g. 5, >0, <=10';
function _validate(input, valueSet) {
var value = input.val();
if (valueSet.has(value) || !value) {
input.removeClass('invalid');
} else {
input.addClass('invalid');
}
}
function _createInput(key, config) {
var value = config.value;
var values = config.options.corpus;
var displayName = config.displayName;
var width = config.options.width || 's4';
var div = $('<div class="input-field col"></div>');
div.addClass(width);
var input = $('<input class="filter-input"></input>');
input.attr('type', config.options.type || 'text');
input.appendTo(div);
var label = $('<label></label>').text(displayName).appendTo(div);
if (value) {
input.attr('value', value);
label.addClass('active');
}
if (config.options.validate == 'inequality') {
input.addClass('validate');
input.attr('pattern', _inequalityRegex);
input.attr('placeholder', _inequalityHint);
label.addClass('active');
}
input.focusout(function() {
config.value = input.val();
});
if (values && values.length > 0) {
var valueSet = new Set(values);
input.sizedAutocomplete({
source: values,
classes: {
'ui-autocomplete': 'card search-bar-menu'
}
});
input.focusout(function() {
_validate(input, valueSet);
});
}
if (values && values.length > 0 && value) {
_validate(input, valueSet);
}
return div;
}
function _verifyCheckboxes(checkboxes, refreshObject) {
var oneChecked = checkboxes.presubmit || checkboxes.postsubmit;
if (!oneChecked) {
refreshObject.addClass('disabled');
} else {
refreshObject.removeClass('disabled');
}
}
function _createRunTypeBoxes(checkboxes, refreshObject) {
var container = $('<div class="run-type-wrapper col s12"></div>');
var presubmit = $('<input type="checkbox" id="presubmit"></input>');
presubmit.appendTo(container);
if (checkboxes.presubmit) {
presubmit.prop('checked', true);
}
container.append('<label for="presubmit">Presubmit</label>');
var postsubmit = $('<input type="checkbox" id="postsubmit"></input>');
postsubmit.appendTo(container);
if (checkboxes.postsubmit) {
postsubmit.prop('checked', true);
}
container.append('<label for="postsubmit">Postsubmit</label>');
presubmit.change(function() {
checkboxes.presubmit = presubmit.prop('checked');
_verifyCheckboxes(checkboxes, refreshObject);
});
postsubmit.change(function() {
checkboxes.postsubmit = postsubmit.prop('checked');
_verifyCheckboxes(checkboxes, refreshObject);
});
return container;
}
function _expand(
container, filters, checkboxes, onRefreshCallback, animate=true) {
var wrapper = $('<div class="search-wrapper"></div>');
var col = $('<div class="col s9"></div>');
col.appendTo(wrapper);
Object.keys(filters).forEach(function(key) {
col.append(_createInput(key, filters[key]));
});
var refreshCol = $('<div class="col s3 refresh-wrapper"></div>');
var refresh = $('<a class="btn-floating btn-medium red right waves-effect waves-light"></a>')
.append($('<i class="medium material-icons">cached</i>'))
.appendTo(refreshCol);
refresh.click(onRefreshCallback);
refreshCol.appendTo(wrapper);
if (Object.keys(checkboxes).length > 0) {
col.append(_createRunTypeBoxes(checkboxes, refresh));
}
if (animate) {
wrapper.hide().appendTo(container).slideDown({
duration: 200,
easing: "easeOutQuart",
queue: false
});
} else {
wrapper.appendTo(container);
}
container.addClass('expanded')
}
function _renderHeader(
container, label, value, filters, checkboxes, expand, onRefreshCallback) {
var div = $('<div class="row card search-bar"></div>');
var wrapper = $('<div class="header-wrapper"></div>');
var header = $('<h5 class="section-header"></h5>');
$('<b></b>').text(label).appendTo(header);
$('<span></span>').text(value).appendTo(header);
header.appendTo(wrapper);
var iconWrapper = $('<div class="search-icon-wrapper"></div>');
$('<i class="material-icons">search</i>').appendTo(iconWrapper);
iconWrapper.appendTo(wrapper);
wrapper.appendTo(div);
if (expand) {
_expand(div, filters, checkboxes, onRefreshCallback, false);
} else {
var expanded = false;
iconWrapper.click(function() {
if (expanded) return;
expanded = true;
_expand(div, filters, checkboxes, onRefreshCallback);
});
}
div.appendTo(container);
}
function _addFilter(filters, displayName, keyName, options, defaultValue) {
filters[keyName] = {};
filters[keyName].displayName = displayName;
filters[keyName].value = defaultValue;
filters[keyName].options = options;
}
function _getOptionString(filters, checkboxes) {
var args = Object.keys(filters).reduce(function(acc, key) {
if (filters[key].value) {
return acc + '&' + key + '=' + encodeURIComponent(filters[key].value);
}
return acc;
}, '');
if (checkboxes.presubmit != undefined && checkboxes.presubmit) {
args += '&showPresubmit='
}
if (checkboxes.postsubmit != undefined && checkboxes.postsubmit) {
args += '&showPostsubmit='
}
return args;
}
/**
* Create a search header element.
* @param label The header label.
* @param value The value to display next to the label.
* @param onRefreshCallback The function to call on refresh.
*/
$.fn.createSearchHeader = function(label, value, onRefreshCallback) {
var self = $(this);
$.widget('custom.sizedAutocomplete', $.ui.autocomplete, {
_resizeMenu : function() {
this.menu.element.outerWidth($('.search-bar .filter-input').width());
}
});
var filters = {};
var checkboxes = {};
var expandOnRender = false;
var displayed = false;
return {
/**
* Add a filter to the display.
* @param displayName The input placeholder/label text.
* @param keyName The URL key to use for the filter options.
* @param options A dict of additional options (e.g. width, type).
* @param defaultValue A default filter value.
*/
addFilter : function(displayName, keyName, options, defaultValue) {
if (displayed) return;
_addFilter(filters, displayName, keyName, options, defaultValue);
if (defaultValue) expandOnRender = true;
},
/**
* Enable run type checkboxes in the filter options.
*
* This will display two checkboxes for selecting pre-/postsubmit runs.
* @param showPresubmit True if presubmit runs are selected.
* @param showPostsubmit True if postsubmit runs are selected.
*
*/
addRunTypeCheckboxes: function(showPresubmit, showPostsubmit) {
if (displayed) return;
checkboxes['presubmit'] = showPresubmit;
checkboxes['postsubmit'] = showPostsubmit;
if (!showPostsubmit || showPresubmit) {
expandOnRender = true;
}
},
/**
* Display the created search bar.
*
* This must be called after filters have been added. After displaying, no
* modifications to the filter options will take effect.
*/
display : function() {
displayed = true;
_renderHeader(
self, label, value, filters, checkboxes, expandOnRender,
onRefreshCallback);
},
/**
* Get the URL arguments string for the current set of filters.
* @returns a URI-encoded component with the search bar keys and values.
*/
args : function () {
return _getOptionString(filters, checkboxes);
}
}
}
})(jQuery);