mirror of
https://github.com/remoteintech/remote-jobs
synced 2024-12-31 22:58:46 +00:00
dac8b04fc8
* Prevent duplicate company names
* Fix output indentation
* Search full profile content using lunr.js
* Remove extra stop words
This wasn't really working correctly - the stop word 'work' would leave
instances of 'working' and 'works' in the index for example.
* Change company name description from "Name" to "Company name"
* Pre-process query:
- Search for terms in AND mode, per
https://lunrjs.com/guides/searching.html#term-presence
- Discard non-alphanumeric characters from the search
- Better handling of contractions and searching for stop words
* Display search query and results in the console
* Add special search token: _incomplete
* Add a link to search for incomplete profiles
* Revert "Add a link to search for incomplete profiles"
This reverts commit f6384c90cb
.
* Add link to search documentation
* Improve search explanation appearance when it spans multiple lines
* Fix searching for contractions
Previously, searching for e.g. "don't" wasn't working correctly. After
trimming the contraction, "do" is a stop word, so it should be ignored.
* Improve "empty search" message
* Prefer matches other than "company name" in search excerpts
* Move inline scripts before external scripts
This probably doesn't matter right now due to the way the scripts are
currently structured, but it might matter one day and it's more logical
this way.
* Fix search engine index progress
* Improve script indentation
* I got 99 problems and they're all bots
* Update script exit code
When a Node.js error occurs the exit code is probably going to be 1, so
we should use a different code.
* Fix the tests
* Update documentation
This was wrong (out of date), but the correct version is obvious from
reading the code.
* Make download progress work in both Chrome and Firefox
See https://stackoverflow.com/a/32799706
222 lines
6.4 KiB
JavaScript
222 lines
6.4 KiB
JavaScript
function setupSearch() {
|
|
var table = document.querySelector( 'table#companies-table' );
|
|
|
|
var searchInput = document.createElement( 'input' );
|
|
searchInput.type = 'text';
|
|
searchInput.placeholder = 'Search';
|
|
searchInput.id = 'search-input';
|
|
|
|
var searchStatus = document.createElement( 'span' );
|
|
searchStatus.id = 'search-status';
|
|
|
|
var companiesHeading = document.querySelector( 'h2#companies' );
|
|
companiesHeading.appendChild( searchInput );
|
|
companiesHeading.appendChild( searchStatus );
|
|
|
|
var searchExplanation = document.createElement( 'p' );
|
|
searchExplanation.id = 'search-explanation';
|
|
searchExplanation.innerHTML = (
|
|
'Use the text box above to search all of our company data. '
|
|
+ ' <a href="https://blog.remoteintech.company/search-help/">More info</a>'
|
|
);
|
|
table.parentNode.insertBefore( searchExplanation, table );
|
|
|
|
var searchLoading = false;
|
|
var searchData = null;
|
|
var searchIndex = null;
|
|
var updateTimeout = null;
|
|
|
|
function updateSearch() {
|
|
if ( ! searchData || searchLoading ) {
|
|
return;
|
|
}
|
|
|
|
var searchValue = searchInput.value
|
|
.replace( /[^a-z0-9_']+/gi, ' ' )
|
|
.trim()
|
|
.split( ' ' )
|
|
.map( function( term ) {
|
|
term = term
|
|
.replace( /('m|'ve|n't|'d|'ll|'ve|'s|'re)$/, '' )
|
|
.replace( /'/g, '' );
|
|
if ( ! lunr.stopWordFilter( term.toLowerCase() ) ) {
|
|
return null;
|
|
} else if ( term ) {
|
|
return '+' + term;
|
|
} else {
|
|
return term;
|
|
}
|
|
} )
|
|
.filter( Boolean )
|
|
.join( ' ' );
|
|
var allMatch = ! searchValue;
|
|
var searchResults = searchValue ? searchIndex.search( searchValue ) : [];
|
|
var searchDisplayValue = (
|
|
searchValue === '+_incomplete'
|
|
? 'Incomplete profile'
|
|
: searchInput.value.trim()
|
|
);
|
|
if ( allMatch ) {
|
|
searchStatus.innerHTML = (
|
|
'Empty search; showing all '
|
|
+ searchData.textData.length
|
|
+ ' companies'
|
|
);
|
|
} else if ( searchResults.length === 1 ) {
|
|
searchStatus.innerText = searchDisplayValue + ': 1 result';
|
|
} else {
|
|
searchStatus.innerText = (
|
|
searchDisplayValue + ': '
|
|
+ searchResults.length + ' results'
|
|
);
|
|
}
|
|
var searchMatches = {};
|
|
searchResults.forEach( function( r ) {
|
|
searchMatches[ +r.ref ] = r;
|
|
} );
|
|
if ( window.console && console.log ) {
|
|
console.log( 'search', { value: searchValue, results: searchResults } );
|
|
}
|
|
searchData.textData.forEach( function( company, index ) {
|
|
var match = searchMatches[ index ];
|
|
var row = document.getElementById( 'company-row-' + index );
|
|
var rowMatch = row.nextElementSibling;
|
|
if ( rowMatch && rowMatch.classList.contains( 'company-match' ) ) {
|
|
rowMatch.parentNode.removeChild( rowMatch );
|
|
}
|
|
row.style.display = ( match || allMatch ? '' : 'none' );
|
|
row.classList.remove( 'has-match' );
|
|
if ( match ) {
|
|
row.classList.add( 'has-match' );
|
|
var metadata = match.matchData.metadata;
|
|
var contextWords = ( window.innerWidth <= 600 ? 4 : 6 );
|
|
var k1, k2, pos;
|
|
loop1: for ( k1 in metadata ) {
|
|
for ( k2 in metadata[ k1 ] ) {
|
|
pos = metadata[ k1 ][ k2 ].position[ 0 ];
|
|
if ( k2 !== 'nameText' ) {
|
|
// Accept company name for matches, but prefer
|
|
// other fields if there are any
|
|
break loop1;
|
|
}
|
|
}
|
|
}
|
|
rowMatch = document.createElement( 'tr' );
|
|
rowMatch.setAttribute( 'class', 'company-match' );
|
|
var rowMatchCell = document.createElement( 'td' );
|
|
rowMatchCell.setAttribute( 'colspan', 3 );
|
|
var spanBefore = document.createElement( 'span' );
|
|
var spanMatch = document.createElement( 'strong' );
|
|
var spanAfter = document.createElement( 'span' );
|
|
var text = company[ k2 ];
|
|
var words = [];
|
|
var currentWord = '';
|
|
var i, inWord, c;
|
|
for ( i = pos[ 0 ] - 1; i >= 0; i-- ) {
|
|
c = text.substring( i, i + 1 );
|
|
inWord = /\S/.test( c );
|
|
if ( inWord ) {
|
|
currentWord = c + currentWord;
|
|
}
|
|
if ( ( ! inWord || i === 0 ) && currentWord ) {
|
|
words.unshift( currentWord );
|
|
currentWord = '';
|
|
if ( words.length === contextWords + 1 ) {
|
|
words[ 0 ] = '\u2026';
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
spanBefore.innerText = (
|
|
( window.innerWidth > 600 ? searchData.headings[ k2 ] + ': ' : '' )
|
|
+ words.join( ' ' )
|
|
+ ' '
|
|
).replace( /\(_incomplete\)/, '(Incomplete)' );
|
|
spanMatch.innerText = text
|
|
.substring( pos[ 0 ], pos[ 0 ] + pos[ 1 ] )
|
|
.replace( /\(_incomplete\)/, '(Incomplete)' );
|
|
words = [];
|
|
currentWord = '';
|
|
for ( i = pos[ 0 ] + pos[ 1 ] + 1; i < text.length; i++ ) {
|
|
c = text.substring( i, i + 1 );
|
|
inWord = /\S/.test( c );
|
|
if ( inWord ) {
|
|
currentWord += c;
|
|
}
|
|
if ( ( ! inWord || i === text.length - 1 ) && currentWord ) {
|
|
words.push( currentWord );
|
|
currentWord = '';
|
|
if ( words.length === contextWords + 1 ) {
|
|
words[ contextWords ] = '\u2026';
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
spanAfter.innerText = (
|
|
' ' + words.join( ' ' )
|
|
).replace( /\(_incomplete\)/, '(Incomplete)' );
|
|
rowMatchCell.appendChild( spanBefore );
|
|
rowMatchCell.appendChild( spanMatch );
|
|
rowMatchCell.appendChild( spanAfter );
|
|
rowMatch.appendChild( rowMatchCell );
|
|
row.parentNode.insertBefore( rowMatch, row.nextSibling );
|
|
}
|
|
} );
|
|
}
|
|
|
|
searchInput.addEventListener( 'focus', function() {
|
|
if ( searchData || searchLoading ) {
|
|
return;
|
|
}
|
|
|
|
searchLoading = true;
|
|
var searchLoadingText = 'Loading search data...';
|
|
|
|
searchStatus.innerHTML = searchLoadingText;
|
|
|
|
var xhr = new XMLHttpRequest();
|
|
xhr.open( 'GET', searchIndexFilename );
|
|
|
|
xhr.onprogress = function( e ) {
|
|
var percentLoaded;
|
|
if ( e.lengthComputable ) {
|
|
percentLoaded = Math.round( 100 * e.loaded / e.total );
|
|
} else {
|
|
percentLoaded = Math.min(
|
|
100,
|
|
Math.round( 100 * e.loaded / searchIndexSize )
|
|
);
|
|
}
|
|
searchStatus.innerHTML = searchLoadingText + ' ' + percentLoaded + '%';
|
|
};
|
|
|
|
xhr.onload = function() {
|
|
searchLoading = false;
|
|
if ( xhr.status === 200 ) {
|
|
searchData = JSON.parse( xhr.response );
|
|
searchIndex = lunr.Index.load( searchData.index );
|
|
updateSearch();
|
|
} else {
|
|
searchStatus.innerHTML = 'Error!';
|
|
}
|
|
};
|
|
|
|
xhr.send();
|
|
} );
|
|
|
|
searchInput.addEventListener( 'keyup', function() {
|
|
if ( updateTimeout ) {
|
|
clearTimeout( updateTimeout );
|
|
}
|
|
updateTimeout = setTimeout( updateSearch, 450 );
|
|
} );
|
|
|
|
document.body.setAttribute(
|
|
'class',
|
|
document.body.getAttribute( 'class' ) + ' search-enabled'
|
|
);
|
|
}
|
|
|
|
document.addEventListener( 'DOMContentLoaded', function( event ) {
|
|
setupSearch();
|
|
} );
|