Skip to content

Commit 3b8688a

Browse files
author
Alexandre Stanislawski
committed
feat(refinementlist): lets configure showmore feature
1 parent c4bf8ec commit 3b8688a

File tree

12 files changed

+193
-27
lines changed

12 files changed

+193
-27
lines changed

dev/app.js

+25
Original file line numberDiff line numberDiff line change
@@ -114,6 +114,31 @@ search.addWidget(
114114
container: '#brands',
115115
attributeName: 'brand',
116116
operator: 'or',
117+
limit: 3,
118+
cssClasses: {
119+
header: 'facet-title',
120+
item: 'facet-value checkbox',
121+
count: 'facet-count pull-right',
122+
active: 'facet-active'
123+
},
124+
templates: {
125+
header: 'Brands'
126+
},
127+
showMore: {
128+
templates: {
129+
'active': '<button>Show less</button>',
130+
'inactive': '<button>Show more</button>'
131+
},
132+
limit: 10
133+
}
134+
})
135+
);
136+
137+
search.addWidget(
138+
instantsearch.widgets.refinementList({
139+
container: '#brands-2',
140+
attributeName: 'brand',
141+
operator: 'or',
117142
limit: 10,
118143
cssClasses: {
119144
header: 'facet-title',

dev/index.html

+1
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@ <h1><a href="./">Instant search demo</a> <small>using instantsearch.js</small></
2424
<div class="facet" id="current-refined-values"></div>
2525
<div class="facet" id="hierarchical-categories"></div>
2626
<div class="facet" id="brands"></div>
27+
<div class="facet" id="brands-2"></div>
2728
<div class="facet" id="price-range"></div>
2829
<div class="facet" id="free-shipping"></div>
2930
<div class="facet" id="price"></div>

dev/style.css

+4
Original file line numberDiff line numberDiff line change
@@ -118,3 +118,7 @@ body {
118118
.ais-refinement-list--label input[type=radio]{
119119
margin: 4px 8px 0 0;
120120
}
121+
122+
.ais-showmore__active, .ais-showmore__inactive {
123+
cursor: pointer;
124+
}

src/components/RefinementList/RefinementList.js

+29-3
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,13 @@ import {isSpecialClick} from '../../lib/utils.js';
55
import Template from '../Template.js';
66

77
class RefinementList extends React.Component {
8+
constructor(props) {
9+
super(props);
10+
this.state = {
11+
isShowMoreOpen: false
12+
};
13+
}
14+
815
refine(value) {
916
this.props.toggleRefinement(value);
1017
}
@@ -45,7 +52,7 @@ class RefinementList extends React.Component {
4552
<div
4653
className={cssClassItem}
4754
key={key}
48-
onClick={this.handleClick.bind(this, facetValue[this.props.attributeNameKey])}
55+
onClick={this.handleItemClick.bind(this, facetValue[this.props.attributeNameKey])}
4956
>
5057
<Template data={templateData} templateKey="item" {...this.props.templateProps} />
5158
{subList}
@@ -69,7 +76,7 @@ class RefinementList extends React.Component {
6976
//
7077
// Finally, we always stop propagation of the event to avoid multiple levels RefinementLists to fail: click
7178
// on child would click on parent also
72-
handleClick(value, e) {
79+
handleItemClick(value, e) {
7380
if (isSpecialClick(e)) {
7481
// do not alter the default browser behavior
7582
// if one special key is down
@@ -101,16 +108,32 @@ class RefinementList extends React.Component {
101108
this.refine(value);
102109
}
103110

111+
handleClickShowMore() {
112+
const isShowMoreOpen = !this.state.isShowMoreOpen;
113+
this.setState({isShowMoreOpen});
114+
}
115+
104116
render() {
105117
// Adding `-lvl0` classes
106118
let cssClassList = [this.props.cssClasses.list];
107119
if (this.props.cssClasses.depth) {
108120
cssClassList.push(`${this.props.cssClasses.depth}${this.props.depth}`);
109121
}
110122

123+
const limit = this.state.isShowMoreOpen ? this.props.limitMax : this.props.limitMin;
124+
const showmoreBtn =
125+
this.props.showMore ?
126+
<Template
127+
onClick={() => this.handleClickShowMore()}
128+
templateKey={'showmore-' + (this.state.isShowMoreOpen ? 'active' : 'inactive')}
129+
{...this.props.templateProps}
130+
/> :
131+
undefined;
132+
111133
return (
112134
<div className={cx(cssClassList)}>
113-
{this.props.facetValues.map(this._generateFacetItem, this)}
135+
{this.props.facetValues.map(this._generateFacetItem, this).slice(0, limit)}
136+
{showmoreBtn}
114137
</div>
115138
);
116139
}
@@ -128,6 +151,9 @@ RefinementList.propTypes = {
128151
}),
129152
depth: React.PropTypes.number,
130153
facetValues: React.PropTypes.array,
154+
limitMax: React.PropTypes.number,
155+
limitMin: React.PropTypes.number,
156+
showMore: React.PropTypes.bool,
131157
templateProps: React.PropTypes.object.isRequired,
132158
toggleRefinement: React.PropTypes.func.isRequired
133159
};

src/components/RefinementList/__tests__/RefinementList-test.js

+23-4
Original file line numberDiff line numberDiff line change
@@ -56,8 +56,8 @@ describe('RefinementList', () => {
5656
</div>
5757
</div>
5858
);
59-
expect(out.props.children[0].key).toEqual('facet1');
60-
expect(out.props.children[1].key).toEqual('facet2');
59+
expect(out.props.children[0][0].key).toEqual('facet1');
60+
expect(out.props.children[0][1].key).toEqual('facet2');
6161
});
6262

6363
it('should render default list highlighted', () => {
@@ -75,7 +75,26 @@ describe('RefinementList', () => {
7575
</div>
7676
</div>
7777
);
78-
expect(out.props.children[0].key).toEqual('facet1/true/42');
78+
expect(out.props.children[0][0].key).toEqual('facet1/true/42');
79+
});
80+
81+
context('showmore', () => {
82+
it('should display the number accordingly to the state : closed', () => {
83+
const out = render({
84+
facetValues: [
85+
{name: 'facet1', isRefined: false, count: 42},
86+
{name: 'facet2', isRefined: false, count: 42}
87+
],
88+
showMore: true,
89+
limitMin: 1,
90+
limitMax: 2
91+
});
92+
expect(out.props.children.length).toBe(2);
93+
});
94+
95+
it('should display the number accordingly to the state : open', () => {
96+
// FIXME find a way to test this state...
97+
});
7998
});
8099

81100
context('sublist', () => {
@@ -140,7 +159,7 @@ describe('RefinementList', () => {
140159

141160
function render(extraProps = {}) {
142161
let props = getProps(extraProps);
143-
renderer.render(<RefinementList {...props} templateProps={{}} />);
162+
renderer.render(<RefinementList {...props} ref="list" templateProps={{}} />);
144163
return renderer.getRenderOutput();
145164
}
146165

src/components/Template.js

+12-8
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,8 @@ function Template(props) {
1313
const compileOptions = useCustomCompileOptions ? props.templatesConfig.compileOptions : {};
1414

1515
const content = renderTemplate({
16-
template: props.templates[props.templateKey],
16+
templates: props.templates,
17+
templateKey: props.templateKey,
1718
compileOptions: compileOptions,
1819
helpers: props.templatesConfig.helpers,
1920
data: transformData(props.transformData, props.templateKey, props.data)
@@ -79,9 +80,10 @@ function transformData(fn, templateKey, originalData) {
7980
let clonedData = cloneDeep(originalData);
8081

8182
let data;
82-
if (typeof fn === 'function') {
83+
const typeFn = typeof fn;
84+
if (typeFn === 'function') {
8385
data = fn(clonedData);
84-
} else if (typeof fn === 'object') {
86+
} else if (typeFn === 'object') {
8587
// ex: transformData: {hit, empty}
8688
if (fn[templateKey]) {
8789
data = fn[templateKey](clonedData);
@@ -91,7 +93,7 @@ function transformData(fn, templateKey, originalData) {
9193
data = originalData;
9294
}
9395
} else {
94-
throw new Error('`transformData` must be a function or an object');
96+
throw new Error(`transformData must be a function or an object, was ${typeFn} (key : ${templateKey})`);
9597
}
9698

9799
let dataType = typeof data;
@@ -102,12 +104,14 @@ function transformData(fn, templateKey, originalData) {
102104
return data;
103105
}
104106

105-
function renderTemplate({template, compileOptions, helpers, data}) {
106-
let isTemplateString = typeof template === 'string';
107-
let isTemplateFunction = typeof template === 'function';
107+
function renderTemplate({templates, templateKey, compileOptions, helpers, data}) {
108+
const template = templates[templateKey];
109+
const templateType = typeof template;
110+
const isTemplateString = templateType === 'string';
111+
const isTemplateFunction = templateType === 'function';
108112

109113
if (!isTemplateString && !isTemplateFunction) {
110-
throw new Error('Template must be `string` or `function`');
114+
throw new Error(`Template must be 'string' or 'function', was '${templateType}' (key: ${templateKey})`);
111115
} else if (isTemplateFunction) {
112116
return template(data);
113117
} else {

src/components/__tests__/Template-test.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@ expect.extend(expectJSX);
1010

1111
let {createRenderer} = TestUtils;
1212

13-
describe.only('Template', () => {
13+
describe('Template', () => {
1414
let renderer;
1515

1616
beforeEach(() => {

src/css/default/_refinement-list.scss

+10
Original file line numberDiff line numberDiff line change
@@ -30,3 +30,13 @@
3030
/* widget footer */
3131
}
3232
}
33+
34+
/* Sub block for the show more of the refinement list */
35+
@include block(showmore) {
36+
@include modifier(active) {
37+
/* Show more button is activated */
38+
}
39+
@include modifier(inactive) {
40+
/* Show more button is deactivated */
41+
}
42+
}

src/lib/utils.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -114,7 +114,7 @@ function prepareTemplateProps({
114114
};
115115
}
116116

117-
function prepareTemplates(defaultTemplates = [], templates = []) {
117+
function prepareTemplates(defaultTemplates = {}, templates = {}) {
118118
const allKeys = uniq([...(keys(defaultTemplates)), ...(keys(templates))]);
119119

120120
return reduce(allKeys, (config, key) => {

src/widgets/refinement-list/__tests__/refinement-list-test.js

+21
Original file line numberDiff line numberDiff line change
@@ -292,4 +292,25 @@ describe('refinementList()', () => {
292292
// Then
293293
});
294294
});
295+
296+
context('show more', () => {
297+
it('should return a configuration with the highest limit value (default value)', () => {
298+
const opts = {container, attributeName: 'attributeName', limit: 1, showMore: {}};
299+
const wdgt = refinementList(opts);
300+
const partialConfig = wdgt.getConfiguration({});
301+
expect(partialConfig.maxValuesPerFacet).toBe(100);
302+
});
303+
304+
it('should return a configuration with the highest limit value (custom value)', () => {
305+
const opts = {container, attributeName: 'attributeName', limit: 1, showMore: {limit: 99}};
306+
const wdgt = refinementList(opts);
307+
const partialConfig = wdgt.getConfiguration({});
308+
expect(partialConfig.maxValuesPerFacet).toBe(opts.showMore.limit);
309+
});
310+
311+
it('should not accept a show more limit that is < limit', () => {
312+
const opts = {container, attributeName: 'attributeName', limit: 100, showMore: {limit: 1}};
313+
expect(() => refinementList(opts)).toThrow();
314+
});
315+
});
295316
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
export default {
2+
active: '<a class="ais-showmore ais-showmore__active">Show less</a>',
3+
inactive: '<a class="ais-showmore ais-showmore__inactive">Show more</a>'
4+
};

0 commit comments

Comments
 (0)