Structured results in Angular InstantSearch
Angular InstantSearch isn’t compatible with Angular’s Ivy view engine. We’re investigating how best to support this. For more information and to vote for Algolia’s support of Angular 16 and beyond, see the GitHub issue Algolia Support for Angular InstantSearch
On this page
Why using structured results?
One of the most interesting recent additions to search result pages are what’s commonly called “structured results.”
Different companies use different names:
- Google calls them “Knowledge Graph Cards”,
- Bing calls them “Visually Rich Snippets”,
-
DuckDuckGo, Yandex, and Baidu all have their own flavors as well.
Traditional search, which consists of displaying long lists of search results once users press “Enter” on their keyboard, usually falls short of piquing their curiosity and triggering engagement. This is when new ways of searching emerge.
Instead, a successful search should attempt to communicate clearly and cleanly with a user on multiple levels when applicable, where different types of results help guide users to the information they want. Sometimes, it’s desirable to showcase a particular piece of relevant content against the rest of a user’s standard query search results. This is why many believe in the value of leveraging “structured results”.
How about your search?
Uses for structured results
You’ll want to use structured results whenever there is a singular piece of information you want to stand out from the rest for a given query.
Here are a few examples:
- Movie titles on an entertainment website
- User details and avatar on a customer relationship management (CRM) system
- Sponsored brands on an ecommerce search (see example and code on GitHub).
You can see a common thread among these examples - a need to search two different types of structured data - and Algolia happens to handle these cases out of the box.
An example
First, take a look at the ecommerce example mentioned in the preceding section. You can view the code and follow along on GitHub.
Imagine a scenario where certain brands within your ecommerce site have requested that their brand will be displayed if it happens to match a user’s query. In your search, if you determine that someone is searching for a particular brand name, you can embed a section displaying the brand and its name within the main search interface.
How will the data look?
One index will be the main index to host traditional results - in your case, tech products - regardless of a match on the brand. Another index will be used to store your brands as records like so:
1
2
3
4
5
{
"name": "Bosch",
"url": "https://logo.clearbit.com/bosch.com",
"objectID": "3414165441"
}
Tweaking the relevance
Now that you added a new index for structured results to use in conjunction with your main index, outline what you’d like to emphasize in your structured results:
- Only display a single result that matches a user’s search
- Continue to handle typos, but be more stringent
- Ensure that you have all words matching in order (so “The Bad News” and “The Bad News Bears” would match but “Bears Bad News” would not)
Next, determine the corresponding settings that would be configured in Algolia.
In this case, you only need to care about results with a minimal number of typos, especially if you only render one result:
The corresponding code for this dashboard configuration would be:
1
2
3
4
5
index.setSettings({
typoTolerance: 'min'
}).then(() => {
// done
});
Next, decrease your typo tolerance by increasing the minimum characters from 4 to 5 for a single typo:
And the corresponding code for this dashboard configuration would be:
1
2
3
index.setSettings({ minWordSizefor1Typo: 5 }).then(() => {
// done
});
These configuration changes make the typo tolerance more stringent. The next section addresses the last point: ensuring that all words match in order.
Displaying the data
Federated search makes it easy to perform a single query against several datasets. You can use it with your structured brand result:
1
2
3
4
5
6
7
8
<ais-instant-search index-name="mainIndex" :search-client="searchClient">
<ais-index index-name="structuredIndex">
<ais-configure :hits-per-page.camel="1" :get-ranking-info.camel="true">
</ais-configure>
<!-- we'll add the structured results widget here -->
</ais-index>
<!-- other widgets -->
</ais-instant-search>
In this snippet, you’re taking advantage of the ais-index
widget to nest multiple indices that share the same UI state (or “search state”). When a user searches in the app, it searches in both the main and the structured indices.
You can use the ais-configure
widget to:
- Stay on the first page (
0
sincepage
is zero-based) because the main index can be paginated, and the structured index would then “inherit” from its pagination state. - Set
hitsPerPage
to1
because you only show the first result. - Set
getRankingInfo
totrue
so that Algolia returns metadata about the hit’s ranking.
When rendering results, take the first result with the ais-hits
connector and do a conditional check:
1
2
3
4
5
6
7
8
<ais-instant-search index-name="mainIndex" :search-client="searchClient">
<ais-index index-name="structuredIndex">
<ais-configure :hits-per-page.camel="1" :get-ranking-info.camel="true">
</ais-configure>
<my-structured-results></my-structured-results>
</ais-index>
<!-- other widgets -->
</ais-instant-search>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
import { Component, Inject, forwardRef, Optional } from '@angular/core';
import {
TypedBaseWidget,
NgAisInstantSearch,
NgAisIndex,
} from 'angular-instantsearch';
import connectHits, {
HitsConnectorParams,
HitsWidgetDescription,
HitsRenderState,
} from 'instantsearch.js/es/connectors/hits/connectHits';
@Component({
selector: 'my-structured-results',
template: `
<div *ngIf="result">
<img [src]="result.url" [alt]="result.name" />
<h3>{{ result.name }}</h3>
</div>
`,
})
export class MyStructuredResults extends TypedBaseWidget<
HitsWidgetDescription,
HitsConnectorParams
> {
public state: HitsRenderState = {
hits: [],
results: undefined,
bindEvent: undefined,
sendEvent: undefined,
};
constructor(
@Inject(forwardRef(() => NgAisIndex))
@Optional()
public parentIndex: NgAisIndex,
@Inject(forwardRef(() => NgAisInstantSearch))
public instantSearchInstance: NgAisInstantSearch
) {
super('MyStructuredResults');
}
public ngOnInit() {
this.createWidget(connectHits);
super.ngOnInit();
}
public updateState = (state: HitsRenderState, isFirstRendering: boolean) => {
if (isFirstRendering) return;
this.state = state;
};
get result() {
return this.state.hits[0];
}
}
By comparing the number of exact word matches to 1 + the total distance between all the words in the match, you can make sure that all matched words are sorted. In other terms, when all matching words against a user’s query are sequential and adjacent, the proximity (or distance between search terms) is always equal to exact word matches - 1
There’s no requirement as to when to display your content, this part is up to you.
Another constraint you may impose on the rendering of the structured result could be to check that the count of exact word matches is equal to the number of words in the value of the attribute being queried. For example, you may only want to show a brand when all words of that brand have been typed. Check the getRankingInfo
documentation to learn about other useful metrics you can access in the results.
Another example, a different use case
Now imagine you’re iterating on an entertainment website that displays structured results on movies: if you can determine that someone is searching for a specific movie, you will embed a section displaying the movie details within the main search interface. That is, you’d like to communicate to users that there might be a salient piece of content worth showcasing.
How will the data look?
Consider how you should structure your data. First, you need a main index, as with the ecommerce example. This is where people will always see traditional results even if there’s no structured data match.
Think about a major search engine when you search a new movie: you might get structured results with the cast, showtimes, and video clips at the same time. In Algolia, that would require three extra indices—one for each result type.
This example is a bit simpler, as it only has a single data type to show structured results: movies. Yet, you can imagine a circumstance where you would match on actors, directors, etc.
Here’s how a sample movie record could look like:
1
2
3
4
5
6
7
8
9
10
{
"title": "The Shawshank Redemption",
"year": 1994,
"image": "https://image.tmdb.org/t/p/w154/9O7gLzmreU0nGkIB6K3BsJbzvNv.jpg",
"color": "#8C634B",
"score": 9.97764206054169,
"rating": 5,
"actors": ["Tim Robbins", "Morgan Freeman", "Bob Gunton"],
"objectID": "439817390"
}
Records with this structure would be stored in another index. Keep in mind though that if your new index doesn’t contain any new data, it could be a replica of your primary index. (for example, searchable attributes or custom ranking order).
You can display the data and render the result exactly the same way as in the previous ecommerce example.
Conclusion
The movie and ecommerce examples you went over are just two scenarios among many that may call for structured data, all following a similar pattern of giving users the ability to search through multiple indices and showcase results accordingly.
Search paradigms continue to evolve, and the rise of “structured results” is part of an ongoing itch to enrich a user’s process of discovery while browsing content. Algolia enables users the flexibility to roll out their own flavor of “structured results” by making simple the ability to search against multiple types of content and customizing how the search is performed.