From 18c0b666630776487e11af16e3e66c285697c5f7 Mon Sep 17 00:00:00 2001 From: Christian Fei Date: Wed, 27 Jan 2016 23:59:40 +0100 Subject: [PATCH] refactored code --- dest/jekyll-search.js | 27 ++++++++++++++----------- dest/jekyll-search.min.js | 2 +- example/js/jekyll-search.js | 35 ++++++++++++++++++++++----------- example/js/jekyll-search.min.js | 2 +- package.json | 2 +- src/index.js | 26 +++++++++++++----------- 6 files changed, 56 insertions(+), 38 deletions(-) diff --git a/dest/jekyll-search.js b/dest/jekyll-search.js index 5d306ad..2eba912 100644 --- a/dest/jekyll-search.js +++ b/dest/jekyll-search.js @@ -315,22 +315,16 @@ function compile(data){ function registerInput(){ options.searchInput.addEventListener('keyup', function(e){ - - // whitelist the following keycodes - var whitelist = [13,16,20,37,38,39,40,91]; - - // if the key pressed isn't whitelisted continue - if( whitelist.indexOf(e.which) === -1 ) { + var key = e.which + var query = e.target.value + if( isWhitelistedKey(key) && isValidQuery(query) ) { emptyResultsContainer(); - if( e.target.value.length > 0 ){ - render( repository.search(e.target.value) ); - } + render( repository.search(query) ); } - }) } - function render(results){ + function render(results) { if( results.length === 0 ){ return appendToResultsContainer(options.noResultsText) } @@ -339,8 +333,17 @@ function compile(data){ } } + function isValidQuery(query) { + return query && query.length > 0 + } + + function isWhitelistedKey(key) { + return [13,16,20,37,38,39,40,91].indexOf(key) === -1 + } + function throwError(message){ throw new Error('SimpleJekyllSearch --- '+ message) } })(window, document); + },{"./JSONLoader":1,"./OptionsValidator":2,"./Repository":3,"./Templater":6,"./utils":8}],8:[function(require,module,exports){ 'use strict' module.exports = { @@ -371,4 +374,4 @@ function isJSON(json){ } },{}]},{},[7]) -//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/home/eli/Repositories/Simple-Jekyll-Search/node_modules/gulp-browserify/node_modules/browserify/node_modules/browser-pack/_prelude.js","/home/eli/Repositories/Simple-Jekyll-Search/src/JSONLoader.js","/home/eli/Repositories/Simple-Jekyll-Search/src/OptionsValidator.js","/home/eli/Repositories/Simple-Jekyll-Search/src/Repository.js","/home/eli/Repositories/Simple-Jekyll-Search/src/SearchStrategies/FuzzySearchStrategy.js","/home/eli/Repositories/Simple-Jekyll-Search/src/SearchStrategies/LiteralSearchStrategy.js","/home/eli/Repositories/Simple-Jekyll-Search/src/Templater.js","/home/eli/Repositories/Simple-Jekyll-Search/src/fake_6064cc2a.js","/home/eli/Repositories/Simple-Jekyll-Search/src/utils.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error(\"Cannot find module '\"+o+\"'\")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","'use strict'\nmodule.exports = {\n  load: load\n}\n\nfunction load(location,callback){\n  var xhr\n  if( window.XMLHttpRequest ){\n    xhr = new XMLHttpRequest()\n  }else{\n    xhr = new ActiveXObject('Microsoft.XMLHTTP')\n  }\n\n  xhr.open('GET', location, true)\n\n  xhr.onreadystatechange = function(){\n    if ( xhr.readyState===4 && xhr.status===200 ){\n      try{\n        callback(null, JSON.parse(xhr.responseText) )\n      }catch(err){\n        callback(err, null)\n      }\n    }\n  }\n\n  xhr.send()\n}\n","'use strict'\nmodule.exports = function OptionsValidator(params){\n  if( !validateParams(params) ){\n    throw new Error('-- OptionsValidator: required options missing')\n  }\n  if( !(this instanceof OptionsValidator) ){\n    return new OptionsValidator(params)\n  }\n\n  var requiredOptions = params.required\n\n  this.getRequiredOptions = function(){\n    return requiredOptions\n  }\n\n  this.validate = function(parameters){\n    var errors = []\n    requiredOptions.forEach(function(requiredOptionName){\n      if( parameters[requiredOptionName] === undefined ){\n        errors.push(requiredOptionName)\n      }\n    })\n    return errors\n  }\n\n  function validateParams(params){\n    if( !params ) {\n      return false\n    }\n    return params.required !== undefined && params.required instanceof Array\n  }\n}","'use strict'\nmodule.exports = {\n  put:put,\n  clear: clear,\n  get: get,\n  search: search,\n  setOptions: setOptions\n}\n\nvar FuzzySearchStrategy = require('./SearchStrategies/FuzzySearchStrategy')\nvar LiteralSearchStrategy = require('./SearchStrategies/LiteralSearchStrategy')\n\nvar data = []\nvar opt = {}\nopt.fuzzy = false\nopt.limit = 10\nopt.searchStrategy = opt.fuzzy ? FuzzySearchStrategy : LiteralSearchStrategy\n\n\nfunction put(data){\n  if( isObject(data) ){\n    return addObject(data)\n  }\n  if( isArray(data) ){\n    return addArray(data)\n  }\n  return undefined\n}\nfunction clear(){\n  data.length = 0\n  return data\n}\n\nfunction get(){\n  return data\n}\n\n\nfunction isObject(obj){ return !!obj && Object.prototype.toString.call(obj) === '[object Object]' }\nfunction isArray(obj){ return !!obj && Object.prototype.toString.call(obj) === '[object Array]' }\n\nfunction addObject(_data){\n  data.push(_data)\n  return data\n}\n\nfunction addArray(_data){\n  var added = []\n  for (var i = 0; i < _data.length; i++){\n    if( isObject(_data[i]) ){\n      added.push(addObject(_data[i]))\n    }\n  }\n  return added\n}\n\n\n\nfunction search(crit){\n  if( !crit ){\n    return []\n  }\n  return findMatches(data,crit,opt.searchStrategy,opt)\n}\n\nfunction setOptions(_opt){\n  opt = _opt || {}\n\n  opt.fuzzy = _opt.fuzzy || false\n  opt.limit = _opt.limit || 10\n  opt.searchStrategy = _opt.fuzzy ? FuzzySearchStrategy : LiteralSearchStrategy\n}\n\nfunction findMatches(data,crit,strategy,opt){\n  var matches = []\n  for(var i = 0; i < data.length && matches.length < opt.limit; i++) {\n    var match = findMatchesInObject(data[i],crit,strategy,opt)\n    if( match ){\n      matches.push(match)\n    }\n  }\n  return matches\n}\n\nfunction findMatchesInObject(obj,crit,strategy,opt){\n  for(var key in obj) {\n    if( !isExcluded(obj[key], opt.exclude) && strategy.matches(obj[key], crit) ){\n      return obj\n    }\n  }\n}\n\nfunction isExcluded(term, excludedTerms){\n  var excluded = false\n  excludedTerms = excludedTerms || []\n  for (var i = 0; i<excludedTerms.length; i++) {\n    var excludedTerm = excludedTerms[i]\n    if( !excluded && new RegExp(term).test(excludedTerm) ){\n      excluded = true\n    }\n  }\n  return excluded\n}\n","'use strict'\nmodule.exports = new FuzzySearchStrategy()\n\nfunction FuzzySearchStrategy(){\n  function makeFuzzy(string){\n    string = string.split('').join('.*?')\n    string = string.replace('??','?')\n    return new RegExp( string, 'gi')\n  }\n\n  this.matches = function(string, crit){\n    if( typeof string !== 'string' || typeof crit !== 'string' ){\n      return false\n    }\n    string = string.trim()\n    return !!makeFuzzy(crit).test(string)\n  }\n}\n","'use strict'\nmodule.exports = new LiteralSearchStrategy()\n\nfunction LiteralSearchStrategy(){\n  function matchesString(string,crit){\n    return string.toLowerCase().indexOf(crit.toLowerCase()) >= 0\n  }\n\n  this.matches = function(string,crit){\n    if( typeof string !== 'string' ){\n      return false\n    }\n    string = string.trim()\n    return matchesString(string, crit)\n  }\n}\n","'use strict'\nmodule.exports = {\n  compile: compile,\n  setOptions: setOptions\n}\n\nvar options = {}\noptions.pattern = /\\{(.*?)\\}/g\noptions.template = ''\noptions.middleware = function(){}\n\nfunction setOptions(_options){\n  options.pattern = _options.pattern || options.pattern\n  options.template = _options.template || options.template\n  if( typeof _options.middleware === 'function' ){\n    options.middleware = _options.middleware\n  }\n}\n\nfunction compile(data){\n  return options.template.replace(options.pattern, function(match, prop) {\n    var value = options.middleware(prop, data[prop], options.template)\n    if( value !== undefined ){\n      return value\n    }\n    return data[prop] || match\n  })\n}\n",";(function(window, document, undefined){\n  'use strict'\n\n  var options = {\n    searchInput: null,\n    resultsContainer: null,\n    json: [],\n    searchResultTemplate: '<li><a href=\"{url}\" title=\"{desc}\">{title}</a></li>',\n    templateMiddleware: function(){},\n    noResultsText: 'No results found',\n    limit: 10,\n    fuzzy: false,\n    exclude: []\n  }\n\n  var requiredOptions = ['searchInput','resultsContainer','json']\n\n  var templater = require('./Templater')\n  var repository = require('./Repository')\n  var jsonLoader = require('./JSONLoader')\n  var optionsValidator = require('./OptionsValidator')({\n    required: requiredOptions\n  })\n  var utils = require('./utils')\n\n  /*\n    Public API\n  */\n  window.SimpleJekyllSearch = function SimpleJekyllSearch(_options){\n    var errors = optionsValidator.validate(_options)\n    if( errors.length > 0 ){\n      throwError('You must specify the following required options: ' + requiredOptions)\n    }\n\n    options = utils.merge(options, _options)\n\n    templater.setOptions({\n      template: options.searchResultTemplate,\n      middleware: options.templateMiddleware,\n    })\n\n    repository.setOptions({\n      fuzzy: options.fuzzy,\n      limit: options.limit,\n    })\n\n    if( utils.isJSON(options.json) ){\n      initWithJSON(options.json)\n    }else{\n      initWithURL(options.json)\n    }\n  }\n\n  // for backwards compatibility\n  window.SimpleJekyllSearch.init = window.SimpleJekyllSearch\n\n\n  function initWithJSON(json){\n    repository.put(json)\n    registerInput()\n  }\n\n  function initWithURL(url){\n    jsonLoader.load(url, function(err,json){\n      if( err ){\n        throwError('failed to get JSON (' + url + ')')\n      }\n      initWithJSON(json)\n    })\n  }\n\n  function emptyResultsContainer(){\n    options.resultsContainer.innerHTML = ''\n  }\n\n  function appendToResultsContainer(text){\n    options.resultsContainer.innerHTML += text\n  }\n\n  function registerInput(){\n    options.searchInput.addEventListener('keyup', function(e){\n\n      // whitelist the following keycodes\n      var whitelist = [13,16,20,37,38,39,40,91];\n\n      // if the key pressed isn't whitelisted continue\n      if( whitelist.indexOf(e.which) === -1 ) {\n        emptyResultsContainer();\n        if( e.target.value.length > 0 ){\n          render( repository.search(e.target.value) );\n        }\n      }\n\n    })\n  }\n\n  function render(results){\n    if( results.length === 0 ){\n      return appendToResultsContainer(options.noResultsText)\n    }\n    for (var i = 0; i < results.length; i++) {\n      appendToResultsContainer( templater.compile(results[i]) )\n    }\n  }\n\n  function throwError(message){ throw new Error('SimpleJekyllSearch --- '+ message) }\n})(window, document);","'use strict'\nmodule.exports = {\n  merge: merge,\n  isJSON: isJSON,\n}\n\nfunction merge(defaultParams, mergeParams){\n  var mergedOptions = {}\n  for(var option in defaultParams){\n    mergedOptions[option] = defaultParams[option]\n    if( mergeParams[option] !== undefined ){\n      mergedOptions[option] = mergeParams[option]\n    }\n  }\n  return mergedOptions\n}\n\nfunction isJSON(json){\n  try{\n    if( json instanceof Object && JSON.parse(JSON.stringify(json)) ){\n      return true\n    }\n    return false\n  }catch(e){\n    return false\n  }\n}\n"]} +//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/Users/saiph/Documents/playground/Simple-Jekyll-Search/node_modules/browser-pack/_prelude.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/JSONLoader.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/OptionsValidator.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/Repository.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/SearchStrategies/FuzzySearchStrategy.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/SearchStrategies/LiteralSearchStrategy.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/Templater.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/fake_1055537f.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/utils.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error(\"Cannot find module '\"+o+\"'\")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","'use strict'\nmodule.exports = {\n  load: load\n}\n\nfunction load(location,callback){\n  var xhr\n  if( window.XMLHttpRequest ){\n    xhr = new XMLHttpRequest()\n  }else{\n    xhr = new ActiveXObject('Microsoft.XMLHTTP')\n  }\n\n  xhr.open('GET', location, true)\n\n  xhr.onreadystatechange = function(){\n    if ( xhr.readyState===4 && xhr.status===200 ){\n      try{\n        callback(null, JSON.parse(xhr.responseText) )\n      }catch(err){\n        callback(err, null)\n      }\n    }\n  }\n\n  xhr.send()\n}\n","'use strict'\nmodule.exports = function OptionsValidator(params){\n  if( !validateParams(params) ){\n    throw new Error('-- OptionsValidator: required options missing')\n  }\n  if( !(this instanceof OptionsValidator) ){\n    return new OptionsValidator(params)\n  }\n\n  var requiredOptions = params.required\n\n  this.getRequiredOptions = function(){\n    return requiredOptions\n  }\n\n  this.validate = function(parameters){\n    var errors = []\n    requiredOptions.forEach(function(requiredOptionName){\n      if( parameters[requiredOptionName] === undefined ){\n        errors.push(requiredOptionName)\n      }\n    })\n    return errors\n  }\n\n  function validateParams(params){\n    if( !params ) {\n      return false\n    }\n    return params.required !== undefined && params.required instanceof Array\n  }\n}","'use strict'\nmodule.exports = {\n  put:put,\n  clear: clear,\n  get: get,\n  search: search,\n  setOptions: setOptions\n}\n\nvar FuzzySearchStrategy = require('./SearchStrategies/FuzzySearchStrategy')\nvar LiteralSearchStrategy = require('./SearchStrategies/LiteralSearchStrategy')\n\nvar data = []\nvar opt = {}\nopt.fuzzy = false\nopt.limit = 10\nopt.searchStrategy = opt.fuzzy ? FuzzySearchStrategy : LiteralSearchStrategy\n\n\nfunction put(data){\n  if( isObject(data) ){\n    return addObject(data)\n  }\n  if( isArray(data) ){\n    return addArray(data)\n  }\n  return undefined\n}\nfunction clear(){\n  data.length = 0\n  return data\n}\n\nfunction get(){\n  return data\n}\n\n\nfunction isObject(obj){ return !!obj && Object.prototype.toString.call(obj) === '[object Object]' }\nfunction isArray(obj){ return !!obj && Object.prototype.toString.call(obj) === '[object Array]' }\n\nfunction addObject(_data){\n  data.push(_data)\n  return data\n}\n\nfunction addArray(_data){\n  var added = []\n  for (var i = 0; i < _data.length; i++){\n    if( isObject(_data[i]) ){\n      added.push(addObject(_data[i]))\n    }\n  }\n  return added\n}\n\n\n\nfunction search(crit){\n  if( !crit ){\n    return []\n  }\n  return findMatches(data,crit,opt.searchStrategy,opt)\n}\n\nfunction setOptions(_opt){\n  opt = _opt || {}\n\n  opt.fuzzy = _opt.fuzzy || false\n  opt.limit = _opt.limit || 10\n  opt.searchStrategy = _opt.fuzzy ? FuzzySearchStrategy : LiteralSearchStrategy\n}\n\nfunction findMatches(data,crit,strategy,opt){\n  var matches = []\n  for(var i = 0; i < data.length && matches.length < opt.limit; i++) {\n    var match = findMatchesInObject(data[i],crit,strategy,opt)\n    if( match ){\n      matches.push(match)\n    }\n  }\n  return matches\n}\n\nfunction findMatchesInObject(obj,crit,strategy,opt){\n  for(var key in obj) {\n    if( !isExcluded(obj[key], opt.exclude) && strategy.matches(obj[key], crit) ){\n      return obj\n    }\n  }\n}\n\nfunction isExcluded(term, excludedTerms){\n  var excluded = false\n  excludedTerms = excludedTerms || []\n  for (var i = 0; i<excludedTerms.length; i++) {\n    var excludedTerm = excludedTerms[i]\n    if( !excluded && new RegExp(term).test(excludedTerm) ){\n      excluded = true\n    }\n  }\n  return excluded\n}\n","'use strict'\nmodule.exports = new FuzzySearchStrategy()\n\nfunction FuzzySearchStrategy(){\n  function makeFuzzy(string){\n    string = string.split('').join('.*?')\n    string = string.replace('??','?')\n    return new RegExp( string, 'gi')\n  }\n\n  this.matches = function(string, crit){\n    if( typeof string !== 'string' || typeof crit !== 'string' ){\n      return false\n    }\n    string = string.trim()\n    return !!makeFuzzy(crit).test(string)\n  }\n}\n","'use strict'\nmodule.exports = new LiteralSearchStrategy()\n\nfunction LiteralSearchStrategy(){\n  function matchesString(string,crit){\n    return string.toLowerCase().indexOf(crit.toLowerCase()) >= 0\n  }\n\n  this.matches = function(string,crit){\n    if( typeof string !== 'string' ){\n      return false\n    }\n    string = string.trim()\n    return matchesString(string, crit)\n  }\n}\n","'use strict'\nmodule.exports = {\n  compile: compile,\n  setOptions: setOptions\n}\n\nvar options = {}\noptions.pattern = /\\{(.*?)\\}/g\noptions.template = ''\noptions.middleware = function(){}\n\nfunction setOptions(_options){\n  options.pattern = _options.pattern || options.pattern\n  options.template = _options.template || options.template\n  if( typeof _options.middleware === 'function' ){\n    options.middleware = _options.middleware\n  }\n}\n\nfunction compile(data){\n  return options.template.replace(options.pattern, function(match, prop) {\n    var value = options.middleware(prop, data[prop], options.template)\n    if( value !== undefined ){\n      return value\n    }\n    return data[prop] || match\n  })\n}\n",";(function(window, document, undefined){\n  'use strict'\n\n  var options = {\n    searchInput: null,\n    resultsContainer: null,\n    json: [],\n    searchResultTemplate: '<li><a href=\"{url}\" title=\"{desc}\">{title}</a></li>',\n    templateMiddleware: function(){},\n    noResultsText: 'No results found',\n    limit: 10,\n    fuzzy: false,\n    exclude: []\n  }\n\n  var requiredOptions = ['searchInput','resultsContainer','json']\n\n  var templater = require('./Templater')\n  var repository = require('./Repository')\n  var jsonLoader = require('./JSONLoader')\n  var optionsValidator = require('./OptionsValidator')({\n    required: requiredOptions\n  })\n  var utils = require('./utils')\n\n  /*\n    Public API\n  */\n  window.SimpleJekyllSearch = function SimpleJekyllSearch(_options){\n    var errors = optionsValidator.validate(_options)\n    if( errors.length > 0 ){\n      throwError('You must specify the following required options: ' + requiredOptions)\n    }\n\n    options = utils.merge(options, _options)\n\n    templater.setOptions({\n      template: options.searchResultTemplate,\n      middleware: options.templateMiddleware,\n    })\n\n    repository.setOptions({\n      fuzzy: options.fuzzy,\n      limit: options.limit,\n    })\n\n    if( utils.isJSON(options.json) ){\n      initWithJSON(options.json)\n    }else{\n      initWithURL(options.json)\n    }\n  }\n\n  // for backwards compatibility\n  window.SimpleJekyllSearch.init = window.SimpleJekyllSearch\n\n\n  function initWithJSON(json){\n    repository.put(json)\n    registerInput()\n  }\n\n  function initWithURL(url){\n    jsonLoader.load(url, function(err,json){\n      if( err ){\n        throwError('failed to get JSON (' + url + ')')\n      }\n      initWithJSON(json)\n    })\n  }\n\n  function emptyResultsContainer(){\n    options.resultsContainer.innerHTML = ''\n  }\n\n  function appendToResultsContainer(text){\n    options.resultsContainer.innerHTML += text\n  }\n\n  function registerInput(){\n    options.searchInput.addEventListener('keyup', function(e){\n      var key = e.which\n      var query = e.target.value\n      if( isWhitelistedKey(key) && isValidQuery(query) ) {\n        emptyResultsContainer();\n        render( repository.search(query) );\n      }\n    })\n  }\n\n  function render(results) {\n    if( results.length === 0 ){\n      return appendToResultsContainer(options.noResultsText)\n    }\n    for (var i = 0; i < results.length; i++) {\n      appendToResultsContainer( templater.compile(results[i]) )\n    }\n  }\n\n  function isValidQuery(query) {\n    return query && query.length > 0\n  }\n\n  function isWhitelistedKey(key) {\n    return [13,16,20,37,38,39,40,91].indexOf(key) === -1\n  }\n\n  function throwError(message){ throw new Error('SimpleJekyllSearch --- '+ message) }\n})(window, document);\n","'use strict'\nmodule.exports = {\n  merge: merge,\n  isJSON: isJSON,\n}\n\nfunction merge(defaultParams, mergeParams){\n  var mergedOptions = {}\n  for(var option in defaultParams){\n    mergedOptions[option] = defaultParams[option]\n    if( mergeParams[option] !== undefined ){\n      mergedOptions[option] = mergeParams[option]\n    }\n  }\n  return mergedOptions\n}\n\nfunction isJSON(json){\n  try{\n    if( json instanceof Object && JSON.parse(JSON.stringify(json)) ){\n      return true\n    }\n    return false\n  }catch(e){\n    return false\n  }\n}\n"]} diff --git a/dest/jekyll-search.min.js b/dest/jekyll-search.min.js index 5405217..c4d5ad8 100644 --- a/dest/jekyll-search.min.js +++ b/dest/jekyll-search.min.js @@ -1 +1 @@ -!function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a="function"==typeof require&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error("Cannot find module '"+o+"'")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}for(var i="function"==typeof require&&require,o=0;o=0}this.matches=function(string,crit){return"string"!=typeof string?!1:(string=string.trim(),matchesString(string,crit))}}module.exports=new LiteralSearchStrategy},{}],6:[function(require,module,exports){"use strict";function setOptions(_options){options.pattern=_options.pattern||options.pattern,options.template=_options.template||options.template,"function"==typeof _options.middleware&&(options.middleware=_options.middleware)}function compile(data){return options.template.replace(options.pattern,function(match,prop){var value=options.middleware(prop,data[prop],options.template);return void 0!==value?value:data[prop]||match})}module.exports={compile:compile,setOptions:setOptions};var options={};options.pattern=/\{(.*?)\}/g,options.template="",options.middleware=function(){}},{}],7:[function(require,module,exports){!function(window,document,undefined){"use strict";function initWithJSON(json){repository.put(json),registerInput()}function initWithURL(url){jsonLoader.load(url,function(err,json){err&&throwError("failed to get JSON ("+url+")"),initWithJSON(json)})}function emptyResultsContainer(){options.resultsContainer.innerHTML=""}function appendToResultsContainer(text){options.resultsContainer.innerHTML+=text}function registerInput(){options.searchInput.addEventListener("keyup",function(e){var whitelist=[13,16,20,37,38,39,40,91];-1===whitelist.indexOf(e.which)&&(emptyResultsContainer(),e.target.value.length>0&&render(repository.search(e.target.value)))})}function render(results){if(0===results.length)return appendToResultsContainer(options.noResultsText);for(var i=0;i{title}',templateMiddleware:function(){},noResultsText:"No results found",limit:10,fuzzy:!1,exclude:[]},requiredOptions=["searchInput","resultsContainer","json"],templater=require("./Templater"),repository=require("./Repository"),jsonLoader=require("./JSONLoader"),optionsValidator=require("./OptionsValidator")({required:requiredOptions}),utils=require("./utils");window.SimpleJekyllSearch=function(_options){var errors=optionsValidator.validate(_options);errors.length>0&&throwError("You must specify the following required options: "+requiredOptions),options=utils.merge(options,_options),templater.setOptions({template:options.searchResultTemplate,middleware:options.templateMiddleware}),repository.setOptions({fuzzy:options.fuzzy,limit:options.limit}),utils.isJSON(options.json)?initWithJSON(options.json):initWithURL(options.json)},window.SimpleJekyllSearch.init=window.SimpleJekyllSearch}(window,document)},{"./JSONLoader":1,"./OptionsValidator":2,"./Repository":3,"./Templater":6,"./utils":8}],8:[function(require,module,exports){"use strict";function merge(defaultParams,mergeParams){var mergedOptions={};for(var option in defaultParams)mergedOptions[option]=defaultParams[option],void 0!==mergeParams[option]&&(mergedOptions[option]=mergeParams[option]);return mergedOptions}function isJSON(json){try{return json instanceof Object&&JSON.parse(JSON.stringify(json))?!0:!1}catch(e){return!1}}module.exports={merge:merge,isJSON:isJSON}},{}]},{},[7]); \ No newline at end of file +!function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a="function"==typeof require&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error("Cannot find module '"+o+"'")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}for(var i="function"==typeof require&&require,o=0;o=0}this.matches=function(string,crit){return"string"!=typeof string?!1:(string=string.trim(),matchesString(string,crit))}}module.exports=new LiteralSearchStrategy},{}],6:[function(require,module,exports){"use strict";function setOptions(_options){options.pattern=_options.pattern||options.pattern,options.template=_options.template||options.template,"function"==typeof _options.middleware&&(options.middleware=_options.middleware)}function compile(data){return options.template.replace(options.pattern,function(match,prop){var value=options.middleware(prop,data[prop],options.template);return void 0!==value?value:data[prop]||match})}module.exports={compile:compile,setOptions:setOptions};var options={};options.pattern=/\{(.*?)\}/g,options.template="",options.middleware=function(){}},{}],7:[function(require,module,exports){!function(window,document,undefined){"use strict";function initWithJSON(json){repository.put(json),registerInput()}function initWithURL(url){jsonLoader.load(url,function(err,json){err&&throwError("failed to get JSON ("+url+")"),initWithJSON(json)})}function emptyResultsContainer(){options.resultsContainer.innerHTML=""}function appendToResultsContainer(text){options.resultsContainer.innerHTML+=text}function registerInput(){options.searchInput.addEventListener("keyup",function(e){var key=e.which,query=e.target.value;isWhitelistedKey(key)&&isValidQuery(query)&&(emptyResultsContainer(),render(repository.search(query)))})}function render(results){if(0===results.length)return appendToResultsContainer(options.noResultsText);for(var i=0;i0}function isWhitelistedKey(key){return-1===[13,16,20,37,38,39,40,91].indexOf(key)}function throwError(message){throw new Error("SimpleJekyllSearch --- "+message)}var options={searchInput:null,resultsContainer:null,json:[],searchResultTemplate:'
  • {title}
  • ',templateMiddleware:function(){},noResultsText:"No results found",limit:10,fuzzy:!1,exclude:[]},requiredOptions=["searchInput","resultsContainer","json"],templater=require("./Templater"),repository=require("./Repository"),jsonLoader=require("./JSONLoader"),optionsValidator=require("./OptionsValidator")({required:requiredOptions}),utils=require("./utils");window.SimpleJekyllSearch=function(_options){var errors=optionsValidator.validate(_options);errors.length>0&&throwError("You must specify the following required options: "+requiredOptions),options=utils.merge(options,_options),templater.setOptions({template:options.searchResultTemplate,middleware:options.templateMiddleware}),repository.setOptions({fuzzy:options.fuzzy,limit:options.limit}),utils.isJSON(options.json)?initWithJSON(options.json):initWithURL(options.json)},window.SimpleJekyllSearch.init=window.SimpleJekyllSearch}(window,document)},{"./JSONLoader":1,"./OptionsValidator":2,"./Repository":3,"./Templater":6,"./utils":8}],8:[function(require,module,exports){"use strict";function merge(defaultParams,mergeParams){var mergedOptions={};for(var option in defaultParams)mergedOptions[option]=defaultParams[option],void 0!==mergeParams[option]&&(mergedOptions[option]=mergeParams[option]);return mergedOptions}function isJSON(json){try{return json instanceof Object&&JSON.parse(JSON.stringify(json))?!0:!1}catch(e){return!1}}module.exports={merge:merge,isJSON:isJSON}},{}]},{},[7]); \ No newline at end of file diff --git a/example/js/jekyll-search.js b/example/js/jekyll-search.js index d3a3d77..2eba912 100644 --- a/example/js/jekyll-search.js +++ b/example/js/jekyll-search.js @@ -15,7 +15,7 @@ function load(location,callback){ xhr.open('GET', location, true) xhr.onreadystatechange = function(){ - if ( xhr.status===200 && xhr.readyState===4 ){ + if ( xhr.readyState===4 && xhr.status===200 ){ try{ callback(null, JSON.parse(xhr.responseText) ) }catch(err){ @@ -170,16 +170,18 @@ function isExcluded(term, excludedTerms){ module.exports = new FuzzySearchStrategy() function FuzzySearchStrategy(){ - function fuzzyRegexFromString(string){ - return new RegExp( string.split('').join('.*?'), 'gi') + function makeFuzzy(string){ + string = string.split('').join('.*?') + string = string.replace('??','?') + return new RegExp( string, 'gi') } - this.matches = function(string,crit){ - if( typeof string !== 'string' ){ + this.matches = function(string, crit){ + if( typeof string !== 'string' || typeof crit !== 'string' ){ return false } string = string.trim() - return !!fuzzyRegexFromString(crit).test(string) + return !!makeFuzzy(crit).test(string) } } @@ -313,14 +315,16 @@ function compile(data){ function registerInput(){ options.searchInput.addEventListener('keyup', function(e){ - emptyResultsContainer() - if( e.target.value.length > 0 ){ - render( repository.search(e.target.value) ) + var key = e.which + var query = e.target.value + if( isWhitelistedKey(key) && isValidQuery(query) ) { + emptyResultsContainer(); + render( repository.search(query) ); } }) } - function render(results){ + function render(results) { if( results.length === 0 ){ return appendToResultsContainer(options.noResultsText) } @@ -329,8 +333,17 @@ function compile(data){ } } + function isValidQuery(query) { + return query && query.length > 0 + } + + function isWhitelistedKey(key) { + return [13,16,20,37,38,39,40,91].indexOf(key) === -1 + } + function throwError(message){ throw new Error('SimpleJekyllSearch --- '+ message) } })(window, document); + },{"./JSONLoader":1,"./OptionsValidator":2,"./Repository":3,"./Templater":6,"./utils":8}],8:[function(require,module,exports){ 'use strict' module.exports = { @@ -361,4 +374,4 @@ function isJSON(json){ } },{}]},{},[7]) -//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/Users/saiph/Documents/playground/Simple-Jekyll-Search/node_modules/gulp-browserify/node_modules/browserify/node_modules/browser-pack/_prelude.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/JSONLoader.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/OptionsValidator.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/Repository.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/SearchStrategies/FuzzySearchStrategy.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/SearchStrategies/LiteralSearchStrategy.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/Templater.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/fake_7751ebf0.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/utils.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error(\"Cannot find module '\"+o+\"'\")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","'use strict'\nmodule.exports = {\n  load: load\n}\n\nfunction load(location,callback){\n  var xhr\n  if( window.XMLHttpRequest ){\n    xhr = new XMLHttpRequest()\n  }else{\n    xhr = new ActiveXObject('Microsoft.XMLHTTP')\n  }\n\n  xhr.open('GET', location, true)\n\n  xhr.onreadystatechange = function(){\n    if ( xhr.status===200 && xhr.readyState===4 ){\n      try{\n        callback(null, JSON.parse(xhr.responseText) )\n      }catch(err){\n        callback(err, null)\n      }\n    }\n  }\n\n  xhr.send()\n}\n","'use strict'\nmodule.exports = function OptionsValidator(params){\n  if( !validateParams(params) ){\n    throw new Error('-- OptionsValidator: required options missing')\n  }\n  if( !(this instanceof OptionsValidator) ){\n    return new OptionsValidator(params)\n  }\n\n  var requiredOptions = params.required\n\n  this.getRequiredOptions = function(){\n    return requiredOptions\n  }\n\n  this.validate = function(parameters){\n    var errors = []\n    requiredOptions.forEach(function(requiredOptionName){\n      if( parameters[requiredOptionName] === undefined ){\n        errors.push(requiredOptionName)\n      }\n    })\n    return errors\n  }\n\n  function validateParams(params){\n    if( !params ) {\n      return false\n    }\n    return params.required !== undefined && params.required instanceof Array\n  }\n}","'use strict'\nmodule.exports = {\n  put:put,\n  clear: clear,\n  get: get,\n  search: search,\n  setOptions: setOptions\n}\n\nvar FuzzySearchStrategy = require('./SearchStrategies/FuzzySearchStrategy')\nvar LiteralSearchStrategy = require('./SearchStrategies/LiteralSearchStrategy')\n\nvar data = []\nvar opt = {}\nopt.fuzzy = false\nopt.limit = 10\nopt.searchStrategy = opt.fuzzy ? FuzzySearchStrategy : LiteralSearchStrategy\n\n\nfunction put(data){\n  if( isObject(data) ){\n    return addObject(data)\n  }\n  if( isArray(data) ){\n    return addArray(data)\n  }\n  return undefined\n}\nfunction clear(){\n  data.length = 0\n  return data\n}\n\nfunction get(){\n  return data\n}\n\n\nfunction isObject(obj){ return !!obj && Object.prototype.toString.call(obj) === '[object Object]' }\nfunction isArray(obj){ return !!obj && Object.prototype.toString.call(obj) === '[object Array]' }\n\nfunction addObject(_data){\n  data.push(_data)\n  return data\n}\n\nfunction addArray(_data){\n  var added = []\n  for (var i = 0; i < _data.length; i++){\n    if( isObject(_data[i]) ){\n      added.push(addObject(_data[i]))\n    }\n  }\n  return added\n}\n\n\n\nfunction search(crit){\n  if( !crit ){\n    return []\n  }\n  return findMatches(data,crit,opt.searchStrategy,opt)\n}\n\nfunction setOptions(_opt){\n  opt = _opt || {}\n\n  opt.fuzzy = _opt.fuzzy || false\n  opt.limit = _opt.limit || 10\n  opt.searchStrategy = _opt.fuzzy ? FuzzySearchStrategy : LiteralSearchStrategy\n}\n\nfunction findMatches(data,crit,strategy,opt){\n  var matches = []\n  for(var i = 0; i < data.length && matches.length < opt.limit; i++) {\n    var match = findMatchesInObject(data[i],crit,strategy,opt)\n    if( match ){\n      matches.push(match)\n    }\n  }\n  return matches\n}\n\nfunction findMatchesInObject(obj,crit,strategy,opt){\n  for(var key in obj) {\n    if( !isExcluded(obj[key], opt.exclude) && strategy.matches(obj[key], crit) ){\n      return obj\n    }\n  }\n}\n\nfunction isExcluded(term, excludedTerms){\n  var excluded = false\n  excludedTerms = excludedTerms || []\n  for (var i = 0; i<excludedTerms.length; i++) {\n    var excludedTerm = excludedTerms[i]\n    if( !excluded && new RegExp(term).test(excludedTerm) ){\n      excluded = true\n    }\n  }\n  return excluded\n}\n","'use strict'\nmodule.exports = new FuzzySearchStrategy()\n\nfunction FuzzySearchStrategy(){\n  function fuzzyRegexFromString(string){\n    return new RegExp( string.split('').join('.*?'), 'gi')\n  }\n\n  this.matches = function(string,crit){\n    if( typeof string !== 'string' ){\n      return false\n    }\n    string = string.trim()\n    return !!fuzzyRegexFromString(crit).test(string)\n  }\n}\n","'use strict'\nmodule.exports = new LiteralSearchStrategy()\n\nfunction LiteralSearchStrategy(){\n  function matchesString(string,crit){\n    return string.toLowerCase().indexOf(crit.toLowerCase()) >= 0\n  }\n\n  this.matches = function(string,crit){\n    if( typeof string !== 'string' ){\n      return false\n    }\n    string = string.trim()\n    return matchesString(string, crit)\n  }\n}\n","'use strict'\nmodule.exports = {\n  compile: compile,\n  setOptions: setOptions\n}\n\nvar options = {}\noptions.pattern = /\\{(.*?)\\}/g\noptions.template = ''\noptions.middleware = function(){}\n\nfunction setOptions(_options){\n  options.pattern = _options.pattern || options.pattern\n  options.template = _options.template || options.template\n  if( typeof _options.middleware === 'function' ){\n    options.middleware = _options.middleware\n  }\n}\n\nfunction compile(data){\n  return options.template.replace(options.pattern, function(match, prop) {\n    var value = options.middleware(prop, data[prop], options.template)\n    if( value !== undefined ){\n      return value\n    }\n    return data[prop] || match\n  })\n}\n",";(function(window, document, undefined){\n  'use strict'\n\n  var options = {\n    searchInput: null,\n    resultsContainer: null,\n    json: [],\n    searchResultTemplate: '<li><a href=\"{url}\" title=\"{desc}\">{title}</a></li>',\n    templateMiddleware: function(){},\n    noResultsText: 'No results found',\n    limit: 10,\n    fuzzy: false,\n    exclude: []\n  }\n\n  var requiredOptions = ['searchInput','resultsContainer','json']\n\n  var templater = require('./Templater')\n  var repository = require('./Repository')\n  var jsonLoader = require('./JSONLoader')\n  var optionsValidator = require('./OptionsValidator')({\n    required: requiredOptions\n  })\n  var utils = require('./utils')\n\n  /*\n    Public API\n  */\n  window.SimpleJekyllSearch = function SimpleJekyllSearch(_options){\n    var errors = optionsValidator.validate(_options)\n    if( errors.length > 0 ){\n      throwError('You must specify the following required options: ' + requiredOptions)\n    }\n\n    options = utils.merge(options, _options)\n\n    templater.setOptions({\n      template: options.searchResultTemplate,\n      middleware: options.templateMiddleware,\n    })\n\n    repository.setOptions({\n      fuzzy: options.fuzzy,\n      limit: options.limit,\n    })\n\n    if( utils.isJSON(options.json) ){\n      initWithJSON(options.json)\n    }else{\n      initWithURL(options.json)\n    }\n  }\n\n  // for backwards compatibility\n  window.SimpleJekyllSearch.init = window.SimpleJekyllSearch\n\n\n  function initWithJSON(json){\n    repository.put(json)\n    registerInput()\n  }\n\n  function initWithURL(url){\n    jsonLoader.load(url, function(err,json){\n      if( err ){\n        throwError('failed to get JSON (' + url + ')')\n      }\n      initWithJSON(json)\n    })\n  }\n\n  function emptyResultsContainer(){\n    options.resultsContainer.innerHTML = ''\n  }\n\n  function appendToResultsContainer(text){\n    options.resultsContainer.innerHTML += text\n  }\n\n  function registerInput(){\n    options.searchInput.addEventListener('keyup', function(e){\n      emptyResultsContainer()\n      if( e.target.value.length > 0 ){\n        render( repository.search(e.target.value) )\n      }\n    })\n  }\n\n  function render(results){\n    if( results.length === 0 ){\n      return appendToResultsContainer(options.noResultsText)\n    }\n    for (var i = 0; i < results.length; i++) {\n      appendToResultsContainer( templater.compile(results[i]) )\n    }\n  }\n\n  function throwError(message){ throw new Error('SimpleJekyllSearch --- '+ message) }\n})(window, document);","'use strict'\nmodule.exports = {\n  merge: merge,\n  isJSON: isJSON,\n}\n\nfunction merge(defaultParams, mergeParams){\n  var mergedOptions = {}\n  for(var option in defaultParams){\n    mergedOptions[option] = defaultParams[option]\n    if( mergeParams[option] !== undefined ){\n      mergedOptions[option] = mergeParams[option]\n    }\n  }\n  return mergedOptions\n}\n\nfunction isJSON(json){\n  try{\n    if( json instanceof Object && JSON.parse(JSON.stringify(json)) ){\n      return true\n    }\n    return false\n  }catch(e){\n    return false\n  }\n}\n"]} +//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/Users/saiph/Documents/playground/Simple-Jekyll-Search/node_modules/browser-pack/_prelude.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/JSONLoader.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/OptionsValidator.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/Repository.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/SearchStrategies/FuzzySearchStrategy.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/SearchStrategies/LiteralSearchStrategy.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/Templater.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/fake_1055537f.js","/Users/saiph/Documents/playground/Simple-Jekyll-Search/src/utils.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error(\"Cannot find module '\"+o+\"'\")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","'use strict'\nmodule.exports = {\n  load: load\n}\n\nfunction load(location,callback){\n  var xhr\n  if( window.XMLHttpRequest ){\n    xhr = new XMLHttpRequest()\n  }else{\n    xhr = new ActiveXObject('Microsoft.XMLHTTP')\n  }\n\n  xhr.open('GET', location, true)\n\n  xhr.onreadystatechange = function(){\n    if ( xhr.readyState===4 && xhr.status===200 ){\n      try{\n        callback(null, JSON.parse(xhr.responseText) )\n      }catch(err){\n        callback(err, null)\n      }\n    }\n  }\n\n  xhr.send()\n}\n","'use strict'\nmodule.exports = function OptionsValidator(params){\n  if( !validateParams(params) ){\n    throw new Error('-- OptionsValidator: required options missing')\n  }\n  if( !(this instanceof OptionsValidator) ){\n    return new OptionsValidator(params)\n  }\n\n  var requiredOptions = params.required\n\n  this.getRequiredOptions = function(){\n    return requiredOptions\n  }\n\n  this.validate = function(parameters){\n    var errors = []\n    requiredOptions.forEach(function(requiredOptionName){\n      if( parameters[requiredOptionName] === undefined ){\n        errors.push(requiredOptionName)\n      }\n    })\n    return errors\n  }\n\n  function validateParams(params){\n    if( !params ) {\n      return false\n    }\n    return params.required !== undefined && params.required instanceof Array\n  }\n}","'use strict'\nmodule.exports = {\n  put:put,\n  clear: clear,\n  get: get,\n  search: search,\n  setOptions: setOptions\n}\n\nvar FuzzySearchStrategy = require('./SearchStrategies/FuzzySearchStrategy')\nvar LiteralSearchStrategy = require('./SearchStrategies/LiteralSearchStrategy')\n\nvar data = []\nvar opt = {}\nopt.fuzzy = false\nopt.limit = 10\nopt.searchStrategy = opt.fuzzy ? FuzzySearchStrategy : LiteralSearchStrategy\n\n\nfunction put(data){\n  if( isObject(data) ){\n    return addObject(data)\n  }\n  if( isArray(data) ){\n    return addArray(data)\n  }\n  return undefined\n}\nfunction clear(){\n  data.length = 0\n  return data\n}\n\nfunction get(){\n  return data\n}\n\n\nfunction isObject(obj){ return !!obj && Object.prototype.toString.call(obj) === '[object Object]' }\nfunction isArray(obj){ return !!obj && Object.prototype.toString.call(obj) === '[object Array]' }\n\nfunction addObject(_data){\n  data.push(_data)\n  return data\n}\n\nfunction addArray(_data){\n  var added = []\n  for (var i = 0; i < _data.length; i++){\n    if( isObject(_data[i]) ){\n      added.push(addObject(_data[i]))\n    }\n  }\n  return added\n}\n\n\n\nfunction search(crit){\n  if( !crit ){\n    return []\n  }\n  return findMatches(data,crit,opt.searchStrategy,opt)\n}\n\nfunction setOptions(_opt){\n  opt = _opt || {}\n\n  opt.fuzzy = _opt.fuzzy || false\n  opt.limit = _opt.limit || 10\n  opt.searchStrategy = _opt.fuzzy ? FuzzySearchStrategy : LiteralSearchStrategy\n}\n\nfunction findMatches(data,crit,strategy,opt){\n  var matches = []\n  for(var i = 0; i < data.length && matches.length < opt.limit; i++) {\n    var match = findMatchesInObject(data[i],crit,strategy,opt)\n    if( match ){\n      matches.push(match)\n    }\n  }\n  return matches\n}\n\nfunction findMatchesInObject(obj,crit,strategy,opt){\n  for(var key in obj) {\n    if( !isExcluded(obj[key], opt.exclude) && strategy.matches(obj[key], crit) ){\n      return obj\n    }\n  }\n}\n\nfunction isExcluded(term, excludedTerms){\n  var excluded = false\n  excludedTerms = excludedTerms || []\n  for (var i = 0; i<excludedTerms.length; i++) {\n    var excludedTerm = excludedTerms[i]\n    if( !excluded && new RegExp(term).test(excludedTerm) ){\n      excluded = true\n    }\n  }\n  return excluded\n}\n","'use strict'\nmodule.exports = new FuzzySearchStrategy()\n\nfunction FuzzySearchStrategy(){\n  function makeFuzzy(string){\n    string = string.split('').join('.*?')\n    string = string.replace('??','?')\n    return new RegExp( string, 'gi')\n  }\n\n  this.matches = function(string, crit){\n    if( typeof string !== 'string' || typeof crit !== 'string' ){\n      return false\n    }\n    string = string.trim()\n    return !!makeFuzzy(crit).test(string)\n  }\n}\n","'use strict'\nmodule.exports = new LiteralSearchStrategy()\n\nfunction LiteralSearchStrategy(){\n  function matchesString(string,crit){\n    return string.toLowerCase().indexOf(crit.toLowerCase()) >= 0\n  }\n\n  this.matches = function(string,crit){\n    if( typeof string !== 'string' ){\n      return false\n    }\n    string = string.trim()\n    return matchesString(string, crit)\n  }\n}\n","'use strict'\nmodule.exports = {\n  compile: compile,\n  setOptions: setOptions\n}\n\nvar options = {}\noptions.pattern = /\\{(.*?)\\}/g\noptions.template = ''\noptions.middleware = function(){}\n\nfunction setOptions(_options){\n  options.pattern = _options.pattern || options.pattern\n  options.template = _options.template || options.template\n  if( typeof _options.middleware === 'function' ){\n    options.middleware = _options.middleware\n  }\n}\n\nfunction compile(data){\n  return options.template.replace(options.pattern, function(match, prop) {\n    var value = options.middleware(prop, data[prop], options.template)\n    if( value !== undefined ){\n      return value\n    }\n    return data[prop] || match\n  })\n}\n",";(function(window, document, undefined){\n  'use strict'\n\n  var options = {\n    searchInput: null,\n    resultsContainer: null,\n    json: [],\n    searchResultTemplate: '<li><a href=\"{url}\" title=\"{desc}\">{title}</a></li>',\n    templateMiddleware: function(){},\n    noResultsText: 'No results found',\n    limit: 10,\n    fuzzy: false,\n    exclude: []\n  }\n\n  var requiredOptions = ['searchInput','resultsContainer','json']\n\n  var templater = require('./Templater')\n  var repository = require('./Repository')\n  var jsonLoader = require('./JSONLoader')\n  var optionsValidator = require('./OptionsValidator')({\n    required: requiredOptions\n  })\n  var utils = require('./utils')\n\n  /*\n    Public API\n  */\n  window.SimpleJekyllSearch = function SimpleJekyllSearch(_options){\n    var errors = optionsValidator.validate(_options)\n    if( errors.length > 0 ){\n      throwError('You must specify the following required options: ' + requiredOptions)\n    }\n\n    options = utils.merge(options, _options)\n\n    templater.setOptions({\n      template: options.searchResultTemplate,\n      middleware: options.templateMiddleware,\n    })\n\n    repository.setOptions({\n      fuzzy: options.fuzzy,\n      limit: options.limit,\n    })\n\n    if( utils.isJSON(options.json) ){\n      initWithJSON(options.json)\n    }else{\n      initWithURL(options.json)\n    }\n  }\n\n  // for backwards compatibility\n  window.SimpleJekyllSearch.init = window.SimpleJekyllSearch\n\n\n  function initWithJSON(json){\n    repository.put(json)\n    registerInput()\n  }\n\n  function initWithURL(url){\n    jsonLoader.load(url, function(err,json){\n      if( err ){\n        throwError('failed to get JSON (' + url + ')')\n      }\n      initWithJSON(json)\n    })\n  }\n\n  function emptyResultsContainer(){\n    options.resultsContainer.innerHTML = ''\n  }\n\n  function appendToResultsContainer(text){\n    options.resultsContainer.innerHTML += text\n  }\n\n  function registerInput(){\n    options.searchInput.addEventListener('keyup', function(e){\n      var key = e.which\n      var query = e.target.value\n      if( isWhitelistedKey(key) && isValidQuery(query) ) {\n        emptyResultsContainer();\n        render( repository.search(query) );\n      }\n    })\n  }\n\n  function render(results) {\n    if( results.length === 0 ){\n      return appendToResultsContainer(options.noResultsText)\n    }\n    for (var i = 0; i < results.length; i++) {\n      appendToResultsContainer( templater.compile(results[i]) )\n    }\n  }\n\n  function isValidQuery(query) {\n    return query && query.length > 0\n  }\n\n  function isWhitelistedKey(key) {\n    return [13,16,20,37,38,39,40,91].indexOf(key) === -1\n  }\n\n  function throwError(message){ throw new Error('SimpleJekyllSearch --- '+ message) }\n})(window, document);\n","'use strict'\nmodule.exports = {\n  merge: merge,\n  isJSON: isJSON,\n}\n\nfunction merge(defaultParams, mergeParams){\n  var mergedOptions = {}\n  for(var option in defaultParams){\n    mergedOptions[option] = defaultParams[option]\n    if( mergeParams[option] !== undefined ){\n      mergedOptions[option] = mergeParams[option]\n    }\n  }\n  return mergedOptions\n}\n\nfunction isJSON(json){\n  try{\n    if( json instanceof Object && JSON.parse(JSON.stringify(json)) ){\n      return true\n    }\n    return false\n  }catch(e){\n    return false\n  }\n}\n"]} diff --git a/example/js/jekyll-search.min.js b/example/js/jekyll-search.min.js index d055f30..c4d5ad8 100644 --- a/example/js/jekyll-search.min.js +++ b/example/js/jekyll-search.min.js @@ -1 +1 @@ -!function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a="function"==typeof require&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error("Cannot find module '"+o+"'")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}for(var i="function"==typeof require&&require,o=0;o=0}this.matches=function(string,crit){return"string"!=typeof string?!1:(string=string.trim(),matchesString(string,crit))}}module.exports=new LiteralSearchStrategy},{}],6:[function(require,module,exports){"use strict";function setOptions(_options){options.pattern=_options.pattern||options.pattern,options.template=_options.template||options.template,"function"==typeof _options.middleware&&(options.middleware=_options.middleware)}function compile(data){return options.template.replace(options.pattern,function(match,prop){var value=options.middleware(prop,data[prop],options.template);return void 0!==value?value:data[prop]||match})}module.exports={compile:compile,setOptions:setOptions};var options={};options.pattern=/\{(.*?)\}/g,options.template="",options.middleware=function(){}},{}],7:[function(require,module,exports){!function(window,document,undefined){"use strict";function initWithJSON(json){repository.put(json),registerInput()}function initWithURL(url){jsonLoader.load(url,function(err,json){err&&throwError("failed to get JSON ("+url+")"),initWithJSON(json)})}function emptyResultsContainer(){options.resultsContainer.innerHTML=""}function appendToResultsContainer(text){options.resultsContainer.innerHTML+=text}function registerInput(){options.searchInput.addEventListener("keyup",function(e){emptyResultsContainer(),e.target.value.length>0&&render(repository.search(e.target.value))})}function render(results){if(0===results.length)return appendToResultsContainer(options.noResultsText);for(var i=0;i{title}',templateMiddleware:function(){},noResultsText:"No results found",limit:10,fuzzy:!1,exclude:[]},requiredOptions=["searchInput","resultsContainer","json"],templater=require("./Templater"),repository=require("./Repository"),jsonLoader=require("./JSONLoader"),optionsValidator=require("./OptionsValidator")({required:requiredOptions}),utils=require("./utils");window.SimpleJekyllSearch=function(_options){var errors=optionsValidator.validate(_options);errors.length>0&&throwError("You must specify the following required options: "+requiredOptions),options=utils.merge(options,_options),templater.setOptions({template:options.searchResultTemplate,middleware:options.templateMiddleware}),repository.setOptions({fuzzy:options.fuzzy,limit:options.limit}),utils.isJSON(options.json)?initWithJSON(options.json):initWithURL(options.json)},window.SimpleJekyllSearch.init=window.SimpleJekyllSearch}(window,document)},{"./JSONLoader":1,"./OptionsValidator":2,"./Repository":3,"./Templater":6,"./utils":8}],8:[function(require,module,exports){"use strict";function merge(defaultParams,mergeParams){var mergedOptions={};for(var option in defaultParams)mergedOptions[option]=defaultParams[option],void 0!==mergeParams[option]&&(mergedOptions[option]=mergeParams[option]);return mergedOptions}function isJSON(json){try{return json instanceof Object&&JSON.parse(JSON.stringify(json))?!0:!1}catch(e){return!1}}module.exports={merge:merge,isJSON:isJSON}},{}]},{},[7]); \ No newline at end of file +!function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a="function"==typeof require&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error("Cannot find module '"+o+"'")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}for(var i="function"==typeof require&&require,o=0;o=0}this.matches=function(string,crit){return"string"!=typeof string?!1:(string=string.trim(),matchesString(string,crit))}}module.exports=new LiteralSearchStrategy},{}],6:[function(require,module,exports){"use strict";function setOptions(_options){options.pattern=_options.pattern||options.pattern,options.template=_options.template||options.template,"function"==typeof _options.middleware&&(options.middleware=_options.middleware)}function compile(data){return options.template.replace(options.pattern,function(match,prop){var value=options.middleware(prop,data[prop],options.template);return void 0!==value?value:data[prop]||match})}module.exports={compile:compile,setOptions:setOptions};var options={};options.pattern=/\{(.*?)\}/g,options.template="",options.middleware=function(){}},{}],7:[function(require,module,exports){!function(window,document,undefined){"use strict";function initWithJSON(json){repository.put(json),registerInput()}function initWithURL(url){jsonLoader.load(url,function(err,json){err&&throwError("failed to get JSON ("+url+")"),initWithJSON(json)})}function emptyResultsContainer(){options.resultsContainer.innerHTML=""}function appendToResultsContainer(text){options.resultsContainer.innerHTML+=text}function registerInput(){options.searchInput.addEventListener("keyup",function(e){var key=e.which,query=e.target.value;isWhitelistedKey(key)&&isValidQuery(query)&&(emptyResultsContainer(),render(repository.search(query)))})}function render(results){if(0===results.length)return appendToResultsContainer(options.noResultsText);for(var i=0;i0}function isWhitelistedKey(key){return-1===[13,16,20,37,38,39,40,91].indexOf(key)}function throwError(message){throw new Error("SimpleJekyllSearch --- "+message)}var options={searchInput:null,resultsContainer:null,json:[],searchResultTemplate:'
  • {title}
  • ',templateMiddleware:function(){},noResultsText:"No results found",limit:10,fuzzy:!1,exclude:[]},requiredOptions=["searchInput","resultsContainer","json"],templater=require("./Templater"),repository=require("./Repository"),jsonLoader=require("./JSONLoader"),optionsValidator=require("./OptionsValidator")({required:requiredOptions}),utils=require("./utils");window.SimpleJekyllSearch=function(_options){var errors=optionsValidator.validate(_options);errors.length>0&&throwError("You must specify the following required options: "+requiredOptions),options=utils.merge(options,_options),templater.setOptions({template:options.searchResultTemplate,middleware:options.templateMiddleware}),repository.setOptions({fuzzy:options.fuzzy,limit:options.limit}),utils.isJSON(options.json)?initWithJSON(options.json):initWithURL(options.json)},window.SimpleJekyllSearch.init=window.SimpleJekyllSearch}(window,document)},{"./JSONLoader":1,"./OptionsValidator":2,"./Repository":3,"./Templater":6,"./utils":8}],8:[function(require,module,exports){"use strict";function merge(defaultParams,mergeParams){var mergedOptions={};for(var option in defaultParams)mergedOptions[option]=defaultParams[option],void 0!==mergeParams[option]&&(mergedOptions[option]=mergeParams[option]);return mergedOptions}function isJSON(json){try{return json instanceof Object&&JSON.parse(JSON.stringify(json))?!0:!1}catch(e){return!1}}module.exports={merge:merge,isJSON:isJSON}},{}]},{},[7]); \ No newline at end of file diff --git a/package.json b/package.json index d45f267..007f76c 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "simple-jekyll-search", - "version": "1.0.9", + "version": "1.1.1", "description": "Simple Jekyll site search using javascript and json", "main": "dest/jekyll-search.js", "scripts": { diff --git a/src/index.js b/src/index.js index 8f9b941..e669489 100644 --- a/src/index.js +++ b/src/index.js @@ -79,22 +79,16 @@ function registerInput(){ options.searchInput.addEventListener('keyup', function(e){ - - // whitelist the following keycodes - var whitelist = [13,16,20,37,38,39,40,91]; - - // if the key pressed isn't whitelisted continue - if( whitelist.indexOf(e.which) === -1 ) { + var key = e.which + var query = e.target.value + if( isWhitelistedKey(key) && isValidQuery(query) ) { emptyResultsContainer(); - if( e.target.value.length > 0 ){ - render( repository.search(e.target.value) ); - } + render( repository.search(query) ); } - }) } - function render(results){ + function render(results) { if( results.length === 0 ){ return appendToResultsContainer(options.noResultsText) } @@ -103,5 +97,13 @@ } } + function isValidQuery(query) { + return query && query.length > 0 + } + + function isWhitelistedKey(key) { + return [13,16,20,37,38,39,40,91].indexOf(key) === -1 + } + function throwError(message){ throw new Error('SimpleJekyllSearch --- '+ message) } -})(window, document); \ No newline at end of file +})(window, document);