highlight-matches-utils
Version:
Utility functions to mark and/or highlight character matches in texts
78 lines (67 loc) • 1.98 kB
JavaScript
// @flow
var match = require('fuzzaldrin-plus').match
/*::
type SplitMatchesResult = Array<{|
isMatch: boolean,
str: string
|}>
*/
function splitMatches(
text /*: string */,
matches /*: number[] */
) /*: SplitMatchesResult */ {
var _matches = matches.slice(0)
var result = []
for (var i = 0; i < text.length; i += 1) {
var isMatch = i === _matches[0]
if (isMatch) {
_matches.shift()
}
var lastIndex = result.length - 1
if (lastIndex !== -1 && result[lastIndex].isMatch === isMatch) {
result[lastIndex].str += text[i]
} else {
result.push({ str: text[i], isMatch: isMatch })
}
}
return result
}
function highlightMatches /*:: <T> */(
text /*: string */,
matches /*: number[] */,
matchesWrapper /*: (s: string, index: number, array: SplitMatchesResult) => T */,
noMatchesWrapper /*: (s: string, index: number, array: SplitMatchesResult) => T */
) /*: Array<T> */ {
if (noMatchesWrapper == null) {
// $FlowFixMe
noMatchesWrapper = function(x) {
return x
}
}
if (matches.length === 0) {
// $FlowFixMe
return [noMatchesWrapper(text)]
}
var splitMatchesResult = splitMatches(text, matches)
var result = splitMatchesResult.map(function(r, i, a) {
return r.isMatch
? matchesWrapper(r.str, i, a)
: noMatchesWrapper(r.str, i, a)
})
return result
}
function highlightChars /*:: <T> */(
text /*: string */,
chars /*: string */,
matchesWrapper /*: (s: string, index: number, array: SplitMatchesResult) => T */,
noMatchesWrapper /*: (s: string, index: number, array: SplitMatchesResult) => T */
) /*: Array<T> */ {
var matches = match(text, chars)
var result = highlightMatches(text, matches, matchesWrapper, noMatchesWrapper)
return result
}
module.exports = {
splitMatches: splitMatches,
highlightMatches: highlightMatches,
highlightChars: highlightChars
}