From 3bea93883c97f81a9f834214ef16c9fff9e74472 Mon Sep 17 00:00:00 2001 From: Trina Choudhury Date: Mon, 12 Apr 2021 22:12:37 +0530 Subject: [PATCH 1/3] Exposed Load method to invoke synchronous loading of component --- src/__mocks__/@loadable/component.js | 1 + .../intersection-observer.test.js | 28 +++++++++++++++++++ .../intersection-observer.test.js | 13 +++++---- src/createLoadableVisibilityComponent.js | 12 ++++++-- src/loadable-components.js | 1 + src/react-loadable.js | 4 +++ 6 files changed, 51 insertions(+), 8 deletions(-) diff --git a/src/__mocks__/@loadable/component.js b/src/__mocks__/@loadable/component.js index c937c2f..0cc7188 100644 --- a/src/__mocks__/@loadable/component.js +++ b/src/__mocks__/@loadable/component.js @@ -8,6 +8,7 @@ const loadableObject = props => { }; loadableObject.preload = jest.fn(); +loadableObject.load = jest.fn(); function loadable(opts) { return loadableObject; diff --git a/src/__tests__/loadable-components/intersection-observer.test.js b/src/__tests__/loadable-components/intersection-observer.test.js index 8d065d4..3ef8502 100644 --- a/src/__tests__/loadable-components/intersection-observer.test.js +++ b/src/__tests__/loadable-components/intersection-observer.test.js @@ -126,6 +126,34 @@ describe("Loadable", () => { expect(queryByTestId("loaded-component")).toBeTruthy(); }); + test("load calls loadable load", () => { + // Mock @loadable/component to get a stable `load` function + jest.doMock("@loadable/component"); + + const loadable = require("@loadable/component"); + const loadableVisiblity = require("../../loadable-components"); // Require our tested module with the above mock applied + + loadableVisiblity(loader).load(); + + expect(loadable().load).toHaveBeenCalled(); + }); + + test("load will cause the loadable component to be displayed", async () => { + const Loader = loadableVisiblity(loader); + let returnedValue; + + const { queryByTestId } = render(); + expect(queryByTestId("loaded-component")).toBeNull(); + + act(() => { + returnedValue = Loader.load() + expect(returnedValue).toBeInstanceOf(Promise); + }); + returnedValue.then(() => { + expect(queryByTestId("loaded-component")).toBeTruthy(); + }) + }); + test("it displays the loadable component when it becomes visible", async () => { const Loader = loadableVisiblity(loader); diff --git a/src/__tests__/react-loadable/intersection-observer.test.js b/src/__tests__/react-loadable/intersection-observer.test.js index f14675b..4b48160 100644 --- a/src/__tests__/react-loadable/intersection-observer.test.js +++ b/src/__tests__/react-loadable/intersection-observer.test.js @@ -99,18 +99,19 @@ describe("Loadable", () => { test("preload will cause the loadable component to be displayed", async () => { const Loader = LoadableVisibility(opts); + let returnedValue; const { queryByTestId } = render(); expect(queryByTestId("loaded-component")).toBeNull(); act(() => { - Loader.preload(); + returnedValue = Loader.preload() + expect(returnedValue).toBeInstanceOf(Promise); }); - - await waitForElement(() => queryByTestId("loaded-component")); - - expect(queryByTestId("loaded-component")).toBeTruthy(); + returnedValue.then(() => { + expect(queryByTestId("loaded-component")).toBeTruthy(); + }) }); test("it displays the loadable component when it becomes visible", async () => { @@ -210,4 +211,4 @@ describe("Loadable.Map", () => { expect(Loadable.Map().preload).toHaveBeenCalled(); }); -}); +}) diff --git a/src/createLoadableVisibilityComponent.js b/src/createLoadableVisibilityComponent.js index c1e8fb7..192cb88 100644 --- a/src/createLoadableVisibilityComponent.js +++ b/src/createLoadableVisibilityComponent.js @@ -23,9 +23,9 @@ if (IntersectionObserver) { function createLoadableVisibilityComponent( args, - { Loadable, preloadFunc, LoadingComponent } + { Loadable, preloadFunc, loadFunc, LoadingComponent } ) { - let preloaded = false; + let preloaded = false, loaded = false; const visibilityHandlers = []; const LoadableComponent = Loadable(...args); @@ -108,6 +108,14 @@ function createLoadableVisibilityComponent( return LoadableComponent[preloadFunc](); }; + LoadableVisibilityComponent[loadFunc] = () => { + if (!loaded) { + loaded = true; + visibilityHandlers.forEach(handler => handler()); + } + return LoadableComponent[loadFunc](); + }; + return LoadableVisibilityComponent; } diff --git a/src/loadable-components.js b/src/loadable-components.js index a28d8d2..406acb1 100644 --- a/src/loadable-components.js +++ b/src/loadable-components.js @@ -8,6 +8,7 @@ function loadableVisiblity(load, opts = {}) { return createLoadableVisibilityComponent([load, opts], { Loadable: loadable, preloadFunc: "preload", + loadFunc: "load", LoadingComponent: opts.fallback ? () => opts.fallback : null }); } else { diff --git a/src/react-loadable.js b/src/react-loadable.js index 4f9a392..88059a6 100644 --- a/src/react-loadable.js +++ b/src/react-loadable.js @@ -8,6 +8,8 @@ function LoadableVisibility (opts) { return createLoadableVisibilityComponent([opts], { Loadable, preloadFunc: 'preload', + /* Preload works same as load function present in loadable/component */ + loadFunc: 'preload', LoadingComponent: opts.loading, }) } else { @@ -20,6 +22,8 @@ function LoadableVisibilityMap (opts) { return createLoadableVisibilityComponent([opts], { Loadable: Loadable.Map, preloadFunc: 'preload', + /* Preload works same as load function present in loadable/component */ + loadFunc: 'preload', LoadingComponent: opts.loading, }) } else { From d95d30c8372a313254c501363feb13d27369b3f4 Mon Sep 17 00:00:00 2001 From: TrinaChoudhury Date: Tue, 13 Apr 2021 16:01:35 +0530 Subject: [PATCH 2/3] Update react-loadable.js --- src/react-loadable.js | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/react-loadable.js b/src/react-loadable.js index 88059a6..77ac76a 100644 --- a/src/react-loadable.js +++ b/src/react-loadable.js @@ -8,7 +8,7 @@ function LoadableVisibility (opts) { return createLoadableVisibilityComponent([opts], { Loadable, preloadFunc: 'preload', - /* Preload works same as load function present in loadable/component */ + /* Preload helps in synchronously loading a component and returns a promise */ loadFunc: 'preload', LoadingComponent: opts.loading, }) @@ -22,7 +22,6 @@ function LoadableVisibilityMap (opts) { return createLoadableVisibilityComponent([opts], { Loadable: Loadable.Map, preloadFunc: 'preload', - /* Preload works same as load function present in loadable/component */ loadFunc: 'preload', LoadingComponent: opts.loading, }) From 1fb3e78aa4dd2bf4002e19bbc83700d6fd16ff4d Mon Sep 17 00:00:00 2001 From: Trina Choudhury Date: Mon, 12 Apr 2021 22:12:37 +0530 Subject: [PATCH 3/3] Exposed Load method to invoke synchronous loading of component --- capacities.js | 6 + createLoadableVisibilityComponent.js | 132 ++++++++++++++++++ index.js | 3 + loadable-components.js | 36 +++++ react-loadable.js | 48 +++++++ src/__mocks__/@loadable/component.js | 1 + .../intersection-observer.test.js | 28 ++++ .../intersection-observer.test.js | 13 +- src/createLoadableVisibilityComponent.js | 12 +- src/loadable-components.js | 1 + src/react-loadable.js | 4 + 11 files changed, 276 insertions(+), 8 deletions(-) create mode 100644 capacities.js create mode 100644 createLoadableVisibilityComponent.js create mode 100644 index.js create mode 100644 loadable-components.js create mode 100644 react-loadable.js diff --git a/capacities.js b/capacities.js new file mode 100644 index 0000000..01b9f3c --- /dev/null +++ b/capacities.js @@ -0,0 +1,6 @@ +"use strict"; + +exports.__esModule = true; +exports.IntersectionObserver = void 0; +var IntersectionObserver = typeof window !== 'undefined' && window.IntersectionObserver; +exports.IntersectionObserver = IntersectionObserver; \ No newline at end of file diff --git a/createLoadableVisibilityComponent.js b/createLoadableVisibilityComponent.js new file mode 100644 index 0000000..cfae611 --- /dev/null +++ b/createLoadableVisibilityComponent.js @@ -0,0 +1,132 @@ +"use strict"; + +exports.__esModule = true; +exports["default"] = void 0; + +var _react = _interopRequireWildcard(require("react")); + +var _capacities = require("./capacities"); + +function _getRequireWildcardCache() { if (typeof WeakMap !== "function") return null; var cache = new WeakMap(); _getRequireWildcardCache = function _getRequireWildcardCache() { return cache; }; return cache; } + +function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } if (obj === null || typeof obj !== "object" && typeof obj !== "function") { return { "default": obj }; } var cache = _getRequireWildcardCache(); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj["default"] = obj; if (cache) { cache.set(obj, newObj); } return newObj; } + +function _extends() { _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; return _extends.apply(this, arguments); } + +var intersectionObserver; +var trackedElements = new Map(); + +if (_capacities.IntersectionObserver) { + intersectionObserver = new window.IntersectionObserver(function (entries, observer) { + entries.forEach(function (entry) { + var visibilityHandler = trackedElements.get(entry.target); + + if (visibilityHandler && (entry.isIntersecting || entry.intersectionRatio > 0)) { + visibilityHandler(); + } + }); + }); +} + +function createLoadableVisibilityComponent(args, _ref) { + var Loadable = _ref.Loadable, + preloadFunc = _ref.preloadFunc, + loadFunc = _ref.loadFunc, + LoadingComponent = _ref.LoadingComponent; + var preloaded = false, + loaded = false; + var visibilityHandlers = []; + var LoadableComponent = Loadable.apply(void 0, args); + + function LoadableVisibilityComponent(props) { + var visibilityElementRef = (0, _react.useRef)(); + + var _useState = (0, _react.useState)(preloaded), + isVisible = _useState[0], + setVisible = _useState[1]; + + function visibilityHandler() { + if (visibilityElementRef.current) { + intersectionObserver.unobserve(visibilityElementRef.current); + trackedElements["delete"](visibilityElementRef.current); + } + + setVisible(true); + } + + (0, _react.useEffect)(function () { + var element = visibilityElementRef.current; + + if (!isVisible && element) { + visibilityHandlers.push(visibilityHandler); + trackedElements.set(element, visibilityHandler); + intersectionObserver.observe(element); + return function () { + var handlerIndex = visibilityHandlers.indexOf(visibilityHandler); + + if (handlerIndex >= 0) { + visibilityHandlers.splice(handlerIndex, 1); + } + + intersectionObserver.unobserve(element); + trackedElements["delete"](element); + }; + } + }, [isVisible, visibilityElementRef.current]); + + if (isVisible) { + return /*#__PURE__*/_react["default"].createElement(LoadableComponent, props); + } + + if (LoadingComponent || props.fallback) { + return /*#__PURE__*/_react["default"].createElement("div", _extends({ + style: { + display: "inline-block", + minHeight: "1px", + minWidth: "1px" + } + }, props, { + ref: visibilityElementRef + }), LoadingComponent ? /*#__PURE__*/_react["default"].createElement(LoadingComponent, _extends({ + isLoading: true + }, props)) : props.fallback); + } + + return /*#__PURE__*/_react["default"].createElement("div", _extends({ + style: { + display: "inline-block", + minHeight: "1px", + minWidth: "1px" + } + }, props, { + ref: visibilityElementRef + })); + } + + LoadableVisibilityComponent[preloadFunc] = function () { + if (!preloaded) { + preloaded = true; + visibilityHandlers.forEach(function (handler) { + return handler(); + }); + } + + return LoadableComponent[preloadFunc](); + }; + + LoadableVisibilityComponent[loadFunc] = function () { + if (!loaded) { + loaded = true; + visibilityHandlers.forEach(function (handler) { + return handler(); + }); + } + + return LoadableComponent[loadFunc](); + }; + + return LoadableVisibilityComponent; +} + +var _default = createLoadableVisibilityComponent; +exports["default"] = _default; \ No newline at end of file diff --git a/index.js b/index.js new file mode 100644 index 0000000..c9c4d31 --- /dev/null +++ b/index.js @@ -0,0 +1,3 @@ +"use strict"; + +module.exports = require("./loadable-components"); \ No newline at end of file diff --git a/loadable-components.js b/loadable-components.js new file mode 100644 index 0000000..a6fa7c6 --- /dev/null +++ b/loadable-components.js @@ -0,0 +1,36 @@ +"use strict"; + +var _react = _interopRequireWildcard(require("react")); + +var _component = _interopRequireDefault(require("@loadable/component")); + +var _createLoadableVisibilityComponent = _interopRequireDefault(require("./createLoadableVisibilityComponent")); + +var _capacities = require("./capacities"); + +function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; } + +function _getRequireWildcardCache() { if (typeof WeakMap !== "function") return null; var cache = new WeakMap(); _getRequireWildcardCache = function _getRequireWildcardCache() { return cache; }; return cache; } + +function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } if (obj === null || typeof obj !== "object" && typeof obj !== "function") { return { "default": obj }; } var cache = _getRequireWildcardCache(); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj["default"] = obj; if (cache) { cache.set(obj, newObj); } return newObj; } + +function loadableVisiblity(load, opts) { + if (opts === void 0) { + opts = {}; + } + + if (_capacities.IntersectionObserver) { + return (0, _createLoadableVisibilityComponent["default"])([load, opts], { + Loadable: _component["default"], + preloadFunc: "preload", + loadFunc: "load", + LoadingComponent: opts.fallback ? function () { + return opts.fallback; + } : null + }); + } else { + return (0, _component["default"])(load, opts); + } +} + +module.exports = loadableVisiblity; \ No newline at end of file diff --git a/react-loadable.js b/react-loadable.js new file mode 100644 index 0000000..750fff4 --- /dev/null +++ b/react-loadable.js @@ -0,0 +1,48 @@ +"use strict"; + +var _react = _interopRequireWildcard(require("react")); + +var _reactLoadable = _interopRequireDefault(require("react-loadable")); + +var _createLoadableVisibilityComponent = _interopRequireDefault(require("./createLoadableVisibilityComponent")); + +var _capacities = require("./capacities"); + +function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; } + +function _getRequireWildcardCache() { if (typeof WeakMap !== "function") return null; var cache = new WeakMap(); _getRequireWildcardCache = function _getRequireWildcardCache() { return cache; }; return cache; } + +function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } if (obj === null || typeof obj !== "object" && typeof obj !== "function") { return { "default": obj }; } var cache = _getRequireWildcardCache(); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj["default"] = obj; if (cache) { cache.set(obj, newObj); } return newObj; } + +function LoadableVisibility(opts) { + if (_capacities.IntersectionObserver) { + return (0, _createLoadableVisibilityComponent["default"])([opts], { + Loadable: _reactLoadable["default"], + preloadFunc: 'preload', + + /* Preload works same as load function present in loadable/component */ + loadFunc: 'preload', + LoadingComponent: opts.loading + }); + } else { + return (0, _reactLoadable["default"])(opts); + } +} + +function LoadableVisibilityMap(opts) { + if (_capacities.IntersectionObserver) { + return (0, _createLoadableVisibilityComponent["default"])([opts], { + Loadable: _reactLoadable["default"].Map, + preloadFunc: 'preload', + + /* Preload works same as load function present in loadable/component */ + loadFunc: 'preload', + LoadingComponent: opts.loading + }); + } else { + return _reactLoadable["default"].Map(opts); + } +} + +LoadableVisibility.Map = LoadableVisibilityMap; +module.exports = LoadableVisibility; \ No newline at end of file diff --git a/src/__mocks__/@loadable/component.js b/src/__mocks__/@loadable/component.js index c937c2f..0cc7188 100644 --- a/src/__mocks__/@loadable/component.js +++ b/src/__mocks__/@loadable/component.js @@ -8,6 +8,7 @@ const loadableObject = props => { }; loadableObject.preload = jest.fn(); +loadableObject.load = jest.fn(); function loadable(opts) { return loadableObject; diff --git a/src/__tests__/loadable-components/intersection-observer.test.js b/src/__tests__/loadable-components/intersection-observer.test.js index 8d065d4..3ef8502 100644 --- a/src/__tests__/loadable-components/intersection-observer.test.js +++ b/src/__tests__/loadable-components/intersection-observer.test.js @@ -126,6 +126,34 @@ describe("Loadable", () => { expect(queryByTestId("loaded-component")).toBeTruthy(); }); + test("load calls loadable load", () => { + // Mock @loadable/component to get a stable `load` function + jest.doMock("@loadable/component"); + + const loadable = require("@loadable/component"); + const loadableVisiblity = require("../../loadable-components"); // Require our tested module with the above mock applied + + loadableVisiblity(loader).load(); + + expect(loadable().load).toHaveBeenCalled(); + }); + + test("load will cause the loadable component to be displayed", async () => { + const Loader = loadableVisiblity(loader); + let returnedValue; + + const { queryByTestId } = render(); + expect(queryByTestId("loaded-component")).toBeNull(); + + act(() => { + returnedValue = Loader.load() + expect(returnedValue).toBeInstanceOf(Promise); + }); + returnedValue.then(() => { + expect(queryByTestId("loaded-component")).toBeTruthy(); + }) + }); + test("it displays the loadable component when it becomes visible", async () => { const Loader = loadableVisiblity(loader); diff --git a/src/__tests__/react-loadable/intersection-observer.test.js b/src/__tests__/react-loadable/intersection-observer.test.js index f14675b..4b48160 100644 --- a/src/__tests__/react-loadable/intersection-observer.test.js +++ b/src/__tests__/react-loadable/intersection-observer.test.js @@ -99,18 +99,19 @@ describe("Loadable", () => { test("preload will cause the loadable component to be displayed", async () => { const Loader = LoadableVisibility(opts); + let returnedValue; const { queryByTestId } = render(); expect(queryByTestId("loaded-component")).toBeNull(); act(() => { - Loader.preload(); + returnedValue = Loader.preload() + expect(returnedValue).toBeInstanceOf(Promise); }); - - await waitForElement(() => queryByTestId("loaded-component")); - - expect(queryByTestId("loaded-component")).toBeTruthy(); + returnedValue.then(() => { + expect(queryByTestId("loaded-component")).toBeTruthy(); + }) }); test("it displays the loadable component when it becomes visible", async () => { @@ -210,4 +211,4 @@ describe("Loadable.Map", () => { expect(Loadable.Map().preload).toHaveBeenCalled(); }); -}); +}) diff --git a/src/createLoadableVisibilityComponent.js b/src/createLoadableVisibilityComponent.js index c1e8fb7..192cb88 100644 --- a/src/createLoadableVisibilityComponent.js +++ b/src/createLoadableVisibilityComponent.js @@ -23,9 +23,9 @@ if (IntersectionObserver) { function createLoadableVisibilityComponent( args, - { Loadable, preloadFunc, LoadingComponent } + { Loadable, preloadFunc, loadFunc, LoadingComponent } ) { - let preloaded = false; + let preloaded = false, loaded = false; const visibilityHandlers = []; const LoadableComponent = Loadable(...args); @@ -108,6 +108,14 @@ function createLoadableVisibilityComponent( return LoadableComponent[preloadFunc](); }; + LoadableVisibilityComponent[loadFunc] = () => { + if (!loaded) { + loaded = true; + visibilityHandlers.forEach(handler => handler()); + } + return LoadableComponent[loadFunc](); + }; + return LoadableVisibilityComponent; } diff --git a/src/loadable-components.js b/src/loadable-components.js index a28d8d2..406acb1 100644 --- a/src/loadable-components.js +++ b/src/loadable-components.js @@ -8,6 +8,7 @@ function loadableVisiblity(load, opts = {}) { return createLoadableVisibilityComponent([load, opts], { Loadable: loadable, preloadFunc: "preload", + loadFunc: "load", LoadingComponent: opts.fallback ? () => opts.fallback : null }); } else { diff --git a/src/react-loadable.js b/src/react-loadable.js index 4f9a392..88059a6 100644 --- a/src/react-loadable.js +++ b/src/react-loadable.js @@ -8,6 +8,8 @@ function LoadableVisibility (opts) { return createLoadableVisibilityComponent([opts], { Loadable, preloadFunc: 'preload', + /* Preload works same as load function present in loadable/component */ + loadFunc: 'preload', LoadingComponent: opts.loading, }) } else { @@ -20,6 +22,8 @@ function LoadableVisibilityMap (opts) { return createLoadableVisibilityComponent([opts], { Loadable: Loadable.Map, preloadFunc: 'preload', + /* Preload works same as load function present in loadable/component */ + loadFunc: 'preload', LoadingComponent: opts.loading, }) } else {