@remote-ui/core
Advanced tools
Comparing version 1.5.4 to 1.6.0-alpha.0
@@ -77,7 +77,5 @@ "use strict"; | ||
var initialProps = rest[0], | ||
initialChildren = rest[1], | ||
moreChildren = rest.slice(2); | ||
var normalizedInitialProps = initialProps !== null && initialProps !== void 0 ? initialProps : {}; | ||
initialChildren = rest[1]; | ||
var normalizedInitialChildren = []; | ||
var normalizedInternalProps = {}; | ||
var normalizedInitialProps = {}; | ||
@@ -96,3 +94,3 @@ if (initialProps) { | ||
if (_key2 === 'children') continue; | ||
normalizedInternalProps[_key2] = makeValueHotSwappable(initialProps[_key2]); | ||
normalizedInitialProps[_key2] = makeValueHotSwappable(initialProps[_key2]); | ||
} | ||
@@ -102,35 +100,14 @@ } | ||
if (initialChildren) { | ||
if (Array.isArray(initialChildren)) { | ||
var _iterator = _createForOfIteratorHelper(initialChildren), | ||
_step; | ||
var _iterator = _createForOfIteratorHelper(initialChildren), | ||
_step; | ||
try { | ||
for (_iterator.s(); !(_step = _iterator.n()).done;) { | ||
var child = _step.value; | ||
normalizedInitialChildren.push(normalizeChild(child, remoteRoot)); | ||
} | ||
} catch (err) { | ||
_iterator.e(err); | ||
} finally { | ||
_iterator.f(); | ||
try { | ||
for (_iterator.s(); !(_step = _iterator.n()).done;) { | ||
var child = _step.value; | ||
normalizedInitialChildren.push(normalizeChild(child, remoteRoot)); | ||
} | ||
} else { | ||
normalizedInitialChildren.push(normalizeChild(initialChildren, remoteRoot)); // The complex tuple type of `rest` makes it so `moreChildren` is | ||
// incorrectly inferred as potentially being the props of the component, | ||
// lazy casting since we know it will be an array of child elements | ||
// (or empty). | ||
var _iterator2 = _createForOfIteratorHelper(moreChildren), | ||
_step2; | ||
try { | ||
for (_iterator2.s(); !(_step2 = _iterator2.n()).done;) { | ||
var _child = _step2.value; | ||
normalizedInitialChildren.push(normalizeChild(_child, remoteRoot)); | ||
} | ||
} catch (err) { | ||
_iterator2.e(err); | ||
} finally { | ||
_iterator2.f(); | ||
} | ||
} catch (err) { | ||
_iterator.e(err); | ||
} finally { | ||
_iterator.f(); | ||
} | ||
@@ -141,4 +118,3 @@ } | ||
var internals = { | ||
externalProps: strict ? Object.freeze(normalizedInitialProps) : normalizedInitialProps, | ||
internalProps: normalizedInternalProps, | ||
props: strict ? Object.freeze(normalizedInitialProps) : normalizedInitialProps, | ||
children: strict ? Object.freeze(normalizedInitialChildren) : normalizedInitialChildren | ||
@@ -155,9 +131,5 @@ }; | ||
get props() { | ||
return internals.externalProps; | ||
return internals.props; | ||
}, | ||
get remoteProps() { | ||
return internals.internalProps; | ||
}, | ||
updateProps: function updateProps(newProps) { | ||
@@ -186,14 +158,14 @@ return _updateProps(component, newProps, internals, rootInternals); | ||
var _iterator3 = _createForOfIteratorHelper(internals.children), | ||
_step3; | ||
var _iterator2 = _createForOfIteratorHelper(internals.children), | ||
_step2; | ||
try { | ||
for (_iterator3.s(); !(_step3 = _iterator3.n()).done;) { | ||
var _child2 = _step3.value; | ||
moveChildToContainer(component, _child2, rootInternals); | ||
for (_iterator2.s(); !(_step2 = _iterator2.n()).done;) { | ||
var _child = _step2.value; | ||
moveChildToContainer(component, _child, rootInternals); | ||
} | ||
} catch (err) { | ||
_iterator3.e(err); | ||
_iterator2.e(err); | ||
} finally { | ||
_iterator3.f(); | ||
_iterator2.f(); | ||
} | ||
@@ -236,3 +208,2 @@ | ||
mount: function mount() { | ||
if (rootInternals.mounted) return Promise.resolve(); | ||
rootInternals.mounted = true; | ||
@@ -255,8 +226,8 @@ return Promise.resolve(channel(_types.ACTION_MOUNT, rootInternals.children.map(serialize))); | ||
if ('children' in element) { | ||
var _iterator4 = _createForOfIteratorHelper(element.children), | ||
_step4; | ||
var _iterator3 = _createForOfIteratorHelper(element.children), | ||
_step3; | ||
try { | ||
for (_iterator4.s(); !(_step4 = _iterator4.n()).done;) { | ||
var child = _step4.value; | ||
for (_iterator3.s(); !(_step3 = _iterator3.n()).done;) { | ||
var child = _step3.value; | ||
withEach(child); | ||
@@ -266,5 +237,5 @@ recurse(child); | ||
} catch (err) { | ||
_iterator4.e(err); | ||
_iterator3.e(err); | ||
} finally { | ||
_iterator4.f(); | ||
_iterator3.f(); | ||
} | ||
@@ -316,3 +287,3 @@ } | ||
var strict = rootInternals.strict; | ||
var currentProps = internals.internalProps; | ||
var currentProps = internals.props; | ||
var normalizedNewProps = {}; | ||
@@ -350,19 +321,18 @@ var hotSwapFunctions = []; | ||
if (hasRemoteChange) { | ||
channel(_types.ACTION_UPDATE_PROPS, component.id, normalizedNewProps); | ||
channel(_types.ACTION_UPDATE_PROPS, component.id, newProps); | ||
} | ||
}, | ||
local: function local() { | ||
var mergedExternalProps = _objectSpread(_objectSpread({}, internals.externalProps), newProps); | ||
var mergedProps = _objectSpread(_objectSpread({}, internals.props), newProps); | ||
internals.externalProps = strict ? Object.freeze(mergedExternalProps) : mergedExternalProps; | ||
internals.internalProps = _objectSpread(_objectSpread({}, internals.internalProps), normalizedNewProps); | ||
internals.props = strict ? Object.freeze(mergedProps) : mergedProps; | ||
var _iterator5 = _createForOfIteratorHelper(hotSwapFunctions), | ||
_step5; | ||
var _iterator4 = _createForOfIteratorHelper(hotSwapFunctions), | ||
_step4; | ||
try { | ||
for (_iterator5.s(); !(_step5 = _iterator5.n()).done;) { | ||
var _step5$value = _slicedToArray(_step5.value, 2), | ||
hotSwappable = _step5$value[0], | ||
_newValue = _step5$value[1]; | ||
for (_iterator4.s(); !(_step4 = _iterator4.n()).done;) { | ||
var _step4$value = _slicedToArray(_step4.value, 2), | ||
hotSwappable = _step4$value[0], | ||
_newValue = _step4$value[1]; | ||
@@ -372,86 +342,121 @@ hotSwappable[FUNCTION_CURRENT_IMPLEMENTATION_KEY] = _newValue; | ||
} catch (err) { | ||
_iterator5.e(err); | ||
_iterator4.e(err); | ||
} finally { | ||
_iterator5.f(); | ||
_iterator4.f(); | ||
} | ||
} | ||
}); | ||
} // Imagine the following remote-ui components we might render in a remote context: | ||
// | ||
// const root = createRemoteRoot(); | ||
// const {value, onChange, onPress} = getPropsForValue(); | ||
// | ||
// const textField = root.createComponent('TextField', {value, onChange}); | ||
// const button = root.createComponent('Button', {onPress}); | ||
// | ||
// root.appendChild(textField); | ||
// root.appendChild(button); | ||
// | ||
// function getPropsForValue(value = '') { | ||
// return { | ||
// value, | ||
// onChange: () => { | ||
// const {value, onChange, onPress} = getPropsForValue(); | ||
// textField.updateProps({value, onChange}); | ||
// button.updateProps({onPress}); | ||
// }, | ||
// onPress: () => console.log(value), | ||
// }; | ||
// } | ||
// | ||
// | ||
// In this example, assume that the `TextField` `onChange` prop is run on blur. | ||
// If this were running on the host, the following steps would happen if you pressed | ||
// on the button: | ||
// | ||
// 1. The text field blurs, and so calls `onChange()` with its current value, which | ||
// then calls `setValue()` with the updated value. | ||
// 2. We synchronously update the `value`, `onChange`, and `onPress` props to point at | ||
// the most current `value`. | ||
// 3. Handling blur is finished, so the browser now handles the click by calling the | ||
// (newly-updated) `Button` `onPress()`, which logs out the new value. | ||
// | ||
// Because remote-ui reproduces a UI tree asynchronously from the remote context, the | ||
// steps above run in a different order: | ||
// | ||
// 1. The text field blurs, and so calls `onChange()` with its current value. | ||
// 2. Handling blur is finished **from the perspective of the main thread**, so the | ||
// browser now handles the click by calling the (original) `Button` `onPress()`, which | ||
// logs out the **initial** value. | ||
// 3. In the remote context, we receive the `onChange()` call, which calls updates the props | ||
// on the `Button` and `TextField` to be based on the new `value`, but by now it’s | ||
// already too late for `onPress` — the old version has already been called! | ||
// | ||
// As you can see, the timing issue introduced by the asynchronous nature of remote-ui | ||
// can cause “old props” to be called from the main thread. This example may seem like | ||
// an unusual pattern, and it is if you are using `@remote-ui/core` directly; you’d generally | ||
// keep a mutable reference to the state, instead of closing over the state with new props. | ||
// However, abstractions on top of `@remote-ui/core`, like the React reconciler in | ||
// `@remote-ui/react`, work almost entirely by closing over state, so this issue is | ||
// much more common with those declarative libraries. | ||
// | ||
// To protect against this, we handle function props a bit differently. When we have a | ||
// function prop, we replace it with a new function that calls the original. However, | ||
// we make the original mutable, by making it a property on the function itself. When | ||
// this function subsequently updates, we don’t send the update to the main thread (as | ||
// we just saw, this can often be "too late" to be of any use). Instead, we swap out | ||
// the mutable reference to the current implementation of the function prop, which can | ||
// be done synchronously. In the example above, this would all happen synchronously in | ||
// the remote context; in our handling of `TextField onChange()`, we update `Button onPress()`, | ||
// and swap out the implementations. Now, when the main thread attempts to call `Button onPress()`, | ||
// it instead calls our wrapper around the function, which can refer to, and call, the | ||
// most recently-applied implementation, instead of directly calling the old implementation. | ||
} | ||
function tryHotSwappingValues(currentValue, newValue) { | ||
if (typeof currentValue === 'function' && FUNCTION_CURRENT_IMPLEMENTATION_KEY in currentValue) { | ||
return [typeof newValue === 'function' ? IGNORE : makeValueHotSwappable(newValue), [[currentValue, newValue]]]; | ||
return [typeof newValue === 'function' ? IGNORE : makeValueHotSwappable(newValue), [currentValue, newValue]]; | ||
} | ||
if (Array.isArray(currentValue)) { | ||
return tryHotSwappingArrayValues(currentValue, newValue); | ||
if (!Array.isArray(newValue)) { | ||
var _collectNestedHotSwap; | ||
return [makeValueHotSwappable(newValue), (_collectNestedHotSwap = collectNestedHotSwappableValues(currentValue)) === null || _collectNestedHotSwap === void 0 ? void 0 : _collectNestedHotSwap.map(function (hotSwappable) { | ||
return [hotSwappable, undefined]; | ||
})]; | ||
} | ||
var hasChanged = false; | ||
var hotSwaps = []; | ||
var newLength = newValue.length; | ||
var currentLength = currentValue.length; | ||
var maxLength = Math.max(currentLength, newLength); | ||
var normalizedNewValue = []; | ||
for (var i = 0; i < maxLength; i++) { | ||
var currentElement = currentValue[i]; | ||
var newElement = newValue[i]; | ||
if (i < newLength) { | ||
if (i >= currentLength) { | ||
hasChanged = true; | ||
normalizedNewValue[i] = makeValueHotSwappable(newValue); | ||
} else { | ||
var _tryHotSwappingValues3 = tryHotSwappingValues(currentElement, newElement), | ||
_tryHotSwappingValues4 = _slicedToArray(_tryHotSwappingValues3, 2), | ||
updatedValue = _tryHotSwappingValues4[0], | ||
elementHotSwaps = _tryHotSwappingValues4[1]; | ||
if (elementHotSwaps) hotSwaps.push.apply(hotSwaps, _toConsumableArray(elementHotSwaps)); | ||
if (updatedValue === IGNORE) { | ||
normalizedNewValue[i] = currentValue; | ||
} else { | ||
hasChanged = true; | ||
normalizedNewValue[i] = updatedValue; | ||
} | ||
} | ||
} else { | ||
hasChanged = true; | ||
var nestedHotSwappables = collectNestedHotSwappableValues(currentElement); | ||
if (nestedHotSwappables) { | ||
hotSwaps.push.apply(hotSwaps, _toConsumableArray(nestedHotSwappables.map(function (hotSwappable) { | ||
return [hotSwappable, undefined]; | ||
}))); | ||
} | ||
} | ||
} | ||
return [hasChanged ? normalizedNewValue : IGNORE, hotSwaps]; | ||
} | ||
if (_typeof(currentValue) === 'object' && currentValue != null) { | ||
return tryHotSwappingObjectValues(currentValue, newValue); | ||
if (_typeof(newValue) !== 'object' || newValue == null) { | ||
var _collectNestedHotSwap2; | ||
return [makeValueHotSwappable(newValue), (_collectNestedHotSwap2 = collectNestedHotSwappableValues(currentValue)) === null || _collectNestedHotSwap2 === void 0 ? void 0 : _collectNestedHotSwap2.map(function (hotSwappable) { | ||
return [hotSwappable, undefined]; | ||
})]; | ||
} | ||
var _hasChanged = false; | ||
var _hotSwaps = []; | ||
var _normalizedNewValue = {}; // eslint-disable-next-line guard-for-in | ||
for (var _key4 in currentValue) { | ||
var _currentElement = currentValue[_key4]; | ||
if (!(_key4 in newValue)) { | ||
_hasChanged = true; | ||
var _nestedHotSwappables = collectNestedHotSwappableValues(_currentElement); | ||
if (_nestedHotSwappables) { | ||
_hotSwaps.push.apply(_hotSwaps, _toConsumableArray(_nestedHotSwappables.map(function (hotSwappable) { | ||
return [hotSwappable, undefined]; | ||
}))); | ||
} | ||
} | ||
var _newElement = newValue[_key4]; | ||
var _tryHotSwappingValues5 = tryHotSwappingValues(_currentElement, _newElement), | ||
_tryHotSwappingValues6 = _slicedToArray(_tryHotSwappingValues5, 2), | ||
_updatedValue = _tryHotSwappingValues6[0], | ||
_elementHotSwaps = _tryHotSwappingValues6[1]; | ||
if (_elementHotSwaps) _hotSwaps.push.apply(_hotSwaps, _toConsumableArray(_elementHotSwaps)); | ||
if (_updatedValue === IGNORE) { | ||
_normalizedNewValue[_key4] = currentValue; | ||
} else { | ||
_hasChanged = true; | ||
_normalizedNewValue[_key4] = _updatedValue; | ||
} | ||
} | ||
for (var _key5 in newValue) { | ||
if (_key5 in _normalizedNewValue) continue; | ||
_hasChanged = true; | ||
_normalizedNewValue[_key5] = makeValueHotSwappable(newValue[_key5]); | ||
} | ||
return [_hasChanged ? _normalizedNewValue : IGNORE, _hotSwaps]; | ||
} | ||
@@ -475,9 +480,2 @@ | ||
return wrappedFunction; | ||
} else if (Array.isArray(value)) { | ||
return value.map(makeValueHotSwappable); | ||
} else if (_typeof(value) === 'object' && value != null) { | ||
return Object.keys(value).reduce(function (newValue, key) { | ||
newValue[key] = makeValueHotSwappable(value[key]); | ||
return newValue; | ||
}, {}); | ||
} | ||
@@ -624,3 +622,3 @@ | ||
type: value.type, | ||
props: value.remoteProps, | ||
props: value.props, | ||
children: value.children.map(function (child) { | ||
@@ -645,109 +643,2 @@ return serialize(child); | ||
}); | ||
} | ||
function tryHotSwappingObjectValues(currentValue, newValue) { | ||
if (_typeof(newValue) !== 'object' || newValue == null) { | ||
var _collectNestedHotSwap; | ||
return [makeValueHotSwappable(newValue), (_collectNestedHotSwap = collectNestedHotSwappableValues(currentValue)) === null || _collectNestedHotSwap === void 0 ? void 0 : _collectNestedHotSwap.map(function (hotSwappable) { | ||
return [hotSwappable, undefined]; | ||
})]; | ||
} | ||
var hasChanged = false; | ||
var hotSwaps = []; | ||
var normalizedNewValue = {}; // eslint-disable-next-line guard-for-in | ||
for (var _key4 in currentValue) { | ||
var currentObjectValue = currentValue[_key4]; | ||
if (!(_key4 in newValue)) { | ||
hasChanged = true; | ||
var nestedHotSwappables = collectNestedHotSwappableValues(currentObjectValue); | ||
if (nestedHotSwappables) { | ||
hotSwaps.push.apply(hotSwaps, _toConsumableArray(nestedHotSwappables.map(function (hotSwappable) { | ||
return [hotSwappable, undefined]; | ||
}))); | ||
} | ||
} | ||
var newObjectValue = newValue[_key4]; | ||
var _tryHotSwappingValues3 = tryHotSwappingValues(currentObjectValue, newObjectValue), | ||
_tryHotSwappingValues4 = _slicedToArray(_tryHotSwappingValues3, 2), | ||
updatedValue = _tryHotSwappingValues4[0], | ||
elementHotSwaps = _tryHotSwappingValues4[1]; | ||
if (elementHotSwaps) hotSwaps.push.apply(hotSwaps, _toConsumableArray(elementHotSwaps)); | ||
if (updatedValue !== IGNORE) { | ||
hasChanged = true; | ||
normalizedNewValue[_key4] = updatedValue; | ||
} | ||
} | ||
for (var _key5 in newValue) { | ||
if (_key5 in normalizedNewValue) continue; | ||
hasChanged = true; | ||
normalizedNewValue[_key5] = makeValueHotSwappable(newValue[_key5]); | ||
} | ||
return [hasChanged ? normalizedNewValue : IGNORE, hotSwaps]; | ||
} | ||
function tryHotSwappingArrayValues(currentValue, newValue) { | ||
if (!Array.isArray(newValue)) { | ||
var _collectNestedHotSwap2; | ||
return [makeValueHotSwappable(newValue), (_collectNestedHotSwap2 = collectNestedHotSwappableValues(currentValue)) === null || _collectNestedHotSwap2 === void 0 ? void 0 : _collectNestedHotSwap2.map(function (hotSwappable) { | ||
return [hotSwappable, undefined]; | ||
})]; | ||
} | ||
var hasChanged = false; | ||
var hotSwaps = []; | ||
var newLength = newValue.length; | ||
var currentLength = currentValue.length; | ||
var maxLength = Math.max(currentLength, newLength); | ||
var normalizedNewValue = []; | ||
for (var i = 0; i < maxLength; i++) { | ||
var currentArrayValue = currentValue[i]; | ||
var newArrayValue = newValue[i]; | ||
if (i < newLength) { | ||
if (i >= currentLength) { | ||
hasChanged = true; | ||
normalizedNewValue[i] = makeValueHotSwappable(newArrayValue); | ||
continue; | ||
} | ||
var _tryHotSwappingValues5 = tryHotSwappingValues(currentArrayValue, newArrayValue), | ||
_tryHotSwappingValues6 = _slicedToArray(_tryHotSwappingValues5, 2), | ||
updatedValue = _tryHotSwappingValues6[0], | ||
elementHotSwaps = _tryHotSwappingValues6[1]; | ||
if (elementHotSwaps) hotSwaps.push.apply(hotSwaps, _toConsumableArray(elementHotSwaps)); | ||
if (updatedValue === IGNORE) { | ||
normalizedNewValue[i] = currentArrayValue; | ||
continue; | ||
} | ||
hasChanged = true; | ||
normalizedNewValue[i] = updatedValue; | ||
} else { | ||
hasChanged = true; | ||
var nestedHotSwappables = collectNestedHotSwappableValues(currentArrayValue); | ||
if (nestedHotSwappables) { | ||
hotSwaps.push.apply(hotSwaps, _toConsumableArray(nestedHotSwappables.map(function (hotSwappable) { | ||
return [hotSwappable, undefined]; | ||
}))); | ||
} | ||
} | ||
} | ||
return [hasChanged ? normalizedNewValue : IGNORE, hotSwaps]; | ||
} |
@@ -5,28 +5,4 @@ "use strict"; | ||
var _receiver = require("../receiver"); | ||
function _createForOfIteratorHelper(o, allowArrayLike) { var it; if (typeof Symbol === "undefined" || o[Symbol.iterator] == null) { if (Array.isArray(o) || (it = _unsupportedIterableToArray(o)) || allowArrayLike && o && typeof o.length === "number") { if (it) o = it; var i = 0; var F = function F() {}; return { s: F, n: function n() { if (i >= o.length) return { done: true }; return { done: false, value: o[i++] }; }, e: function e(_e2) { throw _e2; }, f: F }; } throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } var normalCompletion = true, didErr = false, err; return { s: function s() { it = o[Symbol.iterator](); }, n: function n() { var step = it.next(); normalCompletion = step.done; return step; }, e: function e(_e3) { didErr = true; err = _e3; }, f: function f() { try { if (!normalCompletion && it["return"] != null) it["return"](); } finally { if (didErr) throw err; } } }; } | ||
function _toConsumableArray(arr) { return _arrayWithoutHoles(arr) || _iterableToArray(arr) || _unsupportedIterableToArray(arr) || _nonIterableSpread(); } | ||
function _nonIterableSpread() { throw new TypeError("Invalid attempt to spread non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } | ||
function _iterableToArray(iter) { if (typeof Symbol !== "undefined" && Symbol.iterator in Object(iter)) return Array.from(iter); } | ||
function _arrayWithoutHoles(arr) { if (Array.isArray(arr)) return _arrayLikeToArray(arr); } | ||
function _slicedToArray(arr, i) { return _arrayWithHoles(arr) || _iterableToArrayLimit(arr, i) || _unsupportedIterableToArray(arr, i) || _nonIterableRest(); } | ||
function _nonIterableRest() { throw new TypeError("Invalid attempt to destructure non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } | ||
function _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === "string") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === "Object" && o.constructor) n = o.constructor.name; if (n === "Map" || n === "Set") return Array.from(o); if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); } | ||
function _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len = arr.length; for (var i = 0, arr2 = new Array(len); i < len; i++) { arr2[i] = arr[i]; } return arr2; } | ||
function _iterableToArrayLimit(arr, i) { if (typeof Symbol === "undefined" || !(Symbol.iterator in Object(arr))) return; var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"] != null) _i["return"](); } finally { if (_d) throw _e; } } return _arr; } | ||
function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; } | ||
describe('root', function () { | ||
describe('createComponent()', function () { | ||
describe('createComponent', function () { | ||
it('does not throw error when no allowed components are set', function () { | ||
@@ -66,3 +42,3 @@ var root = (0, _root.createRemoteRoot)(function () {}); | ||
}); | ||
describe('createText()', function () { | ||
describe('createText', function () { | ||
it('does not throw error when appending a child created by the remote root', function () { | ||
@@ -78,3 +54,3 @@ var components = []; | ||
}); | ||
describe('appendChild()', function () { | ||
describe('appendChild', function () { | ||
it('does not throw error when appending a child created by the remote root', function () { | ||
@@ -95,3 +71,3 @@ var root = (0, _root.createRemoteRoot)(function () {}); | ||
}); | ||
describe('insertChildBefore()', function () { | ||
describe('insertChildBefore', function () { | ||
it('does not throw error when calling insertChildBefore for a component created by the remote root', function () { | ||
@@ -116,185 +92,2 @@ var root = (0, _root.createRemoteRoot)(function () {}); | ||
}); | ||
describe('hot-swapping', function () { | ||
it('hot-swaps function props', function () { | ||
var funcOne = jest.fn(); | ||
var funcTwo = jest.fn(); | ||
var receiver = createDelayedReceiver(); | ||
var root = (0, _root.createRemoteRoot)(receiver.receive); | ||
var button = root.createComponent('Button', { | ||
onPress: funcOne | ||
}); | ||
root.appendChild(button); | ||
root.mount(); // After this, the receiver will have the initial Button component | ||
receiver.flush(); | ||
button.updateProps({ | ||
onPress: funcTwo | ||
}); | ||
receiver.children[0].props.onPress(); | ||
expect(funcOne).not.toHaveBeenCalled(); | ||
expect(funcTwo).toHaveBeenCalled(); | ||
}); | ||
it('hot-swaps function props nested in objects', function () { | ||
var funcOne = jest.fn(); | ||
var funcTwo = jest.fn(); | ||
var receiver = createDelayedReceiver(); | ||
var root = (0, _root.createRemoteRoot)(receiver.receive); | ||
var resourceList = root.createComponent('ResourceList', { | ||
filterControl: { | ||
onQueryChange: funcOne, | ||
queryValue: 'foo' | ||
} | ||
}); | ||
root.appendChild(resourceList); | ||
root.mount(); | ||
resourceList.updateProps({ | ||
filterControl: { | ||
onQueryChange: funcTwo, | ||
queryValue: 'bar' | ||
} | ||
}); // After this, the receiver will have the initial ResourceList component | ||
receiver.flush(); | ||
var receivedResourceList = receiver.children[0]; | ||
var queryValue = receivedResourceList.props.filterControl.queryValue; | ||
receivedResourceList.props.filterControl.onQueryChange(); | ||
expect(queryValue).toStrictEqual('bar'); | ||
expect(funcOne).not.toHaveBeenCalled(); | ||
expect(funcTwo).toHaveBeenCalled(); | ||
}); | ||
it('hot-swaps function props nested in objects and arrays', function () { | ||
var funcOne = jest.fn(); | ||
var funcTwo = jest.fn(); | ||
var receiver = createDelayedReceiver(); | ||
var root = (0, _root.createRemoteRoot)(receiver.receive); | ||
var card = root.createComponent('Card', { | ||
actions: [{ | ||
onAction: funcOne | ||
}] | ||
}); | ||
root.appendChild(card); | ||
root.mount(); // After this, the receiver will have the initial Card component | ||
receiver.flush(); | ||
card.updateProps({ | ||
actions: [{ | ||
onAction: funcTwo | ||
}] | ||
}); | ||
receiver.children[0].props.actions[0].onAction(); | ||
expect(funcOne).not.toHaveBeenCalled(); | ||
expect(funcTwo).toHaveBeenCalled(); | ||
}); | ||
it('hot-swaps function props for arrays when the length increases', function () { | ||
var firstActionFuncOne = jest.fn(); | ||
var firstActionFuncTwo = jest.fn(); | ||
var secondActionFunc = jest.fn(); | ||
var receiver = createDelayedReceiver(); | ||
var root = (0, _root.createRemoteRoot)(receiver.receive); | ||
var modal = root.createComponent('Modal', { | ||
secondaryActions: [{ | ||
onAction: firstActionFuncOne | ||
}] | ||
}); | ||
root.appendChild(modal); | ||
root.mount(); | ||
modal.updateProps({ | ||
secondaryActions: [{ | ||
onAction: firstActionFuncTwo | ||
}, { | ||
onAction: secondActionFunc | ||
}] | ||
}); | ||
receiver.flush(); | ||
var _props$secondaryActio = _slicedToArray(receiver.children[0].props.secondaryActions, 2), | ||
firstAction = _props$secondaryActio[0], | ||
secondAction = _props$secondaryActio[1]; | ||
firstAction.onAction(); | ||
secondAction.onAction(); | ||
expect(firstActionFuncOne).not.toHaveBeenCalled(); | ||
expect(firstActionFuncTwo).toHaveBeenCalled(); | ||
expect(secondActionFunc).toHaveBeenCalled(); | ||
}); | ||
it('hot-swaps function props for nested arrays', function () { | ||
var firstActionFuncOne = jest.fn(); | ||
var firstActionFuncTwo = jest.fn(); | ||
var secondActionFuncOne = jest.fn(); | ||
var receiver = createDelayedReceiver(); | ||
var root = (0, _root.createRemoteRoot)(receiver.receive); | ||
var modal = root.createComponent('Modal', { | ||
actionGroups: [{ | ||
actions: [{ | ||
onAction: firstActionFuncOne | ||
}] | ||
}] | ||
}); | ||
root.appendChild(modal); | ||
root.mount(); | ||
modal.updateProps({ | ||
actionGroups: [{ | ||
actions: [{ | ||
onAction: firstActionFuncTwo | ||
}, { | ||
onAction: secondActionFuncOne | ||
}] | ||
}] | ||
}); | ||
receiver.flush(); | ||
var _props$actionGroups = _slicedToArray(receiver.children[0].props.actionGroups, 1), | ||
_props$actionGroups$ = _slicedToArray(_props$actionGroups[0].actions, 2), | ||
actionOne = _props$actionGroups$[0], | ||
actionTwo = _props$actionGroups$[1]; | ||
actionOne.onAction(); | ||
actionTwo.onAction(); | ||
expect(firstActionFuncOne).not.toHaveBeenCalled(); | ||
expect(firstActionFuncTwo).toHaveBeenCalled(); | ||
expect(secondActionFuncOne).toHaveBeenCalled(); | ||
}); | ||
}); | ||
}); | ||
function createDelayedReceiver() { | ||
var receiver = new _receiver.RemoteReceiver(); | ||
var enqueued = new Set(); | ||
return { | ||
get children() { | ||
return receiver.root.children; | ||
}, | ||
receive: function (type) { | ||
for (var _len = arguments.length, args = new Array(_len > 1 ? _len - 1 : 0), _key = 1; _key < _len; _key++) { | ||
args[_key - 1] = arguments[_key]; | ||
} | ||
var perform = function perform() { | ||
receiver.receive.apply(receiver, [type].concat(args)); | ||
enqueued["delete"](perform); | ||
}; | ||
enqueued.add(perform); | ||
}, | ||
flush: function flush() { | ||
var currentlyEnqueued = _toConsumableArray(enqueued); | ||
enqueued.clear(); | ||
var _iterator = _createForOfIteratorHelper(currentlyEnqueued), | ||
_step; | ||
try { | ||
for (_iterator.s(); !(_step = _iterator.n()).done;) { | ||
var perform = _step.value; | ||
perform(); | ||
} | ||
} catch (err) { | ||
_iterator.e(err); | ||
} finally { | ||
_iterator.f(); | ||
} | ||
} | ||
}; | ||
} | ||
}); |
@@ -40,6 +40,5 @@ "use strict"; | ||
const [initialProps, initialChildren, ...moreChildren] = rest; | ||
const normalizedInitialProps = initialProps !== null && initialProps !== void 0 ? initialProps : {}; | ||
const [initialProps, initialChildren] = rest; | ||
const normalizedInitialChildren = []; | ||
const normalizedInternalProps = {}; | ||
const normalizedInitialProps = {}; | ||
@@ -57,3 +56,3 @@ if (initialProps) { | ||
if (key === 'children') continue; | ||
normalizedInternalProps[key] = makeValueHotSwappable(initialProps[key]); | ||
normalizedInitialProps[key] = makeValueHotSwappable(initialProps[key]); | ||
} | ||
@@ -63,15 +62,4 @@ } | ||
if (initialChildren) { | ||
if (Array.isArray(initialChildren)) { | ||
for (const child of initialChildren) { | ||
normalizedInitialChildren.push(normalizeChild(child, remoteRoot)); | ||
} | ||
} else { | ||
normalizedInitialChildren.push(normalizeChild(initialChildren, remoteRoot)); // The complex tuple type of `rest` makes it so `moreChildren` is | ||
// incorrectly inferred as potentially being the props of the component, | ||
// lazy casting since we know it will be an array of child elements | ||
// (or empty). | ||
for (const child of moreChildren) { | ||
normalizedInitialChildren.push(normalizeChild(child, remoteRoot)); | ||
} | ||
for (const child of initialChildren) { | ||
normalizedInitialChildren.push(normalizeChild(child, remoteRoot)); | ||
} | ||
@@ -82,4 +70,3 @@ } | ||
const internals = { | ||
externalProps: strict ? Object.freeze(normalizedInitialProps) : normalizedInitialProps, | ||
internalProps: normalizedInternalProps, | ||
props: strict ? Object.freeze(normalizedInitialProps) : normalizedInitialProps, | ||
children: strict ? Object.freeze(normalizedInitialChildren) : normalizedInitialChildren | ||
@@ -95,9 +82,5 @@ }; | ||
get props() { | ||
return internals.externalProps; | ||
return internals.props; | ||
}, | ||
get remoteProps() { | ||
return internals.internalProps; | ||
}, | ||
updateProps: newProps => updateProps(component, newProps, internals, rootInternals), | ||
@@ -156,3 +139,2 @@ appendChild: child => appendChild(component, normalizeChild(child, remoteRoot), internals, rootInternals), | ||
mount() { | ||
if (rootInternals.mounted) return Promise.resolve(); | ||
rootInternals.mounted = true; | ||
@@ -230,3 +212,3 @@ return Promise.resolve(channel(_types.ACTION_MOUNT, rootInternals.children.map(serialize))); | ||
const { | ||
internalProps: currentProps | ||
props: currentProps | ||
} = internals; | ||
@@ -261,13 +243,10 @@ const normalizedNewProps = {}; | ||
if (hasRemoteChange) { | ||
channel(_types.ACTION_UPDATE_PROPS, component.id, normalizedNewProps); | ||
channel(_types.ACTION_UPDATE_PROPS, component.id, newProps); | ||
} | ||
}, | ||
local: () => { | ||
const mergedExternalProps = { ...internals.externalProps, | ||
const mergedProps = { ...internals.props, | ||
...newProps | ||
}; | ||
internals.externalProps = strict ? Object.freeze(mergedExternalProps) : mergedExternalProps; | ||
internals.internalProps = { ...internals.internalProps, | ||
...normalizedNewProps | ||
}; | ||
internals.props = strict ? Object.freeze(mergedProps) : mergedProps; | ||
@@ -279,80 +258,97 @@ for (const [hotSwappable, newValue] of hotSwapFunctions) { | ||
}); | ||
} // Imagine the following remote-ui components we might render in a remote context: | ||
// | ||
// const root = createRemoteRoot(); | ||
// const {value, onChange, onPress} = getPropsForValue(); | ||
// | ||
// const textField = root.createComponent('TextField', {value, onChange}); | ||
// const button = root.createComponent('Button', {onPress}); | ||
// | ||
// root.appendChild(textField); | ||
// root.appendChild(button); | ||
// | ||
// function getPropsForValue(value = '') { | ||
// return { | ||
// value, | ||
// onChange: () => { | ||
// const {value, onChange, onPress} = getPropsForValue(); | ||
// textField.updateProps({value, onChange}); | ||
// button.updateProps({onPress}); | ||
// }, | ||
// onPress: () => console.log(value), | ||
// }; | ||
// } | ||
// | ||
// | ||
// In this example, assume that the `TextField` `onChange` prop is run on blur. | ||
// If this were running on the host, the following steps would happen if you pressed | ||
// on the button: | ||
// | ||
// 1. The text field blurs, and so calls `onChange()` with its current value, which | ||
// then calls `setValue()` with the updated value. | ||
// 2. We synchronously update the `value`, `onChange`, and `onPress` props to point at | ||
// the most current `value`. | ||
// 3. Handling blur is finished, so the browser now handles the click by calling the | ||
// (newly-updated) `Button` `onPress()`, which logs out the new value. | ||
// | ||
// Because remote-ui reproduces a UI tree asynchronously from the remote context, the | ||
// steps above run in a different order: | ||
// | ||
// 1. The text field blurs, and so calls `onChange()` with its current value. | ||
// 2. Handling blur is finished **from the perspective of the main thread**, so the | ||
// browser now handles the click by calling the (original) `Button` `onPress()`, which | ||
// logs out the **initial** value. | ||
// 3. In the remote context, we receive the `onChange()` call, which calls updates the props | ||
// on the `Button` and `TextField` to be based on the new `value`, but by now it’s | ||
// already too late for `onPress` — the old version has already been called! | ||
// | ||
// As you can see, the timing issue introduced by the asynchronous nature of remote-ui | ||
// can cause “old props” to be called from the main thread. This example may seem like | ||
// an unusual pattern, and it is if you are using `@remote-ui/core` directly; you’d generally | ||
// keep a mutable reference to the state, instead of closing over the state with new props. | ||
// However, abstractions on top of `@remote-ui/core`, like the React reconciler in | ||
// `@remote-ui/react`, work almost entirely by closing over state, so this issue is | ||
// much more common with those declarative libraries. | ||
// | ||
// To protect against this, we handle function props a bit differently. When we have a | ||
// function prop, we replace it with a new function that calls the original. However, | ||
// we make the original mutable, by making it a property on the function itself. When | ||
// this function subsequently updates, we don’t send the update to the main thread (as | ||
// we just saw, this can often be "too late" to be of any use). Instead, we swap out | ||
// the mutable reference to the current implementation of the function prop, which can | ||
// be done synchronously. In the example above, this would all happen synchronously in | ||
// the remote context; in our handling of `TextField onChange()`, we update `Button onPress()`, | ||
// and swap out the implementations. Now, when the main thread attempts to call `Button onPress()`, | ||
// it instead calls our wrapper around the function, which can refer to, and call, the | ||
// most recently-applied implementation, instead of directly calling the old implementation. | ||
} | ||
function tryHotSwappingValues(currentValue, newValue) { | ||
if (typeof currentValue === 'function' && FUNCTION_CURRENT_IMPLEMENTATION_KEY in currentValue) { | ||
return [typeof newValue === 'function' ? IGNORE : makeValueHotSwappable(newValue), [[currentValue, newValue]]]; | ||
return [typeof newValue === 'function' ? IGNORE : makeValueHotSwappable(newValue), [currentValue, newValue]]; | ||
} | ||
if (Array.isArray(currentValue)) { | ||
return tryHotSwappingArrayValues(currentValue, newValue); | ||
if (!Array.isArray(newValue)) { | ||
var _collectNestedHotSwap; | ||
return [makeValueHotSwappable(newValue), (_collectNestedHotSwap = collectNestedHotSwappableValues(currentValue)) === null || _collectNestedHotSwap === void 0 ? void 0 : _collectNestedHotSwap.map(hotSwappable => [hotSwappable, undefined])]; | ||
} | ||
let hasChanged = false; | ||
const hotSwaps = []; | ||
const newLength = newValue.length; | ||
const currentLength = currentValue.length; | ||
const maxLength = Math.max(currentLength, newLength); | ||
const normalizedNewValue = []; | ||
for (let i = 0; i < maxLength; i++) { | ||
const currentElement = currentValue[i]; | ||
const newElement = newValue[i]; | ||
if (i < newLength) { | ||
if (i >= currentLength) { | ||
hasChanged = true; | ||
normalizedNewValue[i] = makeValueHotSwappable(newValue); | ||
} else { | ||
const [updatedValue, elementHotSwaps] = tryHotSwappingValues(currentElement, newElement); | ||
if (elementHotSwaps) hotSwaps.push(...elementHotSwaps); | ||
if (updatedValue === IGNORE) { | ||
normalizedNewValue[i] = currentValue; | ||
} else { | ||
hasChanged = true; | ||
normalizedNewValue[i] = updatedValue; | ||
} | ||
} | ||
} else { | ||
hasChanged = true; | ||
const nestedHotSwappables = collectNestedHotSwappableValues(currentElement); | ||
if (nestedHotSwappables) { | ||
hotSwaps.push(...nestedHotSwappables.map(hotSwappable => [hotSwappable, undefined])); | ||
} | ||
} | ||
} | ||
return [hasChanged ? normalizedNewValue : IGNORE, hotSwaps]; | ||
} | ||
if (typeof currentValue === 'object' && currentValue != null) { | ||
return tryHotSwappingObjectValues(currentValue, newValue); | ||
if (typeof newValue !== 'object' || newValue == null) { | ||
var _collectNestedHotSwap2; | ||
return [makeValueHotSwappable(newValue), (_collectNestedHotSwap2 = collectNestedHotSwappableValues(currentValue)) === null || _collectNestedHotSwap2 === void 0 ? void 0 : _collectNestedHotSwap2.map(hotSwappable => [hotSwappable, undefined])]; | ||
} | ||
let hasChanged = false; | ||
const hotSwaps = []; | ||
const normalizedNewValue = {}; // eslint-disable-next-line guard-for-in | ||
for (const key in currentValue) { | ||
const currentElement = currentValue[key]; | ||
if (!(key in newValue)) { | ||
hasChanged = true; | ||
const nestedHotSwappables = collectNestedHotSwappableValues(currentElement); | ||
if (nestedHotSwappables) { | ||
hotSwaps.push(...nestedHotSwappables.map(hotSwappable => [hotSwappable, undefined])); | ||
} | ||
} | ||
const newElement = newValue[key]; | ||
const [updatedValue, elementHotSwaps] = tryHotSwappingValues(currentElement, newElement); | ||
if (elementHotSwaps) hotSwaps.push(...elementHotSwaps); | ||
if (updatedValue === IGNORE) { | ||
normalizedNewValue[key] = currentValue; | ||
} else { | ||
hasChanged = true; | ||
normalizedNewValue[key] = updatedValue; | ||
} | ||
} | ||
for (const key in newValue) { | ||
if (key in normalizedNewValue) continue; | ||
hasChanged = true; | ||
normalizedNewValue[key] = makeValueHotSwappable(newValue[key]); | ||
} | ||
return [hasChanged ? normalizedNewValue : IGNORE, hotSwaps]; | ||
} | ||
@@ -376,9 +372,2 @@ | ||
return wrappedFunction; | ||
} else if (Array.isArray(value)) { | ||
return value.map(makeValueHotSwappable); | ||
} else if (typeof value === 'object' && value != null) { | ||
return Object.keys(value).reduce((newValue, key) => { | ||
newValue[key] = makeValueHotSwappable(value[key]); | ||
return newValue; | ||
}, {}); | ||
} | ||
@@ -522,3 +511,3 @@ | ||
type: value.type, | ||
props: value.remoteProps, | ||
props: value.props, | ||
children: value.children.map(child => serialize(child)) | ||
@@ -541,92 +530,2 @@ }; | ||
}); | ||
} | ||
function tryHotSwappingObjectValues(currentValue, newValue) { | ||
if (typeof newValue !== 'object' || newValue == null) { | ||
var _collectNestedHotSwap; | ||
return [makeValueHotSwappable(newValue), (_collectNestedHotSwap = collectNestedHotSwappableValues(currentValue)) === null || _collectNestedHotSwap === void 0 ? void 0 : _collectNestedHotSwap.map(hotSwappable => [hotSwappable, undefined])]; | ||
} | ||
let hasChanged = false; | ||
const hotSwaps = []; | ||
const normalizedNewValue = {}; // eslint-disable-next-line guard-for-in | ||
for (const key in currentValue) { | ||
const currentObjectValue = currentValue[key]; | ||
if (!(key in newValue)) { | ||
hasChanged = true; | ||
const nestedHotSwappables = collectNestedHotSwappableValues(currentObjectValue); | ||
if (nestedHotSwappables) { | ||
hotSwaps.push(...nestedHotSwappables.map(hotSwappable => [hotSwappable, undefined])); | ||
} | ||
} | ||
const newObjectValue = newValue[key]; | ||
const [updatedValue, elementHotSwaps] = tryHotSwappingValues(currentObjectValue, newObjectValue); | ||
if (elementHotSwaps) hotSwaps.push(...elementHotSwaps); | ||
if (updatedValue !== IGNORE) { | ||
hasChanged = true; | ||
normalizedNewValue[key] = updatedValue; | ||
} | ||
} | ||
for (const key in newValue) { | ||
if (key in normalizedNewValue) continue; | ||
hasChanged = true; | ||
normalizedNewValue[key] = makeValueHotSwappable(newValue[key]); | ||
} | ||
return [hasChanged ? normalizedNewValue : IGNORE, hotSwaps]; | ||
} | ||
function tryHotSwappingArrayValues(currentValue, newValue) { | ||
if (!Array.isArray(newValue)) { | ||
var _collectNestedHotSwap2; | ||
return [makeValueHotSwappable(newValue), (_collectNestedHotSwap2 = collectNestedHotSwappableValues(currentValue)) === null || _collectNestedHotSwap2 === void 0 ? void 0 : _collectNestedHotSwap2.map(hotSwappable => [hotSwappable, undefined])]; | ||
} | ||
let hasChanged = false; | ||
const hotSwaps = []; | ||
const newLength = newValue.length; | ||
const currentLength = currentValue.length; | ||
const maxLength = Math.max(currentLength, newLength); | ||
const normalizedNewValue = []; | ||
for (let i = 0; i < maxLength; i++) { | ||
const currentArrayValue = currentValue[i]; | ||
const newArrayValue = newValue[i]; | ||
if (i < newLength) { | ||
if (i >= currentLength) { | ||
hasChanged = true; | ||
normalizedNewValue[i] = makeValueHotSwappable(newArrayValue); | ||
continue; | ||
} | ||
const [updatedValue, elementHotSwaps] = tryHotSwappingValues(currentArrayValue, newArrayValue); | ||
if (elementHotSwaps) hotSwaps.push(...elementHotSwaps); | ||
if (updatedValue === IGNORE) { | ||
normalizedNewValue[i] = currentArrayValue; | ||
continue; | ||
} | ||
hasChanged = true; | ||
normalizedNewValue[i] = updatedValue; | ||
} else { | ||
hasChanged = true; | ||
const nestedHotSwappables = collectNestedHotSwappableValues(currentArrayValue); | ||
if (nestedHotSwappables) { | ||
hotSwaps.push(...nestedHotSwappables.map(hotSwappable => [hotSwappable, undefined])); | ||
} | ||
} | ||
} | ||
return [hasChanged ? normalizedNewValue : IGNORE, hotSwaps]; | ||
} |
@@ -5,6 +5,4 @@ "use strict"; | ||
var _receiver = require("../receiver"); | ||
describe('root', () => { | ||
describe('createComponent()', () => { | ||
describe('createComponent', () => { | ||
it('does not throw error when no allowed components are set', () => { | ||
@@ -44,3 +42,3 @@ const root = (0, _root.createRemoteRoot)(() => {}); | ||
}); | ||
describe('createText()', () => { | ||
describe('createText', () => { | ||
it('does not throw error when appending a child created by the remote root', () => { | ||
@@ -56,3 +54,3 @@ const components = []; | ||
}); | ||
describe('appendChild()', () => { | ||
describe('appendChild', () => { | ||
it('does not throw error when appending a child created by the remote root', () => { | ||
@@ -73,3 +71,3 @@ const root = (0, _root.createRemoteRoot)(() => {}); | ||
}); | ||
describe('insertChildBefore()', () => { | ||
describe('insertChildBefore', () => { | ||
it('does not throw error when calling insertChildBefore for a component created by the remote root', () => { | ||
@@ -94,169 +92,2 @@ const root = (0, _root.createRemoteRoot)(() => {}); | ||
}); | ||
describe('hot-swapping', () => { | ||
it('hot-swaps function props', () => { | ||
const funcOne = jest.fn(); | ||
const funcTwo = jest.fn(); | ||
const receiver = createDelayedReceiver(); | ||
const root = (0, _root.createRemoteRoot)(receiver.receive); | ||
const button = root.createComponent('Button', { | ||
onPress: funcOne | ||
}); | ||
root.appendChild(button); | ||
root.mount(); // After this, the receiver will have the initial Button component | ||
receiver.flush(); | ||
button.updateProps({ | ||
onPress: funcTwo | ||
}); | ||
receiver.children[0].props.onPress(); | ||
expect(funcOne).not.toHaveBeenCalled(); | ||
expect(funcTwo).toHaveBeenCalled(); | ||
}); | ||
it('hot-swaps function props nested in objects', () => { | ||
const funcOne = jest.fn(); | ||
const funcTwo = jest.fn(); | ||
const receiver = createDelayedReceiver(); | ||
const root = (0, _root.createRemoteRoot)(receiver.receive); | ||
const resourceList = root.createComponent('ResourceList', { | ||
filterControl: { | ||
onQueryChange: funcOne, | ||
queryValue: 'foo' | ||
} | ||
}); | ||
root.appendChild(resourceList); | ||
root.mount(); | ||
resourceList.updateProps({ | ||
filterControl: { | ||
onQueryChange: funcTwo, | ||
queryValue: 'bar' | ||
} | ||
}); // After this, the receiver will have the initial ResourceList component | ||
receiver.flush(); | ||
const receivedResourceList = receiver.children[0]; | ||
const queryValue = receivedResourceList.props.filterControl.queryValue; | ||
receivedResourceList.props.filterControl.onQueryChange(); | ||
expect(queryValue).toStrictEqual('bar'); | ||
expect(funcOne).not.toHaveBeenCalled(); | ||
expect(funcTwo).toHaveBeenCalled(); | ||
}); | ||
it('hot-swaps function props nested in objects and arrays', () => { | ||
const funcOne = jest.fn(); | ||
const funcTwo = jest.fn(); | ||
const receiver = createDelayedReceiver(); | ||
const root = (0, _root.createRemoteRoot)(receiver.receive); | ||
const card = root.createComponent('Card', { | ||
actions: [{ | ||
onAction: funcOne | ||
}] | ||
}); | ||
root.appendChild(card); | ||
root.mount(); // After this, the receiver will have the initial Card component | ||
receiver.flush(); | ||
card.updateProps({ | ||
actions: [{ | ||
onAction: funcTwo | ||
}] | ||
}); | ||
receiver.children[0].props.actions[0].onAction(); | ||
expect(funcOne).not.toHaveBeenCalled(); | ||
expect(funcTwo).toHaveBeenCalled(); | ||
}); | ||
it('hot-swaps function props for arrays when the length increases', () => { | ||
const firstActionFuncOne = jest.fn(); | ||
const firstActionFuncTwo = jest.fn(); | ||
const secondActionFunc = jest.fn(); | ||
const receiver = createDelayedReceiver(); | ||
const root = (0, _root.createRemoteRoot)(receiver.receive); | ||
const modal = root.createComponent('Modal', { | ||
secondaryActions: [{ | ||
onAction: firstActionFuncOne | ||
}] | ||
}); | ||
root.appendChild(modal); | ||
root.mount(); | ||
modal.updateProps({ | ||
secondaryActions: [{ | ||
onAction: firstActionFuncTwo | ||
}, { | ||
onAction: secondActionFunc | ||
}] | ||
}); | ||
receiver.flush(); | ||
const { | ||
secondaryActions: [firstAction, secondAction] | ||
} = receiver.children[0].props; | ||
firstAction.onAction(); | ||
secondAction.onAction(); | ||
expect(firstActionFuncOne).not.toHaveBeenCalled(); | ||
expect(firstActionFuncTwo).toHaveBeenCalled(); | ||
expect(secondActionFunc).toHaveBeenCalled(); | ||
}); | ||
it('hot-swaps function props for nested arrays', () => { | ||
const firstActionFuncOne = jest.fn(); | ||
const firstActionFuncTwo = jest.fn(); | ||
const secondActionFuncOne = jest.fn(); | ||
const receiver = createDelayedReceiver(); | ||
const root = (0, _root.createRemoteRoot)(receiver.receive); | ||
const modal = root.createComponent('Modal', { | ||
actionGroups: [{ | ||
actions: [{ | ||
onAction: firstActionFuncOne | ||
}] | ||
}] | ||
}); | ||
root.appendChild(modal); | ||
root.mount(); | ||
modal.updateProps({ | ||
actionGroups: [{ | ||
actions: [{ | ||
onAction: firstActionFuncTwo | ||
}, { | ||
onAction: secondActionFuncOne | ||
}] | ||
}] | ||
}); | ||
receiver.flush(); | ||
const { | ||
actionGroups: [{ | ||
actions: [actionOne, actionTwo] | ||
}] | ||
} = receiver.children[0].props; | ||
actionOne.onAction(); | ||
actionTwo.onAction(); | ||
expect(firstActionFuncOne).not.toHaveBeenCalled(); | ||
expect(firstActionFuncTwo).toHaveBeenCalled(); | ||
expect(secondActionFuncOne).toHaveBeenCalled(); | ||
}); | ||
}); | ||
}); | ||
function createDelayedReceiver() { | ||
const receiver = new _receiver.RemoteReceiver(); | ||
const enqueued = new Set(); | ||
return { | ||
get children() { | ||
return receiver.root.children; | ||
}, | ||
receive: (type, ...args) => { | ||
const perform = () => { | ||
receiver.receive(type, ...args); | ||
enqueued.delete(perform); | ||
}; | ||
enqueued.add(perform); | ||
}, | ||
flush() { | ||
const currentlyEnqueued = [...enqueued]; | ||
enqueued.clear(); | ||
for (const perform of currentlyEnqueued) { | ||
perform(); | ||
} | ||
} | ||
}; | ||
} | ||
}); |
@@ -28,2 +28,3 @@ import { ACTION_MOUNT, ACTION_INSERT_CHILD, ACTION_UPDATE_PROPS, ACTION_UPDATE_TEXT } from './types'; | ||
listen<T extends Attachable>({ id }: T, listener: UpdateListener<T>): () => void; | ||
flush(): Promise<void>; | ||
private enqueueUpdate; | ||
@@ -30,0 +31,0 @@ private attach; |
@@ -108,2 +108,6 @@ "use strict"; | ||
} | ||
flush() { | ||
var _a; | ||
return (_a = this.timeout) !== null && _a !== void 0 ? _a : Promise.resolve(); | ||
} | ||
enqueueUpdate(attached) { | ||
@@ -110,0 +114,0 @@ var _a; |
import { RemoteComponentType } from '@remote-ui/types'; | ||
import type { RemoteRoot, RemoteChannel } from './types'; | ||
export interface Options<AllowedComponents extends RemoteComponentType<string, any>> { | ||
readonly strict?: boolean; | ||
readonly components?: readonly AllowedComponents[]; | ||
} | ||
export declare function createRemoteRoot<AllowedComponents extends RemoteComponentType<string, any> = RemoteComponentType<any, any>, AllowedChildrenTypes extends AllowedComponents | boolean = true>(channel: RemoteChannel, { strict, components }?: Options<AllowedComponents>): RemoteRoot<AllowedComponents, AllowedChildrenTypes>; | ||
import type { RemoteRoot, RemoteChannel, RemoteRootOptions } from './types'; | ||
export declare function createRemoteRoot<AllowedComponents extends RemoteComponentType<string, any> = RemoteComponentType<any, any>, AllowedChildrenTypes extends AllowedComponents | boolean = true>(channel: RemoteChannel, { strict, components }?: RemoteRootOptions<AllowedComponents>): RemoteRoot<AllowedComponents, AllowedChildrenTypes>; | ||
//# sourceMappingURL=root.d.ts.map |
@@ -19,4 +19,9 @@ "use strict"; | ||
}; | ||
if (strict) | ||
Object.freeze(components); | ||
const remoteRoot = { | ||
kind: types_1.KIND_ROOT, | ||
options: strict | ||
? Object.freeze({ strict, components }) | ||
: { strict, components }, | ||
get children() { | ||
@@ -414,5 +419,6 @@ return rootInternals.children; | ||
return value.kind === types_1.KIND_TEXT | ||
? { id: value.id, text: value.text } | ||
? { id: value.id, kind: value.kind, text: value.text } | ||
: { | ||
id: value.id, | ||
kind: value.kind, | ||
type: value.type, | ||
@@ -419,0 +425,0 @@ props: value.remoteProps, |
@@ -33,5 +33,10 @@ import { RemoteComponentType, IdentifierForRemoteComponent, PropsForRemoteComponent } from '@remote-ui/types'; | ||
declare type AllowedTextChildren<Root extends RemoteRoot<any, any>, AllowString extends boolean = false> = AllowString extends true ? RemoteText<Root> | string : RemoteText<Root>; | ||
export interface RemoteRootOptions<AllowedComponents extends RemoteComponentType<string, any>> { | ||
readonly strict?: boolean; | ||
readonly components?: readonly AllowedComponents[]; | ||
} | ||
export interface RemoteRoot<AllowedComponents extends RemoteComponentType<string, any> = RemoteComponentType<any, any>, AllowedChildrenTypes extends RemoteComponentType<string, any> | boolean = true> { | ||
readonly kind: typeof KIND_ROOT; | ||
readonly children: readonly AllowedChildren<AllowedChildrenTypes, RemoteRoot<AllowedComponents, AllowedChildrenTypes>>[]; | ||
readonly options: RemoteRootOptions<AllowedComponents>; | ||
appendChild(child: AllowedChildren<AllowedChildrenTypes, RemoteRoot<AllowedComponents, AllowedChildrenTypes>, true>): void | Promise<void>; | ||
@@ -70,3 +75,3 @@ removeChild(child: AllowedChildren<AllowedChildrenTypes, RemoteRoot<AllowedComponents, AllowedChildrenTypes>>): void | Promise<void>; | ||
export declare type RemoteComponentSerialization<Type extends RemoteComponentType<string, any> = RemoteComponentType<string, any>> = { | ||
-readonly [K in 'id' | 'type' | 'props']: RemoteComponent<Type, any>[K]; | ||
-readonly [K in 'id' | 'type' | 'kind' | 'props']: RemoteComponent<Type, any>[K]; | ||
} & { | ||
@@ -76,3 +81,3 @@ children: (RemoteComponentSerialization | RemoteTextSerialization)[]; | ||
export declare type RemoteTextSerialization = { | ||
-readonly [K in 'id' | 'text']: RemoteText<any>[K]; | ||
-readonly [K in 'id' | 'text' | 'kind']: RemoteText<any>[K]; | ||
}; | ||
@@ -79,0 +84,0 @@ export declare type Serialized<T> = T extends RemoteComponent<infer Type, any> ? RemoteComponentSerialization<Type> : T extends RemoteText<any> ? RemoteTextSerialization : never; |
import type { RemoteComponentType } from '@remote-ui/types'; | ||
import type { RemoteRoot, RemoteComponent, RemoteText, RemoteChild } from './types'; | ||
export declare function isRemoteComponent<Type extends RemoteComponentType<string, any, any> = any, Root extends RemoteRoot<any, any> = RemoteRoot<any, any>>(child: RemoteChild<Root>): child is RemoteComponent<Type, Root>; | ||
export declare function isRemoteText<Root extends RemoteRoot<any, any> = RemoteRoot<any, any>>(child: RemoteChild<Root>): child is RemoteText<Root>; | ||
import type { RemoteRoot, RemoteComponent, RemoteText } from './types'; | ||
export declare function isRemoteComponent<Type extends RemoteComponentType<string, any, any> = any, Root extends RemoteRoot<any, any> = RemoteRoot<any, any>>(child: unknown): child is RemoteComponent<Type, Root>; | ||
export declare function isRemoteText<Root extends RemoteRoot<any, any> = RemoteRoot<any, any>>(child: unknown): child is RemoteText<Root>; | ||
//# sourceMappingURL=utilities.d.ts.map |
@@ -6,8 +6,8 @@ "use strict"; | ||
function isRemoteComponent(child) { | ||
return child.kind === types_1.KIND_COMPONENT; | ||
return child != null && child.kind === types_1.KIND_COMPONENT; | ||
} | ||
exports.isRemoteComponent = isRemoteComponent; | ||
function isRemoteText(child) { | ||
return child.kind === types_1.KIND_TEXT; | ||
return child != null && child.kind === types_1.KIND_TEXT; | ||
} | ||
exports.isRemoteText = isRemoteText; |
{ | ||
"name": "@remote-ui/core", | ||
"version": "1.5.4", | ||
"version": "1.6.0-alpha.0", | ||
"publishConfig": { | ||
@@ -23,6 +23,6 @@ "access": "public", | ||
"dependencies": { | ||
"@remote-ui/rpc": "^1.0.11", | ||
"@remote-ui/rpc": "^1.1.0-alpha.0", | ||
"@remote-ui/types": "^1.0.4" | ||
}, | ||
"gitHead": "74cd705c6d0644a915ae96a7c98f5b24856abf96" | ||
"gitHead": "95b058fdbc84958f3e10bb4dfd8029a6f0189078" | ||
} |
@@ -160,2 +160,6 @@ import {retain, release} from '@remote-ui/rpc'; | ||
flush() { | ||
return this.timeout ?? Promise.resolve(); | ||
} | ||
private enqueueUpdate(attached: Attachable) { | ||
@@ -162,0 +166,0 @@ this.timeout = |
@@ -18,11 +18,5 @@ import {RemoteComponentType} from '@remote-ui/types'; | ||
RemoteComponent, | ||
RemoteRootOptions, | ||
} from './types'; | ||
export interface Options< | ||
AllowedComponents extends RemoteComponentType<string, any> | ||
> { | ||
readonly strict?: boolean; | ||
readonly components?: readonly AllowedComponents[]; | ||
} | ||
type AnyChild = RemoteText<any> | RemoteComponent<any, any>; | ||
@@ -70,3 +64,3 @@ type AnyParent = RemoteRoot<any, any> | RemoteComponent<any, any>; | ||
channel: RemoteChannel, | ||
{strict = true, components}: Options<AllowedComponents> = {}, | ||
{strict = true, components}: RemoteRootOptions<AllowedComponents> = {}, | ||
): RemoteRoot<AllowedComponents, AllowedChildrenTypes> { | ||
@@ -87,4 +81,9 @@ type Root = RemoteRoot<AllowedComponents, AllowedChildrenTypes>; | ||
if (strict) Object.freeze(components); | ||
const remoteRoot: Root = { | ||
kind: KIND_ROOT, | ||
options: strict | ||
? Object.freeze({strict, components}) | ||
: {strict, components}, | ||
get children() { | ||
@@ -679,5 +678,6 @@ return rootInternals.children as any; | ||
return value.kind === KIND_TEXT | ||
? {id: value.id, text: value.text} | ||
? {id: value.id, kind: value.kind, text: value.text} | ||
: { | ||
id: value.id, | ||
kind: value.kind, | ||
type: value.type, | ||
@@ -684,0 +684,0 @@ props: value.remoteProps, |
@@ -83,2 +83,9 @@ import { | ||
export interface RemoteRootOptions< | ||
AllowedComponents extends RemoteComponentType<string, any> | ||
> { | ||
readonly strict?: boolean; | ||
readonly components?: readonly AllowedComponents[]; | ||
} | ||
export interface RemoteRoot< | ||
@@ -96,2 +103,3 @@ AllowedComponents extends RemoteComponentType< | ||
>[]; | ||
readonly options: RemoteRootOptions<AllowedComponents>; | ||
appendChild( | ||
@@ -212,3 +220,6 @@ child: AllowedChildren< | ||
> = { | ||
-readonly [K in 'id' | 'type' | 'props']: RemoteComponent<Type, any>[K]; | ||
-readonly [K in 'id' | 'type' | 'kind' | 'props']: RemoteComponent< | ||
Type, | ||
any | ||
>[K]; | ||
} & { | ||
@@ -219,3 +230,3 @@ children: (RemoteComponentSerialization | RemoteTextSerialization)[]; | ||
export type RemoteTextSerialization = { | ||
-readonly [K in 'id' | 'text']: RemoteText<any>[K]; | ||
-readonly [K in 'id' | 'text' | 'kind']: RemoteText<any>[K]; | ||
}; | ||
@@ -222,0 +233,0 @@ |
import type {RemoteComponentType} from '@remote-ui/types'; | ||
import type { | ||
RemoteRoot, | ||
RemoteComponent, | ||
RemoteText, | ||
RemoteChild, | ||
} from './types'; | ||
import type {RemoteRoot, RemoteComponent, RemoteText} from './types'; | ||
import {KIND_COMPONENT, KIND_TEXT} from './types'; | ||
@@ -13,4 +8,4 @@ | ||
Root extends RemoteRoot<any, any> = RemoteRoot<any, any> | ||
>(child: RemoteChild<Root>): child is RemoteComponent<Type, Root> { | ||
return child.kind === KIND_COMPONENT; | ||
>(child: unknown): child is RemoteComponent<Type, Root> { | ||
return child != null && (child as any).kind === KIND_COMPONENT; | ||
} | ||
@@ -20,4 +15,4 @@ | ||
Root extends RemoteRoot<any, any> = RemoteRoot<any, any> | ||
>(child: RemoteChild<Root>): child is RemoteText<Root> { | ||
return child.kind === KIND_TEXT; | ||
>(child: unknown): child is RemoteText<Root> { | ||
return child != null && (child as any).kind === KIND_TEXT; | ||
} |
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
No v1
QualityPackage is not semver >=1. This means it is not stable and does not support ^ ranges.
Found 1 instance in 1 package
74
548510
5241
2