Skip to content

Commit 58275dc

Browse files
author
vvo
committed
feat(menu,refinementList): add header/item/footer templating solution
fixes #101 BREAKING CHANGE: Removed from menu and refinementList: - rootClass => cssClasses.root - itemCLass => cssClasses.item - template => templates.item Added to menu and refinementList: - cssClasses{root,list,item} - templates{header,item,footer} - widget (container) is automatically hidden by default - hideWhenNoResults=true This was done to allow more templating solutions like discussed in #101.
1 parent 827e779 commit 58275dc

File tree

10 files changed

+202
-72
lines changed

10 files changed

+202
-72
lines changed

README.md

+26-12
Original file line numberDiff line numberDiff line change
@@ -367,12 +367,20 @@ search.addWidget(
367367
* @param {String} options.facetName Name of the attribute for faceting
368368
* @param {String} options.operator How to apply refinements. Possible values: `or`, `and`
369369
* @param {String[]} [options.sortBy=['count:desc']] How to sort refinements. Possible values: `count|isRefined|name:asc|desc`
370-
* @param {String} [options.limit=100] How much facet values to get.
371-
* @param {String|String[]} [options.rootClass=null] CSS class(es) for the root `<ul>` element
372-
* @param {String|String[]} [options.itemClass=null] CSS class(es) for the item `<li>` element
373-
* @param {String|Function} [options.template] Item template, provided with `name`, `count`, `isRefined`
370+
* @param {String} [options.limit=100] How much facet values to get
371+
* @param {Object} [options.cssClasses] Css classes to add to the wrapping elements: root, list, item
372+
* @param {String|String[]} [options.cssClasses.root]
373+
* @param {String|String[]} [options.cssClasses.list]
374+
* @param {String|String[]} [options.cssClasses.item]
375+
* @param {Object} [options.templates] Templates to use for the widget
376+
* @param {String|Function} [options.templates.header] Header template
377+
* @param {String|Function} [options.templates.item=`<label>
378+
<input type="checkbox" value="{{name}}" {{#isRefined}}checked{{/isRefined}} />{{name}} <span>{{count}}</span>
379+
</label>`] Item template, provided with `name`, `count`, `isRefined`
380+
* @param {String|Function} [options.templates.footer] Footer template
374381
* @param {String|Function} [options.singleRefine=true] Are multiple refinements allowed or only one at the same time. You can use this
375-
* to build radio based refinement lists for example.
382+
* to build radio based refinement lists for example
383+
* @param {boolean} [hideWhenNoResults=true] Hide the container when no results match
376384
* @return {Object}
377385
*/
378386
```
@@ -406,10 +414,16 @@ search.addWidget(
406414
* @param {String|DOMElement} options.container Valid CSS Selector as a string or DOMElement
407415
* @param {String} options.facetName Name of the attribute for faceting
408416
* @param {String[]} [options.sortBy=['count:desc']] How to sort refinements. Possible values: `count|isRefined|name:asc|desc`
409-
* @param {String} [options.limit=100] How much facet values to get.
410-
* @param {String|String[]} [options.rootClass=null] CSS class(es) for the root `<ul>` element
411-
* @param {String|String[]} [options.itemClass=null] CSS class(es) for the item `<li>` element
412-
* @param {String|Function} [options.template] Item template, provided with `name`, `count`, `isRefined`
417+
* @param {String} [options.limit=100] How much facet values to get
418+
* @param {Object} [options.cssClasses] Css classes to add to the wrapping elements: root, list, item
419+
* @param {String|String[]} [options.cssClasses.root]
420+
* @param {String|String[]} [options.cssClasses.list]
421+
* @param {String|String[]} [options.cssClasses.item]
422+
* @param {Object} [options.templates] Templates to use for the widget
423+
* @param {String|Function} [options.templates.header=''] Header template
424+
* @param {String|Function} [options.templates.item='<a href="{{href}}">{{name}}</a> {{count}}'] Item template, provided with `name`, `count`, `isRefined`
425+
* @param {String|Function} [options.templates.footer=''] Footer template
426+
* @param {boolean} [hideWhenNoResults=true] Hide the container when no results match
413427
* @return {Object}
414428
*/
415429
```
@@ -472,13 +486,13 @@ search.addWidget(
472486

473487
## Browser support
474488

475-
We support IE9+ and all other modern browsers.
489+
We support IE10+ and all other modern browsers.
476490

477-
To get IE8 support, please insert this in the `<head>`:
491+
To get < IE10 support, please insert this in the `<head>`:
478492

479493
```html
480494
<meta http-equiv="X-UA-Compatible" content="IE=Edge">
481-
<!--[if lte IE 8]>
495+
<!--[if lte IE 9]>
482496
<script src="https://cdnjs.cloudflare.com/ajax/libs/aight/1.2.2/aight.min.js"></script>
483497
<![endif]-->
484498
```

components/RefinementList.js

+48-16
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
var React = require('react');
22

33
var Template = require('./Template');
4+
var cx = require('classnames');
45

56
class RefinementList extends React.Component {
67
refine(value) {
@@ -38,37 +39,68 @@ class RefinementList extends React.Component {
3839

3940
render() {
4041
var facetValues = this.props.facetValues;
41-
var template = this.props.template;
42+
var templates = this.props.templates;
43+
var rootClass = cx(this.props.cssClasses.root);
44+
var listClass = cx(this.props.cssClasses.list);
45+
var itemClass = cx(this.props.cssClasses.item);
4246

4347
return (
44-
<div className={this.props.rootClass}>
48+
<div className={rootClass}>
49+
<Template template={templates.header} />
50+
<div className={listClass}>
4551
{facetValues.map(facetValue => {
4652
return (
47-
<div className={this.props.itemClass} key={facetValue.name} onClick={this.handleClick.bind(this, facetValue.name)}>
48-
<Template data={facetValue} template={template} />
53+
<div className={itemClass} key={facetValue.name} onClick={this.handleClick.bind(this, facetValue.name)}>
54+
<Template data={facetValue} template={templates.item} />
4955
</div>
5056
);
5157
})}
58+
</div>
59+
<Template template={templates.footer} />
5260
</div>
5361
);
5462
}
5563
}
5664

5765
RefinementList.propTypes = {
58-
rootClass: React.PropTypes.oneOfType([
59-
React.PropTypes.string,
60-
React.PropTypes.arrayOf(React.PropTypes.string)
61-
]),
62-
itemClass: React.PropTypes.oneOfType([
63-
React.PropTypes.string,
64-
React.PropTypes.arrayOf(React.PropTypes.string)
65-
]),
66+
cssClasses: React.PropTypes.shape({
67+
root: React.PropTypes.oneOfType([
68+
React.PropTypes.string,
69+
React.PropTypes.arrayOf(React.PropTypes.string)
70+
]),
71+
item: React.PropTypes.oneOfType([
72+
React.PropTypes.string,
73+
React.PropTypes.arrayOf(React.PropTypes.string)
74+
]),
75+
list: React.PropTypes.oneOfType([
76+
React.PropTypes.string,
77+
React.PropTypes.arrayOf(React.PropTypes.string)
78+
])
79+
}),
6680
facetValues: React.PropTypes.array,
67-
template: React.PropTypes.oneOfType([
68-
React.PropTypes.string,
69-
React.PropTypes.func
70-
]).isRequired,
81+
templates: React.PropTypes.shape({
82+
header: React.PropTypes.oneOfType([
83+
React.PropTypes.string,
84+
React.PropTypes.func
85+
]),
86+
item: React.PropTypes.oneOfType([
87+
React.PropTypes.string,
88+
React.PropTypes.func
89+
]).isRequired,
90+
footer: React.PropTypes.oneOfType([
91+
React.PropTypes.string,
92+
React.PropTypes.func
93+
])
94+
}),
7195
toggleRefinement: React.PropTypes.func.isRequired
7296
};
7397

98+
RefinementList.defaultProps = {
99+
cssClasses: {
100+
root: null,
101+
item: null,
102+
list: null
103+
}
104+
};
105+
74106
module.exports = RefinementList;

components/Template.js

+4
Original file line numberDiff line numberDiff line change
@@ -23,4 +23,8 @@ Template.propTypes = {
2323
data: React.PropTypes.object
2424
};
2525

26+
Template.defaultProps = {
27+
data: {}
28+
};
29+
2630
module.exports = Template;

example/app.js

+21-5
Original file line numberDiff line numberDiff line change
@@ -58,8 +58,13 @@ search.addWidget(
5858
facetName: 'brand',
5959
operator: 'or',
6060
limit: 10,
61-
rootClass: 'nav nav-stacked',
62-
template: require('./templates/or.html')
61+
cssClasses: {
62+
list: 'nav nav-stacked panel-body'
63+
},
64+
templates: {
65+
header: '<div class="panel-heading">Brands</div>',
66+
item: require('./templates/or.html')
67+
}
6368
})
6469
);
6570

@@ -69,8 +74,13 @@ search.addWidget(
6974
facetName: 'price_range',
7075
operator: 'and',
7176
limit: 10,
72-
rootClass: 'nav nav-stacked',
73-
template: require('./templates/and.html')
77+
cssClasses: {
78+
root: 'list-group'
79+
},
80+
templates: {
81+
header: '<div class="panel-heading">Price ranges</div>',
82+
item: require('./templates/and.html')
83+
}
7484
})
7585
);
7686

