import Fuse = require('fuse.js'); import preact = require("preact"); const { h, Component } = preact; type DocItem = { display_html: string name: string module: string link: string } function loadJSON(path: string, success: (json: DocItem[]) => void, error: (xhr: XMLHttpRequest) => void) { const xhr = new XMLHttpRequest(); xhr.onreadystatechange = () => { if (xhr.readyState === XMLHttpRequest.DONE) { if (xhr.status === 200) { if (success) { try { success(JSON.parse(xhr.responseText)); } catch (exc) { error(xhr); } } } else { if (error) { error(xhr); } } } }; xhr.open("GET", path, true); xhr.send(); } // -------------------------------------------------------------------------- // class PageMenuButton extends Component<any, any> { render(props: { title: string, onClick: () => void }) { function onClick(e: Event) { e.preventDefault(); props.onClick(); } return <li><a href="#" onClick={onClick}>{props.title}</a></li>; } } function addSearchPageMenuButton(action: () => void) { const pageMenu = document.querySelector('#page-menu') as HTMLUListElement; const dummy = document.createElement('li'); pageMenu.insertBefore(dummy, pageMenu.firstChild); preact.render(<PageMenuButton onClick={action} title="Quick Jump" />, pageMenu, dummy); } // -------------------------------------------------------------------------- // function take<T>(n: number, arr: T[]) { if (arr.length <= n) { return arr; } return arr.slice(0, n); } type FuseResult<T> = { score: number item: T } type ResultsInModule = { module: string, totalScore: number, items: FuseResult<DocItem>[] } type QuickJumpProps = { baseUrl: string showHideTrigger: (action: () => void) => void } type QuickJumpState = { searchString: string isVisible: boolean expanded: { [moduleName: string]: true } activeLinkIndex: number moduleResults: ResultsInModule[] failedLoading?: boolean fuse: Fuse } class QuickJump extends Component<QuickJumpProps, QuickJumpState> { private linkIndex: number = 0; private focusPlease: boolean = false; private navigatedByKeyboard: boolean = false; private activeLink: undefined | HTMLAnchorElement; private activeLinkAction: undefined | (() => void); private input: undefined | HTMLInputElement; private searchResults: undefined | Element; componentWillMount() { this.setState({ searchString: '', isVisible: false, expanded: {}, activeLinkIndex: -1, moduleResults: [] }); loadJSON(this.props.baseUrl + "/doc-index.json", (data) => { this.setState({ fuse: new Fuse(data, { threshold: 0.25, caseSensitive: true, includeScore: true, tokenize: true, keys: [ { name: "name", weight: 0.7 }, { name: "module", weight: 0.3 } ] }), moduleResults: [] }); }, (err) => { if (console) { console.error("could not load 'doc-index.json' for searching", err); } this.setState({ failedLoading: true }); }); document.addEventListener('mousedown', this.hide.bind(this)); document.addEventListener('keydown', (e) => { if (this.state.isVisible) { if (e.key === 'Escape') { this.hide(); } else if (e.key === 'ArrowUp' || (e.key === 'k' && e.ctrlKey)) { e.preventDefault(); this.navigateLinks(-1); } else if (e.key === 'ArrowDown' || (e.key === 'j' && e.ctrlKey)) { e.preventDefault(); this.navigateLinks(+1); } else if (e.key === 'Enter' && this.state.activeLinkIndex >= 0) { this.followActiveLink(); } } if (e.key === 's' && (e.target as HTMLElement).tagName.toLowerCase() !== 'input') { e.preventDefault(); this.show(); } }) } hide() { this.setState({ isVisible: false }); } show() { if (!this.state.isVisible) { this.focusPlease = true; this.setState({ isVisible: true, activeLinkIndex: -1 }); } } toggleVisibility() { if (this.state.isVisible) { this.hide(); } else { this.show(); } } navigateLinks(change: number) { const newActiveLinkIndex = Math.max(-1, Math.min(this.linkIndex-1, this.state.activeLinkIndex + change)); this.navigatedByKeyboard = true; this.setState({ activeLinkIndex: newActiveLinkIndex }); } followActiveLink() { if (!this.activeLinkAction) { return; } this.activeLinkAction(); } updateResults() { const searchString = (this.input && this.input.value) || ''; const results: FuseResult<DocItem>[] = this.state.fuse.search(searchString); const resultsByModule: { [name: string]: FuseResult<DocItem>[] } = {}; results.forEach((result) => { const moduleName = result.item.module; const resultsInModule = resultsByModule[moduleName] || (resultsByModule[moduleName] = []); resultsInModule.push(result); }); const moduleResults: ResultsInModule[] = []; for (const moduleName in resultsByModule) { const items = resultsByModule[moduleName]; let sumOfInverseScores = 0; items.forEach((item) => { sumOfInverseScores += 1/item.score; }); moduleResults.push({ module: moduleName, totalScore: 1/sumOfInverseScores, items: items }); } moduleResults.sort((a, b) => a.totalScore - b.totalScore); this.setState({ searchString: searchString, isVisible: true, moduleResults: moduleResults }); } componentDidUpdate() { if (this.searchResults && this.activeLink && this.navigatedByKeyboard) { const rect = this.activeLink.getClientRects()[0]; const searchResultsTop = this.searchResults.getClientRects()[0].top; if (rect.bottom > window.innerHeight) { this.searchResults.scrollTop += rect.bottom - window.innerHeight + 80; } else if (rect.top < searchResultsTop) { this.searchResults.scrollTop -= searchResultsTop - rect.top + 80; } } if (this.focusPlease && this.input) { this.input.focus(); } this.navigatedByKeyboard = false; this.focusPlease = false; } componentDidMount() { this.props.showHideTrigger(this.toggleVisibility.bind(this)); } render(props: any, state: QuickJumpState) { if (state.failedLoading) { const usingFileProtocol = window.location.protocol == 'file:'; return <div id="search" class={state.isVisible ? '' : 'hidden'}> <div id="search-results"> <p class="error">Failed to load file 'doc-index.json' containing definitions in this package.</p> {usingFileProtocol ? <p class="error"> To use quick jump, load this page with HTTP (from a local static file web server) instead of using the <code>file://</code> protocol. (For security reasons, it is not possible to fetch auxiliary files using JS in a HTML page opened with <code>file://</code>.) </p> : [] } </div> </div>; } this.linkIndex = 0; const stopPropagation = (e: Event) => { e.stopPropagation(); }; const onMouseOver = (e: MouseEvent) => { let target: null | Element = e.target as Element; while (target && typeof target.getAttribute === 'function') { const linkIndexString = target.getAttribute('data-link-index'); if (typeof linkIndexString == 'string') { const linkIndex = parseInt(linkIndexString, 10); this.setState({ activeLinkIndex: linkIndex }); break; } target = target.parentNode as null | Element; } }; const items = take(10, state.moduleResults).map((r) => this.renderResultsInModule(r)); return <div id="search" class={state.isVisible ? '' : 'hidden'}> <div id="search-form" onMouseDown={stopPropagation}> <input placeholder="Search in package by name" ref={(input) => { this.input = input as undefined | HTMLInputElement; }} onFocus={this.show.bind(this)} onClick={this.show.bind(this)} onInput={this.updateResults.bind(this)} /> </div> <div id="search-results" ref={(el) => { this.searchResults = el; }} onMouseDown={stopPropagation} onMouseOver={onMouseOver}> {state.searchString === '' ? [<IntroMsg />, <KeyboardShortcuts />] : items.length == 0 ? <NoResultsMsg searchString={state.searchString} /> : <ul>{items}</ul>} </div> </div>; } renderResultsInModule(resultsInModule: ResultsInModule): JSX.Element { const items = resultsInModule.items; const moduleName = resultsInModule.module; const showAll = this.state.expanded[moduleName] || items.length <= 10; const visibleItems = showAll ? items : take(8, items); const expand = () => { const newExpanded = Object.assign({}, this.state.expanded); newExpanded[moduleName] = true; this.setState({ expanded: newExpanded }); }; const renderItem = (item: DocItem) => { return <li class="search-result"> {this.navigationLink(this.props.baseUrl + "/" + item.link, {}, <DocHtml html={item.display_html} /> )} </li>; }; return <li class="search-module"> <h4>{moduleName}</h4> <ul> {visibleItems.map((item) => renderItem(item.item))} {showAll ? [] : <li class="more-results"> {this.actionLink(expand, {}, "show " + (items.length - visibleItems.length) + " more results from this module")} </li>} </ul> </li>; } navigationLink(href: string, attrs: JSX.HTMLAttributes&JSX.SVGAttributes&{[propName: string]: any}, ...children: (JSX.Element|JSX.Element[]|string)[]) { const fullAttrs = Object.assign({ href: href, onClick: this.hide.bind(this) }, attrs); const action = () => { window.location.href = href; this.hide(); }; return this.menuLink(fullAttrs, action, ...children); } actionLink(callback: () => void, attrs: JSX.HTMLAttributes&JSX.SVGAttributes&{[propName: string]: any}, ...children: (JSX.Element|JSX.Element[]|string)[]) { const onClick = (e: Event) => { e.preventDefault(); callback(); }; const fullAttrs = Object.assign({ href: '#', onClick: onClick }, attrs); return this.menuLink(fullAttrs, callback, ...children); } menuLink(attrs: JSX.HTMLAttributes&JSX.SVGAttributes&{[propName: string]: any}, action: () => void, ...children: (JSX.Element|JSX.Element[]|string)[]) { const linkIndex = this.linkIndex; if (linkIndex === this.state.activeLinkIndex) { attrs['class'] = (attrs['class'] ? attrs['class'] + ' ' : '') + 'active-link'; attrs.ref = (link?: Element) => { if (link) this.activeLink = link as HTMLAnchorElement; }; this.activeLinkAction = action; } const newAttrs = Object.assign({ 'data-link-index': linkIndex }, attrs); this.linkIndex += 1; return h('a', newAttrs, ...children); } } class DocHtml extends Component<{ html: string }, {}> { shouldComponentUpdate(newProps: { html: string }) { return this.props.html !== newProps.html; } render(props: { html: string }) { return <div dangerouslySetInnerHTML={{__html: props.html}} />; } }; function KeyboardShortcuts() { return <table class="keyboard-shortcuts"> <tr> <th>Key</th> <th>Shortcut</th> </tr> <tr> <td><span class="key">s</span></td> <td>Open this search box</td> </tr> <tr> <td><span class="key">esc</span></td> <td>Close this search box</td> </tr> <tr> <td> <span class="key">↓</span>, <span class="key">ctrl</span> + <span class="key">j</span> </td> <td>Move down in search results</td> </tr> <tr> <td> <span class="key">↑</span>, <span class="key">ctrl</span> + <span class="key">k</span> </td> <td>Move up in search results</td> </tr> <tr> <td><span class="key">↵</span></td> <td>Go to active search result</td> </tr> </table>; } function IntroMsg() { return <p>You can find any exported type, constructor, class, function or pattern defined in this package by (approximate) name.</p>; } function NoResultsMsg(props: { searchString: string }) { const messages = [ <p> Your search for '{props.searchString}' produced the following list of results: <code>[]</code>. </p>, <p> <code>Nothing</code> matches your query for '{props.searchString}'. </p>, <p> <code> Left "no matches for '{props.searchString}'" :: Either String (NonEmpty SearchResult) </code> </p> ]; return messages[(props.searchString || 'a').charCodeAt(0) % messages.length]; } export function init(docBaseUrl?: string, showHide?: (action: () => void) => void) { preact.render( <QuickJump baseUrl={docBaseUrl || "."} showHideTrigger={showHide || addSearchPageMenuButton} />, document.body ); } // export to global object (window as any).quickNav = { init: init };