Tutorial: How To Build A Spotlight Like Search And Navigation Experience
Search, navigate and run JS functions (think actions) with your searchbar
Table of contents
Your Searchbar is the most powerful widget on your site! Offer a SoTA search Ux similar to Spotlight, Chrome, Google, Amazon by integrating search, navigation and the ability to run custom JavaScript functions from right within your searchbar.
In this tutorial, I will show how to achieve using the SearchBox UI component of ReactiveSearch, an Apache 2.0 licensed search UI kit that works out-of-the-box with search engines such as Elasticsearch, OpenSearch, and MongoDB Atlas Search.
Beyond Search Box
Use cases for using the searchbox component in your application:
In-app navigation: Offer an in-app navigation experience through your searchbar allowing users to navigate through your site or dashboard without needing to jump between multiple menus, or relying on header/footer navigations.
Onboarding of new users: New users who are not familiar with the product or how to use it can just go to the search bar and search for the documentation of the product or search for help if needed. Through the in-app navigation, a user can search for frequently asked questions and navigate to the page effortlessly.
User productivity with actions: When actions like toggling between light mode or dark mode of your app can be done directly from the search bar, it can further enhance the search user experience.
Things You Need
Configuring Reactivesearch for use with React:
Reactivesearch is an Elasticsearch UI component library. It is an open-source library that offers rich and highly customizable UIs that can be used to index Elasticsearch hosted anywhere. We will be using the search box UI provided by Reactivesearch with React to build the spotlight-like search experience.
Postman for making API requests:
Postman is an application used for API testing. It is an HTTP client that tests HTTP requests, utilizing a graphical user interface, through which we obtain different types of responses that need to be subsequently validated. We will be leveraging on making API requests to an endpoint to store our required fields for our search box.
Implementing the SearchBox
Step 1: Setup
Create React App
So let's start by making sure we have node installed on our computer. You can use the command below to check if you have it installed. If you don't, you can install it by clicking here
node -v
// OR
node --version
After installing node on our computer, let's start creating our project. Go into your desired directory and install react. We will be using the popularly known create-react-app to initialize our project. Use the command below in your terminal to install react.
npx create-react-app spotlight-search
cd spotlight-search
npm start
Note: npx comes with npm 5.2+ and higher, see instructions for an older release here.
After creating your react app, it comes with a great folder structure like the one below.
spotlight-search
├── README.md
├── node_modules
├── package.json
├── .gitignore
├── package-lock.json
├── public
│ └── favicon.ico
│ └── index.html
│ └── manifest.json
└── src
└── App.css
└── App.js
└── App.test.js
└── index.css
└── index.js
└── logo.svg
└── registerServiceWorker.js
Install Reactive Search
Next, we will be installing the ReactiveSearch library using the command below on our terminal
npm install @appbaseio/reactivesearch
You can clone the app directly from github here
This is the current state of our app below
Step 2: Building Components
ReactiveBase Component We will use the ReactiveBase component to wrap our whole application in ./src/app.js. ReactiveBase is a container component provided to bind the backend (data source) and the UI components to enable them to interact effectively when there is a change in the data.
import { ReactiveBase } from '@appbaseio/reactivesearch';
function App() {
return (
<ReactiveBase
app='good-books-ds'
url='https://c3416aa3be0a:1dca7a12-f050-45a7-80e8-d9f296a06e4f@appbase-demo-ansible-abxiydt-arc.searchbase.io'
enableAppbase
appbaseConfig={{
recordAnalytics: true,
userId: 'jon',
}}
themePreset='light'
></ReactiveBase>
);
}
export default App
Note: The
url
passed in the<ReactiveBase />
is a sample credential and not going to work unless generated from the appbase website
From the props passed into the ReactiveBase the most important, we will be talking about is the url
. The url
is connecting our frontend with our backend to allow communication between the client-side and the server or data source. For this tutorial, we will be creating and hosting our data on the appbase app. Create an appbase account and follow the instructions to create a cluster. You can manage your data set graphically using dejavu by providing the url
and app name in the fields given in the app. For this app, you can view the data set by following this link
Refer to the github page to clone the current state of the app.
You can read more about the other props by visiting the appbase documentation on ReactiveBase
SearchBox Component
<SearchBox />
is a component used in building the interface for a search box. It offers a lightweight and performant Searchbox UI that can search on an index as well as offer search and navigation on curated suggestions.
<SearchBox
title='SearchBox'
defaultValue=''
dataField={['original_title', 'original_title.search']}
componentId='BookSensor'
highlight
URLParams
enablePopularSuggestions
popularSuggestionsConfig={{
size: 3,
minChars: 2,
index: 'good-books-ds',
}}
enableRecentSuggestions
recentSuggestionsConfig={{
size: 3,
index: 'good-books-ds',
minChars: 4,
}}
size={14}
enablePredictiveSuggestions
index='good-books-ds'
showClear
renderNoSuggestion='No suggestions found.'
innerClass={{
'suggestion-item': 'test-suggestion',
'active-suggestion-item': 'active-test-suggestion',
'section-header': 'section-header',
}}
/>
Below I will briefly talk about some of the props supplied to the search box and their functions.
dataField: The
dataField
takes an array of strings that can be used to specify which field in the database is being searched when the query is passed.enablePopularSuggestions and popularSuggestionsConfig: This is used to specify how popular searches are displayed on the suggestion section of the search bar. It also applies to the other suggestions like the recent and predictive suggestions
size: The
size
value is used to specify how many suggestions in total are being displayed. The default is 10 but can be changed based on what suits your needs.innerClass: The
innerClass
prop can be used to access each field in the search box which can help us in styling how we want our search UI looks.
You can read more about the SearchBox here.
The sandbox below is the state of our app after adding the search box
Step 3: Introduction to featuredSuggestions
The featuredSuggestions
prop is the newly added functionality for the <SearchBox />
component where it comes with a variety of use cases but most importantly improves the user experience of our application. Below I will show you how to add and use the prop in our book search website we have created so far. The highlight of what we will be doing is adding a function to toggle between the light and dark mode of our website, navigating to an external link on a new page, and adding a function to trigger an alert on the window of the browser.
Going back to our <SearchBox />
component, we are going to add an extra prop to enable the featuredSuggestions
and ingest our data from an external source.
enableFeaturedSuggestions={true}
featuredSuggestionsConfig={{
featuredSuggestionsGroupId: 'spotlight-search', // # mandatory
maxSuggestionsPerSection: 10,
sectionsOrder: ['repositories', 'docs', 'functions'],
}}
After adding this block of code, our search box will have a different but aesthetic look. The picture below illustrates how our search box is displayed when expanded.
Next, I will be explaining what the props of the featuredSuggestions
are and I will take you through how to add the new functionalities to your <SearchBox />
component.
enableFeaturedSuggestions: This prop tells the
<SearchBox />
to enable the use of thefeaturedSuggestions
feature for our search bar. It takes in a boolean value which that is either true or false. To use thefeaturedSuggestions
, we set the value to true as seen above.featuredSuggestionsConfig: This takes the configuration of how we want to structure our omnibar which gives us the spotlight experience which enhances our search bar and gives the user ease in navigating throughout our webpage and beyond. This prop takes a series of arguments wrapped into double curly braces. I will further break down those attributes in the next section.
Step 4: Styling our component
To give our application a unique look to make it stand out and also appeal to our users, I decided to create a custom styling that targets the tags and class names given in our component building stage.
The CSS styling below is linked to our app to give it a unique look as you will also see in a sandbox shortly.
body {
max-width: 1200px;
margin: 0 auto;
padding-top: 15px;
padding-inline:15px;
}
.test-suggestion img {
height: 25px;
max-height: 25px !important;
}
.active-test-suggestion img {
height: 25px;
max-height: 25px !important;
}
.section-header h2{
margin:0;
}
So here we are, we have built the interface of our application. Below is a sandbox showing the progress of what our application currently is.
Step 5: Details about building featuredSuggestions
To create the featuredSuggestions
section in our application, we need to perform a request to a backend API that is going to populate our featured suggestions section or the omnibar section. Below we have a cURL making a PUT request to a backend endpoint.
curl --location -g --request PUT 'http://{{user}}:{{password}}@{{host}}/_featured_suggestions_group/{{featured_suggestions_group_id}}' \
--header 'Content-Type: application/json' \
--data-raw '{
"sectionLabels": {
"repositories": {
"label": "<h2>Repositories</h2>"
},
"docs": {
"label": "<h2>Documentation</h2>"
},
"functions": {
"label": "<h2>Functions</h2>"
}
},
"suggestions": [
{
"label": "<h3>@appbaseio/reactivesearch</h3>",
"value": "reactivesearch repository",
"description": "Github repository for <mark>reactivesearch</mark>",
"action": "navigate",
"subAction": "{\"link\": \"https://github.com/appbaseio/reactivesearch\", \"target\": \"_blank\" }",
"sectionId": "repositories",
"iconURL": "https://camo.githubusercontent.com/95874e325a752e6ffb604991feb719c6e8bae6a552ed23d4a566bc4d518bc090/68747470733a2f2f692e696d6775722e636f6d2f696952397741732e706e67",
"order": 1
},
{
"label": "<h3>@appbaseio/reactivesearch-vue</h3>",
"value": "reactivesearch repository for vuejs",
"description": "Github repository for <mark>reactivesearch-vue</mark>",
"action": "navigate",
"subAction": "{\"link\": \"https://github.com/appbaseio/reactivesearch/tree/next/packages/vue\", \"target\": \"_blank\" }",
"sectionId": "repositories",
"iconURL": "https://camo.githubusercontent.com/95874e325a752e6ffb604991feb719c6e8bae6a552ed23d4a566bc4d518bc090/68747470733a2f2f692e696d6775722e636f6d2f696952397741732e706e67",
"order": 2
},
{
"label": "<h3>@appbaseio/react-searchbox</h3>",
"value": "appbase react-searchbox repository",
"description": "Github repository for <mark>react-searchbox</mark>",
"action": "navigate",
"subAction": "{\"link\": \"https://github.com/appbaseio/searchbox/tree/master/packages/react-searchbox\", \"target\": \"_blank\" }",
"sectionId": "repositories",
"iconURL": "https://camo.githubusercontent.com/95874e325a752e6ffb604991feb719c6e8bae6a552ed23d4a566bc4d518bc090/68747470733a2f2f692e696d6775722e636f6d2f696952397741732e706e67",
"order": 3
},
{
"label": "<h3>@appbaseio/vue-searchbox</h3>",
"value": "appbase vue-searchbox repository",
"description": "Github repository for <mark>vue-searchbox</mark>",
"action": "navigate",
"subAction": "{\"link\": \"https://github.com/appbaseio/searchbox/tree/master/packages/vue-searchbox\" , \"target\": \"_blank\"}",
"sectionId": "repositories",
"iconURL": "https://camo.githubusercontent.com/95874e325a752e6ffb604991feb719c6e8bae6a552ed23d4a566bc4d518bc090/68747470733a2f2f692e696d6775722e636f6d2f696952397741732e706e67",
"order": 4
},
{
"label": "<h3>@appbaseio/reactivesearch</h3>",
"value": "reactivesearch repository",
"description": "Documentation for <mark>reactivesearch</mark>",
"action": "navigate",
"subAction": "{\"link\": \"https://docs.appbase.io/docs/reactivesearch/v3/overview/quickstart\", \"target\": \"_blank\" }",
"sectionId": "docs",
"iconURL": "https://camo.githubusercontent.com/95874e325a752e6ffb604991feb719c6e8bae6a552ed23d4a566bc4d518bc090/68747470733a2f2f692e696d6775722e636f6d2f696952397741732e706e67",
"order": 1
},
{
"label": "<h3>@appbaseio/reactivesearch-vue</h3>",
"value": "documentation for vuejs reactivesearch",
"description": "Documentation for <mark>reactivesearch-vue</mark>",
"action": "navigate",
"subAction": "{\"link\": \"https://docs.appbase.io/docs/reactivesearch/vue/overview/QuickStart\", \"target\": \"_blank\" }",
"sectionId": "docs",
"iconURL": "https://camo.githubusercontent.com/95874e325a752e6ffb604991feb719c6e8bae6a552ed23d4a566bc4d518bc090/68747470733a2f2f692e696d6775722e636f6d2f696952397741732e706e67",
"order": 2
},
{
"label": "<h3>@appbaseio/react-searchbox</h3>",
"value": "appbase react-searchbox docs",
"description": "Documentation for <mark>react-searchbox</mark>",
"action": "navigate",
"subAction": "{\"link\": \"https://docs.appbase.io/docs/reactivesearch/react-searchbox/quickstart\", \"target\": \"_blank\" }",
"sectionId": "docs",
"iconURL": "https://camo.githubusercontent.com/95874e325a752e6ffb604991feb719c6e8bae6a552ed23d4a566bc4d518bc090/68747470733a2f2f692e696d6775722e636f6d2f696952397741732e706e67",
"order": 3
},
{
"label": "<h3>@appbaseio/vue-searchbox</h3>",
"value": "appbase vue-searchbox docs",
"description": "Documentation for <mark>vue-searchbox</mark>",
"action": "navigate",
"subAction": "{\"link\": \"https://docs.appbase.io/docs/reactivesearch/vue-searchbox/quickstart\" , \"target\": \"_blank\"}",
"sectionId": "docs",
"iconURL": "https://camo.githubusercontent.com/95874e325a752e6ffb604991feb719c6e8bae6a552ed23d4a566bc4d518bc090/68747470733a2f2f692e696d6775722e636f6d2f696952397741732e706e67",
"order": 4
},
{
"label": "<h3>Alert</h3>",
"value": "alert",
"description": "Invokes an alert message",
"action": "function",
"subAction": "function() { alert('Hello World! with theme');}",
"sectionId": "functions",
"iconURL": "https://www.svgrepo.com/show/281847/bell.svg",
"order": 1
},
{
"label": "<h3>Light</h3>",
"value": "light mode",
"description": "Change theme to light mode",
"action": "function",
"subAction": "function(){ window.history.pushState({ theme: 'white'}, '', '/light');window.location.reload();}",
"sectionId": "functions",
"iconURL": "https://www.svgrepo.com/show/20546/sun.svg",
"order": 2
},
{
"label": "<h3>Dark</h3>",
"value": "dark mode",
"description": "Change theme to dark mode",
"action": "function",
"subAction": " function(){ window.history.pushState({ theme: 'dark'}, '', '/dark');window.location.reload();}",
"sectionId": "functions",
"iconURL": "https://www.svgrepo.com/show/8265/moon.svg",
"order": 3
},
{
"label": "<h3>Search on Google</h3>",
"value": "google search",
"description": "Search keyword on google",
"action": "function",
"subAction": "function() {window.open('https://www.google.com');}",
"sectionId": "functions",
"iconURL": "https://www.svgrepo.com/show/122724/google.svg",
"order": 4
}
]
}'
The endpoint
The endpoint is used to create or update the list of featured suggestions. In our case, our endpoint will look like the example below:
'https://c3416aa3be0a:1dca7a12-f050-45a7-80e8-d9f296a06e4f@appbase-demo-ansible-abxiydt-arc.searchbase.io/_featured_suggestions_group/spotlight_search'
// Compare the above and below
'http://{{user}}:{{password}}@{{host}}/_featured_suggestions_group/{{featured_suggestions_group_id}}'
// the {{user}}:{{password}}@{{host}} is simply our url from the ReactiveBase container component above
The endpoint takes in a featured_suggestions_group_id
that can be anything based on your preference. Using this endpoint perform, a PUT request on postman or any other API testing platform of your choice.
You will pass a body which mainly consists of two main parts sectionLabels and
suggestions.
The appbase.io team is working on providing a GUI for ingesting the featured suggestions which will make it possible to add featured suggestions with a point-and-click control plane.
The body
The body of our request is very vital in this article. Therefore, I will take my time to explain everything using simple terms to make it easy to create one. The body from the above request is written in JSON format, it is passed to the request endpoint body. The body is categorized into two main parts:
- sectionLabels: The section label is used to tell the user what a particular section in the search bar is doing. It helps the user understand what action is performed when he interacts with a component. Below are the section labels from our cURL:
"sectionLabels": {
"repositories": {
"label": "<h2>Repositories</h2>"
},
"docs": {
"label": "<h2>Documentation</h2>"
},
"functions": {
"label": "<h2>Functions</h2>"
}
},
We have three main sections from the above repositories
, docs
, and functions
. The label which is enclosed within the curly braces is what the user sees when using the UI.
- suggestions: The suggestions are an array that contains all your suggestions and the necessary values needed to display the props, It has different values like the
label
,value
,action
, and so on.
Below we have a sample from our suggestions array above:
{
"label": "<h3>@appbaseio/vue-searchbox</h3>",
"value": "appbase vue-searchbox repository",
"description": "Github repository for <mark>vue-searchbox</mark>",
"action": "navigate",
"subAction": "{\"link\": \"https://github.com/appbaseio/searchbox/tree/master/packages/vue-searchbox\" , \"target\": \"_blank\"}",
"sectionId": "repositories",
"iconURL": "https://camo.githubusercontent.com/95874e325a752e6ffb604991feb719c6e8bae6a552ed23d4a566bc4d518bc090/68747470733a2f2f692e696d6775722e636f6d2f696952397741732e706e67",
"order": 4
},
Below we have the definition of the necessary values in a suggestion.
- label: The
label
is used to display the name of the suggestion, it accepts HTML and a string depending on how you want the element to be displayed. - value: The
value
element is used to give the suggestion a unique value. - description: The
description
value is used to display a mini description under a suggestion for more clarification for the user. action: The
action
is a very essential part of the suggestion. It takes two valuesnavigate
andfunction
.When
navigate
is passed as the action, it identifies that the suggestion is a link element that can be used to navigate through the website or even access an external link. Thefunction
is passed into the action when you have an intention to perform or trigger a function on the website. This function can range from displaying an alert to changing certain flows of the website like toggling between light and dark mode.- subAction: This depends on the action provided. If the provided action is
navigate
then we have to pass a link to the subAction like the below illustration
"subAction": "{\"link\": \"https://docs.appbase.io/docs/reactivesearch/vue-searchbox/quickstart\" , \"target\": \"_blank\"}"
it takes a link
pointing to the location of the suggestion and a target
value to either open the link on a new page when _blank
is passed or open it on the same page when self
is passed as the target
.
The second action is function
, when this is passed the subAction
now takes in a function to perform an action on the website. The below illustration shows an example of a function to display an alert
"subAction": "function(currentSuggestion, inputValue, customEvents) { alert('Hello World! with theme');}"
currentSuggestion
is the currently selected suggestion item. It's an object having information about the suggestion.inputValue
is the current input value of the SearchBox.customEvents
is a key-value pair object that is injected via theReactiveBase
component as depicted below.
<Reactivebase
appbaseConfig={{
customEvents: {
platform: "ios",
device: "iphoneX"
}
}}
>
</Reactivebase>
- sectionId: This takes the value where we want our suggestion to appear. For the
sectionId
, you pass any value that is present in thesectionLabels
. The suggestion is going to appear among the values in a particular section.
- iconURL or icon: Next, we want to have an icon that will be shown beside our suggestion. We can achieve this in two ways. we can use the
iconURL
for passing an external link to the icon we want to display or use theicon
to pass an HTML tag for our icon. The illustration below shows an example of the two use cases"iconURL": "https://www.svgrepo.com/show/281847/bell.svg", // OR "icon": "<img src="<img src=\"\"/>" alt='icon description' height='10' width='10' /> "
- order: This simply takes the order in terms of relevance we want to display each suggestion in a section.
You can get the full body of the request here
Step 6: Connecting Everything
Finally, to put everything together, We'll go to the postman and make our request. Make the PUT request to your endpoint based on your credentials gotten from appbase app.
We are going to get a response returning the featured_id
we supplied. We will then use that id in our featuredSuggestion
prop.
featuredSuggestionsConfig={{
featuredSuggestionsGroupId: 'spotlight', // id returned after the request.
maxSuggestionsPerSection: 10,
sectionsOrder: ['repositories', 'docs', 'functions'],
}}
Our final app will look like the below sandbox now.
You can also access the code on github here
Live Demo and Code
Finally, We have reached the end of our book search app which comes with a spotlight search experience that can give an incredible user search experience. You can access the live site here and the full code is on github which can be accessed here.
Summary
Here is a summary of all the steps we covered in this tutorial to build a Spotlight like search experience with the SearchBox UI component:
If you enjoyed this tutorial, you should 🌟 the ReactiveSearch UI project over here.