@@ -88,7 +98,13 @@ search.addWidget(
8898
container: '#categories',
8999
facetName: 'categories',
90100
limit: 10,
91-
template: require('./templates/category.html')
101+
cssClasses: {
102+
root: 'list-group'
103+
},
104+
templates: {
105+
header: '<div class="panel-heading">Categories</div>',
106+
item: require('./templates/category.html')
107+
}
92108
})
93109
);
94110

example/index.html

+3-12
Original file line numberDiff line numberDiff line change
@@ -19,20 +19,11 @@ <h1>Instant search demo <small>using instantsearch.js</small></h1>
1919
<div class="panel-body" id="search-box"></div>
2020
</div>
2121

22-
<div class="panel panel-default">
23-
<div class="panel-heading">Categories</div>
24-
<div id="categories" class="list-group"></div>
25-
</div>
22+
<div class="panel panel-default" id="categories"></div>
2623

27-
<div class="panel panel-default">
28-
<div class="panel-heading">Brands</div>
29-
<div class="panel-body" id="brands"></div>
30-
</div>
24+
<div class="panel panel-default" id="brands"></div>
3125

32-
<div class="panel panel-default">
33-
<div class="panel-heading">Price Ranges</div>
34-
<div class="list-group" id="price_range"></div>
35-
</div>
26+
<div class="panel panel-default" id="price-range"></div>
3627

