|  |  | @ -1,115 +1,152 @@ | 
			
		
	
		
		
			
				
					
					|  |  |  | // adapted from https://github.com/Glench/fuzzyset.js/blob/master/lib/fuzzyset.js
 |  |  |  | // adapted from https://github.com/Glench/fuzzyset.js/blob/master/lib/fuzzyset.js
 | 
			
		
	
		
		
			
				
					
					|  |  |  | // BSD Licensed
 |  |  |  | // BSD Licensed
 | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | export default function FuzzySet( |  |  |  | const GRAM_SIZE_LOWER = 2; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 	arr, |  |  |  | const GRAM_SIZE_UPPER = 3; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 	useLevenshtein, |  |  |  | 
 | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 	gramSizeLower, |  |  |  | // return an edit distance from 0 to 1
 | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 	gramSizeUpper |  |  |  | function _distance(str1: string, str2: string) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | ) { |  |  |  | 	if (str1 === null && str2 === null) | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 	// default options
 |  |  |  | 		throw 'Trying to compare two null values'; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 	arr = arr || []; |  |  |  | 	if (str1 === null || str2 === null) return 0; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 	this.gramSizeLower = gramSizeLower || 2; |  |  |  | 	str1 = String(str1); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 	this.gramSizeUpper = gramSizeUpper || 3; |  |  |  | 	str2 = String(str2); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 	this.useLevenshtein = typeof useLevenshtein !== 'boolean' |  |  |  | 
 | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 		? true |  |  |  | 	const distance = levenshtein(str1, str2); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 		: useLevenshtein; |  |  |  | 	if (str1.length > str2.length) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 		return 1 - distance / str1.length; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 	// define all the object functions and attributes
 |  |  |  | 	} else { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 	this.exactSet = {}; |  |  |  | 		return 1 - distance / str2.length; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 	this.matchDict = {}; |  |  |  | 	} | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 	this.items = {}; |  |  |  | } | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 	// helper functions
 |  |  |  | // helper functions
 | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 	function levenshtein(str1, str2) { |  |  |  | function levenshtein(str1: string, str2: string) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 		const current = []; |  |  |  | 	const current: number[] = []; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 		let prev; |  |  |  | 	let prev; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 		let value; |  |  |  | 	let value; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 		for (let i = 0; i <= str2.length; i++) { |  |  |  | 	for (let i = 0; i <= str2.length; i++) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 			for (let j = 0; j <= str1.length; j++) { |  |  |  | 		for (let j = 0; j <= str1.length; j++) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 				if (i && j) { |  |  |  | 			if (i && j) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 					if (str1.charAt(j - 1) === str2.charAt(i - 1)) { |  |  |  | 				if (str1.charAt(j - 1) === str2.charAt(i - 1)) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 						value = prev; |  |  |  | 					value = prev; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 					} else { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 						value = Math.min(current[j], current[j - 1], prev) + 1; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 					} |  |  |  |  | 
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  | 				} else { |  |  |  | 				} else { | 
			
		
	
		
		
			
				
					
					|  |  |  | 					value = i + j; |  |  |  | 					value = Math.min(current[j], current[j - 1], prev) + 1; | 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  | 				} |  |  |  | 				} | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 			} else { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 				prev = current[j]; |  |  |  | 				value = i + j; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 				current[j] = value; |  |  |  |  | 
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  | 			} |  |  |  | 			} | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 			prev = current[j]; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 			current[j] = value; | 
			
		
	
		
		
			
				
					
					|  |  |  | 		} |  |  |  | 		} | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 	} | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 	return current.pop(); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | const _nonWordRe = /[^\w, ]+/; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | 		return current.pop(); |  |  |  | function _iterateGrams(value: string, gramSize: number) { | 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 	gramSize = gramSize || 2; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 	const simplified = '-' + value.toLowerCase().replace(_nonWordRe, '') + '-'; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 	const lenDiff = gramSize - simplified.length; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 	const results = []; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 	if (lenDiff > 0) { | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 		for (let i = 0; i < lenDiff; ++i) { | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 			value += '-'; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 		} | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 	} | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 	for (let i = 0; i < simplified.length - gramSize + 1; ++i) { | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 		results.push(simplified.slice(i, i + gramSize)); | 
			
		
	
		
		
			
				
					
					|  |  |  | 	} |  |  |  | 	} | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 	return results; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | function _gramCounter(value: string, gramSize: number) { | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 	// return an object where key=gram, value=number of occurrences
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 	gramSize = gramSize || 2; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 	const result = {}; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 	const grams = _iterateGrams(value, gramSize); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 	let i = 0; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | 	// return an edit distance from 0 to 1
 |  |  |  | 	for (i; i < grams.length; ++i) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 	function _distance(str1, str2) { |  |  |  | 		if (grams[i] in result) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 		if (str1 === null && str2 === null) |  |  |  | 			result[grams[i]] += 1; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 			throw 'Trying to compare two null values'; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		if (str1 === null || str2 === null) return 0; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		str1 = String(str1); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		str2 = String(str2); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		const distance = levenshtein(str1, str2); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		if (str1.length > str2.length) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			return 1 - distance / str1.length; |  |  |  |  | 
			
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  | 		} else { |  |  |  | 		} else { | 
			
		
	
		
		
			
				
					
					|  |  |  | 			return 1 - distance / str2.length; |  |  |  | 			result[grams[i]] = 1; | 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  | 		} |  |  |  | 		} | 
			
		
	
		
		
			
				
					
					|  |  |  | 	} |  |  |  | 	} | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 	return result; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | 	const _nonWordRe = /[^\w, ]+/; |  |  |  | function sortDescending(a, b) { | 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 	return b[0] - a[0]; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | 	function _iterateGrams(value, gramSize) { |  |  |  | export default class FuzzySet { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 		gramSize = gramSize || 2; |  |  |  | 	exactSet: object; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 		const simplified = '-' + value.toLowerCase().replace(_nonWordRe, '') + '-'; |  |  |  | 	matchDict: object; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 		const lenDiff = gramSize - simplified.length; |  |  |  | 	items: object; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 		const results = []; |  |  |  |  | 
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | 		if (lenDiff > 0) { |  |  |  | 	constructor(arr: string[]) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 			for (let i = 0; i < lenDiff; ++i) { |  |  |  | 		// define all the object functions and attributes
 | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 				value += '-'; |  |  |  | 		this.exactSet = {}; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 			} |  |  |  | 		this.matchDict = {}; | 
			
				
				
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 		this.items = {}; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 		// initialization
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 		for (let i = GRAM_SIZE_LOWER; i < GRAM_SIZE_UPPER + 1; ++i) { | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 			this.items[i] = []; | 
			
		
	
		
		
			
				
					
					|  |  |  | 		} |  |  |  | 		} | 
			
		
	
		
		
			
				
					
					|  |  |  | 		for (let i = 0; i < simplified.length - gramSize + 1; ++i) { |  |  |  | 		// add all the items to the set
 | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 			results.push(simplified.slice(i, i + gramSize)); |  |  |  | 		for (let i = 0; i < arr.length; ++i) { | 
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 			this.add(arr[i]); | 
			
		
	
		
		
			
				
					
					|  |  |  | 		} |  |  |  | 		} | 
			
		
	
		
		
			
				
					
					|  |  |  | 		return results; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	} |  |  |  | 	} | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | 	function _gramCounter(value, gramSize) { |  |  |  | 	add(value: string) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 		// return an object where key=gram, value=number of occurrences
 |  |  |  | 		const normalizedValue = value.toLowerCase(); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 		gramSize = gramSize || 2; |  |  |  | 		if (normalizedValue in this.exactSet) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 		const result = {}; |  |  |  | 			return false; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 		const grams = _iterateGrams(value, gramSize); |  |  |  | 		} | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 		let i = 0; |  |  |  |  | 
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | 		for (i; i < grams.length; ++i) { |  |  |  | 		let i = GRAM_SIZE_LOWER; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 			if (grams[i] in result) { |  |  |  | 		for (i; i < GRAM_SIZE_UPPER + 1; ++i) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 				result[grams[i]] += 1; |  |  |  | 			this._add(value, i); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 			} else { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 				result[grams[i]] = 1; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			} |  |  |  |  | 
			
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  | 		} |  |  |  | 		} | 
			
		
	
		
		
			
				
					
					|  |  |  | 		return result; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	} |  |  |  | 	} | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | 	// the main functions
 |  |  |  | 	_add(value: string, gramSize: number) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 	this.get = function(value, defaultValue) { |  |  |  | 		const normalizedValue = value.toLowerCase(); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 		// check for value in set, returning defaultValue or null if none found
 |  |  |  | 		const items = this.items[gramSize] || []; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 		const result = this._get(value); |  |  |  | 		const index = items.length; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 		if (!result && typeof defaultValue !== 'undefined') { |  |  |  | 
 | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 			return defaultValue; |  |  |  | 		items.push(0); | 
			
				
				
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 		const gramCounts = _gramCounter(normalizedValue, gramSize); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 		let sumOfSquareGramCounts = 0; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 		let gram; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 		let gramCount; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 		for (gram in gramCounts) { | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 			gramCount = gramCounts[gram]; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 			sumOfSquareGramCounts += Math.pow(gramCount, 2); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 			if (gram in this.matchDict) { | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 				this.matchDict[gram].push([index, gramCount]); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 			} else { | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 				this.matchDict[gram] = [[index, gramCount]]; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 			} | 
			
		
	
		
		
			
				
					
					|  |  |  | 		} |  |  |  | 		} | 
			
		
	
		
		
			
				
					
					|  |  |  | 		return result; |  |  |  | 		const vectorNormal = Math.sqrt(sumOfSquareGramCounts); | 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 		items[index] = [vectorNormal, normalizedValue]; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 		this.items[gramSize] = items; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 		this.exactSet[normalizedValue] = value; | 
			
		
	
		
		
			
				
					
					|  |  |  | 	}; |  |  |  | 	}; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | 	this._get = function(value) { |  |  |  | 	get(value: string) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 		const normalizedValue = this._normalizeStr(value); |  |  |  | 		const normalizedValue = value.toLowerCase(); | 
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  | 		const result = this.exactSet[normalizedValue]; |  |  |  | 		const result = this.exactSet[normalizedValue]; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | 		if (result) { |  |  |  | 		if (result) { | 
			
		
	
	
		
		
			
				
					|  |  | @ -119,8 +156,8 @@ export default function FuzzySet( | 
			
		
	
		
		
			
				
					
					|  |  |  | 		let results = []; |  |  |  | 		let results = []; | 
			
		
	
		
		
			
				
					
					|  |  |  | 		// start with high gram size and if there are no results, go to lower gram sizes
 |  |  |  | 		// start with high gram size and if there are no results, go to lower gram sizes
 | 
			
		
	
		
		
			
				
					
					|  |  |  | 		for ( |  |  |  | 		for ( | 
			
		
	
		
		
			
				
					
					|  |  |  | 			let gramSize = this.gramSizeUpper; |  |  |  | 			let gramSize = GRAM_SIZE_UPPER; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 			gramSize >= this.gramSizeLower; |  |  |  | 			gramSize >= GRAM_SIZE_LOWER; | 
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  | 			--gramSize |  |  |  | 			--gramSize | 
			
		
	
		
		
			
				
					
					|  |  |  | 		) { |  |  |  | 		) { | 
			
		
	
		
		
			
				
					
					|  |  |  | 			results = this.__get(value, gramSize); |  |  |  | 			results = this.__get(value, gramSize); | 
			
		
	
	
		
		
			
				
					|  |  | @ -129,10 +166,10 @@ export default function FuzzySet( | 
			
		
	
		
		
			
				
					
					|  |  |  | 			} |  |  |  | 			} | 
			
		
	
		
		
			
				
					
					|  |  |  | 		} |  |  |  | 		} | 
			
		
	
		
		
			
				
					
					|  |  |  | 		return null; |  |  |  | 		return null; | 
			
		
	
		
		
			
				
					
					|  |  |  | 	}; |  |  |  | 	} | 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | 	this.__get = function(value, gramSize) { |  |  |  | 	__get(value: string, gramSize: number) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 		const normalizedValue = this._normalizeStr(value); |  |  |  | 		const normalizedValue = value.toLowerCase(); | 
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  | 		const matches = {}; |  |  |  | 		const matches = {}; | 
			
		
	
		
		
			
				
					
					|  |  |  | 		const gramCounts = _gramCounter(normalizedValue, gramSize); |  |  |  | 		const gramCounts = _gramCounter(normalizedValue, gramSize); | 
			
		
	
		
		
			
				
					
					|  |  |  | 		const items = this.items[gramSize]; |  |  |  | 		const items = this.items[gramSize]; | 
			
		
	
	
		
		
			
				
					|  |  | @ -159,17 +196,6 @@ export default function FuzzySet( | 
			
		
	
		
		
			
				
					
					|  |  |  | 			} |  |  |  | 			} | 
			
		
	
		
		
			
				
					
					|  |  |  | 		} |  |  |  | 		} | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | 		function isEmptyObject(obj) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			for (const prop in obj) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 				if (obj.hasOwnProperty(prop)) return false; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			} |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			return true; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		} |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		if (isEmptyObject(matches)) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			return null; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		} |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		const vectorNormal = Math.sqrt(sumOfSquareGramCounts); |  |  |  | 		const vectorNormal = Math.sqrt(sumOfSquareGramCounts); | 
			
		
	
		
		
			
				
					
					|  |  |  | 		let results = []; |  |  |  | 		let results = []; | 
			
		
	
		
		
			
				
					
					|  |  |  | 		let matchScore; |  |  |  | 		let matchScore; | 
			
		
	
	
		
		
			
				
					|  |  | @ -182,127 +208,28 @@ export default function FuzzySet( | 
			
		
	
		
		
			
				
					
					|  |  |  | 				items[matchIndex][1], |  |  |  | 				items[matchIndex][1], | 
			
		
	
		
		
			
				
					
					|  |  |  | 			]); |  |  |  | 			]); | 
			
		
	
		
		
			
				
					
					|  |  |  | 		} |  |  |  | 		} | 
			
		
	
		
		
			
				
					
					|  |  |  | 		function sortDescending(a, b) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			if (a[0] < b[0]) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 				return 1; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			} else if (a[0] > b[0]) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 				return -1; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			} else { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 				return 0; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			} |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		} |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | 		results.sort(sortDescending); |  |  |  | 		results.sort(sortDescending); | 
			
		
	
		
		
			
				
					
					|  |  |  | 		if (this.useLevenshtein) { |  |  |  | 
 | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 			const newResults = []; |  |  |  | 		let newResults = []; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 			const endIndex = Math.min(50, results.length); |  |  |  | 		const endIndex = Math.min(50, results.length); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 			// truncate somewhat arbitrarily to 50
 |  |  |  | 		// truncate somewhat arbitrarily to 50
 | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 			for (let i = 0; i < endIndex; ++i) { |  |  |  | 		for (let i = 0; i < endIndex; ++i) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 				newResults.push([ |  |  |  | 			newResults.push([ | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 					_distance(results[i][1], normalizedValue), |  |  |  | 				_distance(results[i][1], normalizedValue), | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 					results[i][1], |  |  |  | 				results[i][1], | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 				]); |  |  |  | 			]); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 			} |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			results = newResults; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			results.sort(sortDescending); |  |  |  |  | 
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  | 		} |  |  |  | 		} | 
			
		
	
		
		
			
				
					
					|  |  |  | 		const newResults = []; |  |  |  | 		results = newResults; | 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 		results.sort(sortDescending); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 		newResults = []; | 
			
		
	
		
		
			
				
					
					|  |  |  | 		for (let i = 0; i < results.length; ++i) { |  |  |  | 		for (let i = 0; i < results.length; ++i) { | 
			
		
	
		
		
			
				
					
					|  |  |  | 			if (results[i][0] == results[0][0]) { |  |  |  | 			if (results[i][0] == results[0][0]) { | 
			
		
	
		
		
			
				
					
					|  |  |  | 				newResults.push([results[i][0], this.exactSet[results[i][1]]]); |  |  |  | 				newResults.push([results[i][0], this.exactSet[results[i][1]]]); | 
			
		
	
		
		
			
				
					
					|  |  |  | 			} |  |  |  | 			} | 
			
		
	
		
		
			
				
					
					|  |  |  | 		} |  |  |  | 		} | 
			
		
	
		
		
			
				
					
					|  |  |  | 		return newResults; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	}; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	this.add = function(value) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		const normalizedValue = this._normalizeStr(value); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		if (normalizedValue in this.exactSet) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			return false; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		} |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		let i = this.gramSizeLower; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		for (i; i < this.gramSizeUpper + 1; ++i) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			this._add(value, i); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		} |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	}; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	this._add = function(value, gramSize) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		const normalizedValue = this._normalizeStr(value); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		const items = this.items[gramSize] || []; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		const index = items.length; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		items.push(0); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		const gramCounts = _gramCounter(normalizedValue, gramSize); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		let sumOfSquareGramCounts = 0; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		let gram; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		let gramCount; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		for (gram in gramCounts) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			gramCount = gramCounts[gram]; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			sumOfSquareGramCounts += Math.pow(gramCount, 2); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			if (gram in this.matchDict) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 				this.matchDict[gram].push([index, gramCount]); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			} else { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 				this.matchDict[gram] = [[index, gramCount]]; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			} |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		} |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		const vectorNormal = Math.sqrt(sumOfSquareGramCounts); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		items[index] = [vectorNormal, normalizedValue]; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		this.items[gramSize] = items; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		this.exactSet[normalizedValue] = value; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	}; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	this._normalizeStr = function(str) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		if (Object.prototype.toString.call(str) !== '[object String]') |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			throw 'Must use a string as argument to FuzzySet functions'; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		return str.toLowerCase(); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	}; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | 	// return length of items in set
 |  |  |  | 		return newResults; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 	this.length = function() { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		let count = 0; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		let prop; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		for (prop in this.exactSet) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			if (this.exactSet.hasOwnProperty(prop)) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 				count += 1; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			} |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		} |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		return count; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	}; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	// return is set is empty
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	this.isEmpty = function() { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		for (const prop in this.exactSet) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			if (this.exactSet.hasOwnProperty(prop)) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 				return false; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			} |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		} |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		return true; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	}; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	// return list of values loaded into set
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	this.values = function() { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		const values = []; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		for (const prop in this.exactSet) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			if (this.exactSet.hasOwnProperty(prop)) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 				values.push(this.exactSet[prop]); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 			} |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		} |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		return values; |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  | 	}; |  |  |  | 	}; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	// initialization
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	let i = this.gramSizeLower; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	for (i; i < this.gramSizeUpper + 1; ++i) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		this.items[i] = []; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	} |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	// add all the items to the set
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	for (i = 0; i < arr.length; ++i) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 		this.add(arr[i]); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	} |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 	return this; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } |