Skip to content

Commit e820cc3

Browse files
committed
fix(searchBox): allow searchBox to reuse an <input>
This no longer uses React for the `input` display (it still uses it for the optionnal `PoweredBy` widget). All HTML attributes already set on the `input` are kept, but sensible defaults (like `autocorrect`, etc) are added. It listens to the `keyup` event only to start the search. There is no `render` method because it does not need to be updated on results, only triggering it.
1 parent 53605d9 commit e820cc3

File tree

5 files changed

+48
-125
lines changed

5 files changed

+48
-125
lines changed

README.md

+3-6
Original file line numberDiff line numberDiff line change
@@ -199,12 +199,7 @@ npm run test:watch # developer mode, test only
199199
* Instantiate a searchbox
200200
* @param {String|DOMElement} options.container CSS Selector or DOMElement to insert the widget
201201
* @param {String} [options.placeholder='Search here'] Input's placeholder
202-
* @param {Object} [options.cssClasses] CSS classes to add to the wrapping elements: root, input
203-
* @param {String|String[]} [options.cssClasses.root=null]
204-
* @param {String|String[]} [options.cssClasses.input=null]
205-
* @param {Object} [options.templates] Templates to use for the widget
206-
* @param {String|Function} [options.templates.header=''] Header template
207-
* @param {String|Function} [options.templates.footer=''] Footer template
202+
* @param {Object} [options.cssClass] CSS classes to add to the input
208203
* @param {boolean} [poweredBy=false] Show a powered by Algolia link below the input
209204
* @return {Object}
210205
*/
@@ -213,6 +208,8 @@ npm run test:watch # developer mode, test only
213208
#### Usage
214209

215210
```html
211+
<input id="search-box" />
212+
Or
216213
<div id="search-box"></div>
217214
```
218215

components/SearchBox.js

-68
This file was deleted.

example/app.js

-3
Original file line numberDiff line numberDiff line change
@@ -17,9 +17,6 @@ search.addWidget(
1717
instantsearch.widgets.searchBox({
1818
container: '#search-box',
1919
placeholder: 'Search for products',
20-
cssClasses: {
21-
input: 'form-control'
22-
},
2320
poweredBy: true
2421
})
2522
);

example/index.html

+3-1
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,9 @@ <h1>Instant search demo <small>using instantsearch.js</small></h1>
2020
<div class="row">
2121
<div class="col-md-3">
2222
<div class="panel">
23-
<div class="panel-body" id="search-box"></div>
23+
<div class="panel-body">
24+
<input id="search-box" class="form-control" />
25+
</div>
2426
</div>
2527

2628
<div class="panel panel-default" id="categories"></div>

widgets/search-box.js

+42-47
Original file line numberDiff line numberDiff line change
@@ -1,63 +1,58 @@
1-
var React = require('react');
2-
31
var utils = require('../lib/utils.js');
4-
var headerFooter = require('../decorators/headerFooter');
5-
var SearchBox = headerFooter(require('../components/SearchBox'));
2+
var forEach = require('lodash/collection/forEach');
63

74
/**
85
* Instantiate a searchbox
96
* @param {String|DOMElement} options.container CSS Selector or DOMElement to insert the widget
107
* @param {String} [options.placeholder='Search here'] Input's placeholder
11-
* @param {Object} [options.cssClasses] CSS classes to add to the wrapping elements: root, input
12-
* @param {String|String[]} [options.cssClasses.root=null]
13-
* @param {String|String[]} [options.cssClasses.input=null]
14-
* @param {Object} [options.templates] Templates to use for the widget
15-
* @param {String|Function} [options.templates.header=''] Header template
16-
* @param {String|Function} [options.templates.footer=''] Footer template
8+
* @param {Object} [options.cssClass] CSS classes to add to the input
179
* @param {boolean} [poweredBy=false] Show a powered by Algolia link below the input
1810
* @return {Object}
1911
*/
20-
function searchbox({
21-
container = null,
22-
placeholder = 'Search here',
23-
cssClasses = {
24-
input: null,
25-
root: null
26-
},
27-
templates = {
28-
header: '',
29-
footer: ''
30-
},
31-
poweredBy = false
32-
}) {
33-
var containerNode = utils.getContainerNode(container);
34-
var isFocused = false;
12+
function searchbox(params) {
13+
// Hook on an existing input, or add one if none targeted
14+
var input = utils.getContainerNode(params.container);
15+
if (input.tagName !== 'INPUT') {
16+
input = input.appendChild(document.createElement('input'));
17+
}
3518

3619
return {
37-
_render(state, helper) {
38-
React.render(
39-
<SearchBox
40-
onFocus={()=> { isFocused = true; }}
41-
onBlur={()=> { isFocused = false; }}
42-
setQuery={helper.setQuery.bind(helper)}
43-
search={helper.search.bind(helper)}
44-
placeholder={placeholder}
45-
templates={templates}
46-
cssClasses={cssClasses}
47-
value={state.query}
48-
poweredBy={poweredBy}
49-
/>,
50-
containerNode
51-
);
52-
},
20+
init: function(initialState, helper) {
21+
var defaultAttributes = {
22+
autocapitalize: 'off',
23+
autocomplete: 'off',
24+
autocorrect: 'off',
25+
autofocus: 'autofocus',
26+
className: params.cssClass,
27+
placeholder: params.placeholder,
28+
role: 'textbox',
29+
spellcheck: 'false',
30+
type: 'text',
31+
value: initialState.query
32+
};
33+
34+
// Overrides attributes if not already set
35+
forEach(defaultAttributes, (value, key) => {
36+
if (input.getAttribute(key)) {
37+
return;
38+
}
39+
input.setAttribute(key, value);
40+
});
41+
42+
// Always add our own classes
43+
input.classList.add('as-search-box__input');
5344

54-
init(initialState, helper) {
55-
this._render(initialState, helper);
56-
},
45+
input.addEventListener('keyup', () => {
46+
helper.setQuery(input.value).search();
47+
});
5748

58-
render({state, helper}) {
59-
if (!isFocused) {
60-
this._render(state, helper);
49+
// Optional "powered by Algolia" widget
50+
if (params.poweredBy) {
51+
var React = require('react');
52+
var PoweredBy = require('../components/PoweredBy');
53+
var poweredByContainer = document.createElement('div');
54+
input.parentNode.appendChild(poweredByContainer);
55+
React.render(<PoweredBy display />, poweredByContainer);
6156
}
6257
}
6358
};

0 commit comments

Comments
 (0)