Skip to content
This repository has been archived by the owner on Dec 30, 2022. It is now read-only.

Commit

Permalink
fix(widget): prevent component recreation using key
Browse files Browse the repository at this point in the history
  • Loading branch information
Haroenv committed Apr 20, 2021
1 parent 8b945a7 commit 2dcf646
Showing 1 changed file with 29 additions and 124 deletions.
153 changes: 29 additions & 124 deletions src/components/DynamicWidgets.js
Original file line number Diff line number Diff line change
@@ -1,124 +1,27 @@
import { createWidgetMixin } from '../mixins/widget';

function getAttribute(widget, initOptions) {
try {
const { widgetParams } = widget.getWidgetRenderState(initOptions);

const attribute =
'attribute' in widgetParams
? widgetParams.attribute
: widgetParams.attributes[0];

if (typeof attribute !== 'string') throw new Error();

return attribute;
} catch (e) {
throw new Error(
`Could not find the attribute of the widget:
${JSON.stringify(widget)}
Please check whether the widget's getWidgetRenderState returns widgetParams.attribute correctly.`
);
}
}

const connectDynamicWidgets = function connectDynamicWidgets(
renderFn,
unmountFn = () => {}
) {
return widgetParams => {
const { widgets, transformItems = items => items } = widgetParams;

const localWidgets = new Map();

return {
$$type: 'ais.dynamicWidgets',
init(initOptions) {
widgets.forEach(widget => {
const attribute = getAttribute(widget, initOptions);
localWidgets.set(attribute, { widget, isMounted: false });
});
},
render(renderOptions) {
const { parent } = renderOptions;
const renderState = this.getWidgetRenderState(renderOptions);

const widgetsToUnmount = [];
const widgetsToMount = [];

localWidgets.forEach(({ widget, isMounted }, attribute) => {
const shouldMount =
renderState.attributesToRender.indexOf(attribute) > -1;

if (!isMounted && shouldMount) {
widgetsToMount.push(widget);
localWidgets.set(attribute, {
widget,
isMounted: true,
});
} else if (isMounted && !shouldMount) {
widgetsToUnmount.push(widget);

localWidgets.set(attribute, {
widget,
isMounted: false,
});
}
});

parent.addWidgets(widgetsToMount);
// make sure this only happens after the regular render, otherwise it
// happens too quick, since render is "deferred" for the next microtask,
// so this needs to be a whole task later
setTimeout(() => parent.removeWidgets(widgetsToUnmount), 0);

renderFn(
Object.assign({}, renderState, {
instantSearchInstance: renderOptions.instantSearchInstance,
}),
false
);
},
dispose() {
unmountFn();
},
getRenderState(renderState) {
// @TODO: decide whether it makes sense to return anything,
// knowing that you could have multiple instances of dynamic
// and they are hard/impossible to distinguish at this point
return renderState;
},
getWidgetRenderState({ results }) {
if (!results) {
return { attributesToRender: [], widgetParams };
}

// retrieve the facet order out of the results:
// results.facetOrder.map(facet => facet.attribute)
const attributesToRender = transformItems([], results);

return { attributesToRender, widgetParams };
},
};
};
};
import { connectDynamicWidgets } from 'instantsearch.js/es/connectors';

function getVueAttribute(vnode) {
if (!vnode.componentOptions) {
return undefined;
if (vnode.componentOptions) {
if (vnode.componentOptions.propsData.attribute) {
return vnode.componentOptions.propsData.attribute;
}
if (
vnode.componentOptions &&
Array.isArray(vnode.componentOptions.propsData.attributes)
) {
return vnode.componentOptions.propsData.attributes[0];
}
}

if (vnode.componentOptions.propsData.attribute) {
return vnode.componentOptions.propsData.attribute;
}
if (Array.isArray(vnode.componentOptions.propsData.attributes)) {
return vnode.componentOptions.propsData.attributes[0];
}
const children =
vnode.componentOptions && vnode.componentOptions.children
? vnode.componentOptions.children
: vnode.children;

if (Array.isArray(vnode.componentOptions.children)) {
if (Array.isArray(children)) {
// return first child with a truthy attribute
return vnode.componentOptions.children.reduce(
return children.reduce(
(acc, curr) => (acc ? acc : getVueAttribute(curr)),
undefined
);
Expand All @@ -128,7 +31,7 @@ function getVueAttribute(vnode) {
}

export default {
name: 'AisDynamicWidgets',
name: 'AisDynamicCustom',
mixins: [createWidgetMixin({ connector: connectDynamicWidgets })],
props: {
transformItems: {
Expand All @@ -139,20 +42,22 @@ export default {
},
},
render(createElement) {
if (!this.state) {
return createElement(
'div',
{ attrs: { hidden: true } },
this.$slots.default
);
}

const components = new Map();
this.$slots.default.forEach(vnode => {
const attribute = getVueAttribute(vnode);
if (attribute) components.set(attribute, vnode);
if (attribute) {
components.set(attribute, Object.assign({}, vnode, { key: attribute }));
}
});

// by default, render everything, but hidden so that the routing doesn't disappear
if (!this.state) {
const allComponents = [];
components.forEach(vnode => allComponents.push(vnode));

return createElement('div', { attrs: { hidden: true } }, allComponents);
}

return createElement(
'div',
{},
Expand Down

0 comments on commit 2dcf646

Please sign in to comment.