3728
<div class="panel panel-default">
3829
<div class="panel-heading">Shipping</div>

example/style.css

+5
Original file line numberDiff line numberDiff line change
@@ -39,3 +39,8 @@ body {
3939
#price {
4040
padding: 30px 0;
4141
}
42+
43+
.panel-heading {
44+
background-image: linear-gradient(to bottom,#f5f5f5 0,#e8e8e8 100%);
45+
border-color: #ddd;
46+
}

index.js

+2
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,8 @@ var toFactory = require('to-factory');
33
var InstantSearch = require('./lib/InstantSearch');
44
var instantsearch = toFactory(InstantSearch);
55

6+
require('style?prepend!raw!./lib/style.css');
7+
68
instantsearch.widgets = {
79
hits: require('./widgets/hits'),
810
indexSelector: require('./widgets/index-selector'),

lib/style.css

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
.as-display-none {
2+
display: none;
3+
}

widgets/menu.js

+44-13
Original file line numberDiff line numberDiff line change
@@ -1,31 +1,45 @@
11
var React = require('react');
2-
var cx = require('classnames');
32

43
var utils = require('../lib/utils.js');
54

6-
var defaultTemplate = `<a href="{{href}}">{{name}}</a> {{count}}`;
5+
6+
var defaultTemplates = {
7+
header: '',
8+
footer: '',
9+
item: '<a href="{{href}}">{{name}}</a> {{count}}'
10+
};
711

812
var hierarchicalCounter = 0;
913

1014
/**
11-
* Instantiate a list of refinements based on a facet
15+
* Create a menu out of a facet
1216
* @param {String|DOMElement} options.container Valid CSS Selector as a string or DOMElement
1317
* @param {String} options.facetName Name of the attribute for faceting
1418
* @param {String[]} [options.sortBy=['count:desc']] How to sort refinements. Possible values: `count|isRefined|name:asc|desc`
15-
* @param {String} [options.limit=100] How much facet values to get.
16-
* @param {String|String[]} [options.rootClass=null] CSS class(es) for the root `<ul>` element
17-
* @param {String|String[]} [options.itemClass=null] CSS class(es) for the item `<li>` element
18-
* @param {String|Function} [options.template] Item template, provided with `name`, `count`, `isRefined`
19+
* @param {String} [options.limit=100] How much facet values to get
20+
* @param {Object} [options.cssClasses] Css classes to add to the wrapping elements: root, list, item
21+
* @param {String|String[]} [options.cssClasses.root]
22+
* @param {String|String[]} [options.cssClasses.list]
23+
* @param {String|String[]} [options.cssClasses.item]
24+
* @param {Object} [options.templates] Templates to use for the widget
25+
* @param {String|Function} [options.templates.header=''] Header template
26+
* @param {String|Function} [options.templates.item='<a href="{{href}}">{{name}}</a> {{count}}'] Item template, provided with `name`, `count`, `isRefined`
27+
* @param {String|Function} [options.templates.footer=''] Footer template
28+
* @param {boolean} [hideWhenNoResults=true] Hide the container when no results match
1929
* @return {Object}
2030
*/
2131
function menu({
2232
container = null,
2333
facetName = null,
2434
sortBy = ['count:desc'],
2535
limit = 100,
26-
rootClass = null,
27-
itemClass = null,
28-
template = defaultTemplate
36+
cssClasses = {
37+
root: null,
38+
list: null,
39+
item: null
40+
},
41+
hideWhenNoResults = true,
42+
templates = defaultTemplates
2943
}) {
3044
hierarchicalCounter++;
3145

@@ -38,6 +52,10 @@ function menu({
3852
throw new Error(usage);
3953
}
4054

55+
if (templates !== defaultTemplates) {
56+
templates = Object.assign({}, defaultTemplates, templates);
57+
}
58+
4159
var hierarchicalFacetName = 'instantsearch.js' + hierarchicalCounter;
4260

4361
return {
@@ -48,12 +66,25 @@ function menu({
4866
}]
4967
}),
5068
render: function({results, helper}) {
69+
var values = getFacetValues(results, hierarchicalFacetName, sortBy, limit);
70+
71+
if (values.length === 0) {
72+
React.render(<div/>, containerNode);
73+
if (hideWhenNoResults === true) {
74+
containerNode.classList.add('as-display-none');
75+
}
76+
return;
77+
}
78+
79+
if (hideWhenNoResults === true) {
80+
containerNode.classList.remove('as-display-none');
81+
}
82+
5183
React.render(
5284
<RefinementList
53-
rootClass={cx(rootClass)}
54-
itemClass={cx(itemClass)}
85+
cssClasses={cssClasses}
5586
facetValues={getFacetValues(results, hierarchicalFacetName, sortBy, limit)}
56-
template={template}
87+
templates={templates}
5788
toggleRefinement={toggleRefinement.bind(null, helper, hierarchicalFacetName)}
5889
/>,
5990
containerNode

0 commit comments

Comments
 (0)