Skip to content
This repository has been archived by the owner on Jun 11, 2021. It is now read-only.

Commit

Permalink
feat(docsearch): add hitComponent and transformItems options
Browse files Browse the repository at this point in the history
  • Loading branch information
francoischalifour committed Apr 9, 2020
1 parent 7137e86 commit daaafe5
Show file tree
Hide file tree
Showing 4 changed files with 61 additions and 22 deletions.
37 changes: 18 additions & 19 deletions packages/docsearch-react/src/DocSearch.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -14,20 +14,23 @@ import {
} from './types';
import { createSearchClient, groupBy, noop } from './utils';
import { createStoredSearches } from './stored-searches';
import { Hit } from './Hit';
import { SearchBox } from './SearchBox';
import { ScreenState } from './ScreenState';
import { Footer } from './Footer';

interface DocSearchProps
extends Omit<
PublicAutocompleteOptions<InternalDocSearchHit>,
'onStateChange' | 'getSources'
> {
extends Pick<PublicAutocompleteOptions<InternalDocSearchHit>, 'navigator'> {
appId?: string;
apiKey: string;
indexName: string;
searchParameters: any;
onClose(): void;
searchParameters?: any;
onClose?(): void;
transformItems?(items: DocSearchHit[]): DocSearchHit[];
hitComponent?(props: {
hit: DocSearchHit;
children: React.ReactNode;
}): JSX.Element;
}

export function DocSearch({
Expand All @@ -36,7 +39,9 @@ export function DocSearch({
indexName,
searchParameters,
onClose = noop,
...autocompleteProps
transformItems = x => x,
hitComponent = Hit,
navigator,
}: DocSearchProps) {
const [state, setState] = React.useState<
AutocompleteState<InternalDocSearchHit>
Expand Down Expand Up @@ -107,7 +112,7 @@ export function DocSearch({
initialState: {
query: initialQuery,
},
...autocompleteProps,
navigator,
onStateChange({ state }) {
setState(state as any);
},
Expand Down Expand Up @@ -191,17 +196,7 @@ export function DocSearch({
throw error;
})
.then((hits: DocSearchHit[]) => {
const formattedHits = hits.map(hit => {
const url = new URL(hit.url);
return {
...hit,
url: hit.url
// @TODO: temporary convenience for development.
.replace(url.origin, '')
.replace('#__docusaurus', ''),
};
});
const sources = groupBy(formattedHits, hit => hit.hierarchy.lvl0);
const sources = groupBy(hits, hit => hit.hierarchy.lvl0);

// We store the `lvl0`s to display them as search suggestions
// in the “no results“ screen.
Expand All @@ -224,6 +219,7 @@ export function DocSearch({
return Object.values(
groupBy(items, item => item.hierarchy.lvl1)
)
.map(transformItems)
.map(hits =>
hits.map(item => {
return {
Expand Down Expand Up @@ -256,6 +252,8 @@ export function DocSearch({
favoriteSearches,
saveRecentSearch,
initialQuery,
navigator,
transformItems,
]
);

Expand Down Expand Up @@ -338,6 +336,7 @@ export function DocSearch({
<ScreenState
{...autocomplete}
state={state}
hitComponent={hitComponent}
recentSearches={recentSearches}
favoriteSearches={favoriteSearches}
onItemClick={item => {
Expand Down
12 changes: 12 additions & 0 deletions packages/docsearch-react/src/Hit.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
import React from 'react';

import { DocSearchHit } from './types';

interface HitProps {
hit: DocSearchHit;
children: React.ReactNode;
}

export function Hit({ hit, children }: HitProps) {
return <a href={hit.url}>{children}</a>;
}
16 changes: 13 additions & 3 deletions packages/docsearch-react/src/Results.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,11 @@ import {
AutocompleteState,
} from '@francoischalifour/autocomplete-core';

import { StoredDocSearchHit, InternalDocSearchHit } from './types';
import {
DocSearchHit,
InternalDocSearchHit,
StoredDocSearchHit,
} from './types';
import { Snippet } from './Snippet';

interface ResultsProps<TItem>
Expand All @@ -19,6 +23,10 @@ interface ResultsProps<TItem>
renderIcon(props: { item: TItem; index: number }): React.ReactNode;
renderAction(props: { item: TItem }): React.ReactNode;
onItemClick(item: TItem): void;
hitComponent(props: {
hit: DocSearchHit;
children: React.ReactNode;
}): JSX.Element;
}

export function Results<TItem extends StoredDocSearchHit>(
Expand All @@ -28,6 +36,8 @@ export function Results<TItem extends StoredDocSearchHit>(
return null;
}

const Hit = props.hitComponent;

return (
<section className="DocSearch-Hits">
<div className="DocSearch-Hit-source">{props.title}</div>
Expand All @@ -52,7 +62,7 @@ export function Results<TItem extends StoredDocSearchHit>(
},
})}
>
<a href={item.url}>
<Hit hit={item}>
<div className="DocSearch-Hit-Container">
{props.renderIcon({ item, index })}

Expand Down Expand Up @@ -110,7 +120,7 @@ export function Results<TItem extends StoredDocSearchHit>(

{props.renderAction({ item })}
</div>
</a>
</Hit>
</li>
);
})}
Expand Down
18 changes: 18 additions & 0 deletions packages/website/src/theme/SearchBar/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import React, { useState, useEffect, useCallback } from 'react';
import { createPortal } from 'react-dom';
import useDocusaurusContext from '@docusaurus/useDocusaurusContext';
import { useHistory } from '@docusaurus/router';
import Link from '@docusaurus/Link';
import { SearchButton } from 'docsearch-react';

let DocSearch = null;
Expand Down Expand Up @@ -97,11 +98,28 @@ function SearchBar() {
history.push(suggestionUrl);
},
}}
transformItems={items => {
return items.map(item => {
const url = new URL(item.url);

return {
...item,
url: item.url
.replace(url.origin, '')
.replace('#__docusaurus', ''),
};
});
}}
hitComponent={Hit}
/>,
document.body
)}
</>
);
}

function Hit({ hit, children }) {
return <Link to={hit.url}>{children}</Link>;
}

export default SearchBar;

0 comments on commit daaafe5

Please sign in to comment.