@thoughtbot/superglue 0.0.0 → 0.30.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,288 @@
1
+ "use strict";
2
+
3
+ exports.__esModule = true;
4
+ exports.addFlash = addFlash;
5
+ exports.appendReceivedFragmentsOntoPage = appendReceivedFragmentsOntoPage;
6
+ exports.graftNodeOntoPage = graftNodeOntoPage;
7
+ exports.handleGraft = handleGraft;
8
+ exports.metaReducer = metaReducer;
9
+ exports.pageReducer = pageReducer;
10
+ exports.rootReducer = void 0;
11
+ exports.superglueReducer = superglueReducer;
12
+
13
+ var _utils = require("../utils");
14
+
15
+ var _actions = require("../actions");
16
+
17
+ var _config = require("../config");
18
+
19
+ function addPlaceholdersToDeferredNodes(existingPage, page) {
20
+ var _existingPage$defers = existingPage.defers,
21
+ defers = _existingPage$defers === void 0 ? [] : _existingPage$defers;
22
+ var prevDefers = defers.map(function (_ref) {
23
+ var path = _ref.path;
24
+ var node = (0, _utils.getIn)(existingPage, path);
25
+ var copy = JSON.stringify(node);
26
+ return [path, JSON.parse(copy)];
27
+ });
28
+ return prevDefers.reduce(function (memo, _ref2) {
29
+ var path = _ref2[0],
30
+ node = _ref2[1];
31
+ return (0, _utils.setIn)(page, path, node);
32
+ }, page);
33
+ }
34
+
35
+ function constrainPagesSize(state) {
36
+ var maxPages = _config.config.maxPages;
37
+ var allPageKeys = Object.keys(state);
38
+ var cacheTimesRecentFirst = allPageKeys.map(function (key) {
39
+ return state[key].savedAt;
40
+ }).sort(function (a, b) {
41
+ return b - a;
42
+ });
43
+
44
+ for (var _i = 0, _Array$from = Array.from(allPageKeys); _i < _Array$from.length; _i++) {
45
+ var key = _Array$from[_i];
46
+
47
+ if (state[key].savedAt <= cacheTimesRecentFirst[maxPages - 1]) {
48
+ delete state[key];
49
+ }
50
+ }
51
+ }
52
+
53
+ function saveResponse(state, pageKey, page) {
54
+ state = Object.assign({}, state);
55
+ page = Object.assign({
56
+ pageKey: pageKey,
57
+ fragments: []
58
+ }, page, {
59
+ savedAt: Date.now()
60
+ });
61
+ var existingPage = state[pageKey];
62
+
63
+ if (existingPage) {
64
+ page = addPlaceholdersToDeferredNodes(existingPage, page);
65
+ }
66
+
67
+ constrainPagesSize(state);
68
+ state[pageKey] = page;
69
+ return state;
70
+ }
71
+
72
+ function appendReceivedFragmentsOntoPage(state, pageKey, receivedFragments) {
73
+ if (!pageKey) {
74
+ return state;
75
+ }
76
+
77
+ if (receivedFragments.length === 0) {
78
+ return state;
79
+ }
80
+
81
+ var currentPage = state[pageKey];
82
+ var _currentPage$fragment = currentPage.fragments,
83
+ prevFragments = _currentPage$fragment === void 0 ? [] : _currentPage$fragment;
84
+ var nextFragments = [].concat(prevFragments);
85
+ var existingKeys = {};
86
+ prevFragments.forEach(function (frag) {
87
+ return existingKeys[frag.path] = true;
88
+ });
89
+ receivedFragments.forEach(function (frag) {
90
+ if (!existingKeys[frag.path]) {
91
+ nextFragments.push(frag);
92
+ }
93
+ });
94
+ var nextPage = Object.assign({}, currentPage, {
95
+ fragments: nextFragments
96
+ });
97
+ var nextState = Object.assign({}, state);
98
+ nextState[pageKey] = nextPage;
99
+ return nextState;
100
+ }
101
+
102
+ function addFlash(state, pageKey, receivedFlash) {
103
+ var nextState = Object.assign({}, state);
104
+ var nextPage = Object.assign({}, state[pageKey]);
105
+ nextPage.flash = Object.assign({}, nextPage.flash, receivedFlash);
106
+ nextState[pageKey] = nextPage;
107
+ return nextState;
108
+ }
109
+
110
+ function graftNodeOntoPage(state, pageKey, node, pathToNode) {
111
+ if (!node) {
112
+ console.warn('There was no node returned in the response. Do you have the correct key path in your props_at?');
113
+ return state;
114
+ }
115
+
116
+ if (!pathToNode || !pageKey) {
117
+ return state;
118
+ }
119
+
120
+ var fullPathToNode = [pageKey, pathToNode].join('.');
121
+ return (0, _utils.setIn)(state, fullPathToNode, node);
122
+ }
123
+
124
+ function handleGraft(state, pageKey, page) {
125
+ var currentPage = state[pageKey];
126
+
127
+ if (!currentPage) {
128
+ var error = new Error("Superglue was looking for " + pageKey + " in your state, but could not find it in your mapping. Did you forget to pass in a valid pageKey to this.props.remote or this.props.visit?");
129
+ throw error;
130
+ }
131
+
132
+ var receivedNode = page.data,
133
+ pathToNode = page.path,
134
+ _page$fragments = page.fragments,
135
+ receivedFragments = _page$fragments === void 0 ? [] : _page$fragments,
136
+ receivedFlash = page.flash;
137
+ return [function (nextState) {
138
+ return graftNodeOntoPage(nextState, pageKey, receivedNode, pathToNode);
139
+ }, function (nextState) {
140
+ return appendReceivedFragmentsOntoPage(nextState, pageKey, receivedFragments);
141
+ }, function (nextState) {
142
+ return addFlash(nextState, pageKey, receivedFlash);
143
+ }].reduce(function (memo, fn) {
144
+ return fn(memo);
145
+ }, state);
146
+ }
147
+
148
+ function pageReducer(state, action) {
149
+ if (state === void 0) {
150
+ state = {};
151
+ }
152
+
153
+ switch (action.type) {
154
+ case _actions.SAVE_RESPONSE:
155
+ {
156
+ var _action$payload = action.payload,
157
+ pageKey = _action$payload.pageKey,
158
+ page = _action$payload.page;
159
+ return saveResponse(state, pageKey, page);
160
+ }
161
+
162
+ case _actions.HANDLE_GRAFT:
163
+ {
164
+ var _action$payload2 = action.payload,
165
+ _pageKey = _action$payload2.pageKey,
166
+ _page = _action$payload2.page;
167
+ return handleGraft(state, _pageKey, _page);
168
+ }
169
+
170
+ case _actions.UPDATE_FRAGMENTS:
171
+ {
172
+ var changedFragments = action.payload.changedFragments;
173
+ var nextState = state;
174
+ Object.entries(state).forEach(function (_ref3) {
175
+ var pageKey = _ref3[0],
176
+ page = _ref3[1];
177
+ page.fragments.forEach(function (fragment) {
178
+ var type = fragment.type,
179
+ path = fragment.path;
180
+ var changedNode = changedFragments[type];
181
+ var currentNode = (0, _utils.getIn)(nextState, pageKey + "." + path);
182
+
183
+ if (type in changedFragments && changedNode !== currentNode) {
184
+ var nextNode = JSON.parse(JSON.stringify(changedNode));
185
+ nextState = (0, _utils.setIn)(nextState, pageKey + "." + path, nextNode);
186
+ }
187
+ });
188
+ });
189
+ return nextState;
190
+ }
191
+
192
+ case _actions.CLEAR_FLASH:
193
+ {
194
+ var _pageKey2 = action.payload.pageKey;
195
+
196
+ var _nextState = Object.assign({}, state);
197
+
198
+ var nextPage = Object.assign({}, state[_pageKey2]);
199
+ nextPage.flash = {};
200
+ _nextState[_pageKey2] = nextPage;
201
+ return _nextState;
202
+ }
203
+
204
+ case _actions.COPY_PAGE:
205
+ {
206
+ var _nextState2 = Object.assign({}, state);
207
+
208
+ var _action$payload3 = action.payload,
209
+ from = _action$payload3.from,
210
+ to = _action$payload3.to;
211
+ _nextState2[(0, _utils.urlToPageKey)(to)] = JSON.parse(JSON.stringify(_nextState2[from]));
212
+ return _nextState2;
213
+ }
214
+
215
+ case _actions.REMOVE_PAGE:
216
+ {
217
+ var _pageKey3 = action.payload.pageKey;
218
+
219
+ var _nextState3 = Object.assign({}, state);
220
+
221
+ delete _nextState3[_pageKey3];
222
+ return _nextState3;
223
+ }
224
+
225
+ default:
226
+ return state;
227
+ }
228
+ }
229
+
230
+ function metaReducer(state, action) {
231
+ if (state === void 0) {
232
+ state = {};
233
+ }
234
+
235
+ switch (action.type) {
236
+ case _actions.HISTORY_CHANGE:
237
+ {
238
+ var _action$payload4 = action.payload,
239
+ pathname = _action$payload4.pathname,
240
+ search = _action$payload4.search,
241
+ hash = _action$payload4.hash;
242
+ var currentPageKey = (0, _utils.urlToPageKey)(pathname + search);
243
+ return Object.assign({}, state, {
244
+ currentPageKey: currentPageKey,
245
+ pathname: pathname,
246
+ search: search,
247
+ hash: hash
248
+ });
249
+ }
250
+
251
+ case _actions.SAVE_RESPONSE:
252
+ {
253
+ var _action$payload$page = action.payload.page,
254
+ csrfToken = _action$payload$page.csrfToken,
255
+ assets = _action$payload$page.assets;
256
+ return Object.assign({}, state, {
257
+ csrfToken: csrfToken,
258
+ assets: assets
259
+ });
260
+ }
261
+
262
+ case _actions.SET_CSRF_TOKEN:
263
+ {
264
+ var _csrfToken = action.payload.csrfToken;
265
+ return Object.assign({}, state, {
266
+ csrfToken: _csrfToken
267
+ });
268
+ }
269
+
270
+ default:
271
+ return state;
272
+ }
273
+ }
274
+
275
+ function superglueReducer(state, action) {
276
+ if (state === void 0) {
277
+ state = {};
278
+ }
279
+
280
+ var meta = metaReducer(state, action);
281
+ return Object.assign({}, meta);
282
+ }
283
+
284
+ var rootReducer = {
285
+ superglue: superglueReducer,
286
+ pages: pageReducer
287
+ };
288
+ exports.rootReducer = rootReducer;
@@ -0,0 +1,36 @@
1
+ "use strict";
2
+
3
+ exports.__esModule = true;
4
+ exports.argsForHistory = argsForHistory;
5
+ exports.extractNodeAndPath = extractNodeAndPath;
6
+ exports.isGraft = isGraft;
7
+
8
+ var _url = require("./url");
9
+
10
+ function isGraft(page) {
11
+ return page.action === 'graft';
12
+ }
13
+
14
+ function extractNodeAndPath(page) {
15
+ var node = page.data,
16
+ action = page.action,
17
+ pathToNode = page.path;
18
+
19
+ if (action === 'graft') {
20
+ return {
21
+ node: node,
22
+ pathToNode: pathToNode
23
+ };
24
+ } else {
25
+ var errMsg = 'Expected page to be a graft response rendered from node filtering.';
26
+ throw new Error(errMsg);
27
+ }
28
+ }
29
+
30
+ function argsForHistory(path) {
31
+ var pageKey = (0, _url.urlToPageKey)(path);
32
+ return [path, {
33
+ superglue: true,
34
+ pageKey: pageKey
35
+ }];
36
+ }
@@ -0,0 +1,176 @@
1
+ "use strict";
2
+
3
+ exports.__esModule = true;
4
+ exports.KeyPathError = void 0;
5
+ exports.getIn = getIn;
6
+ exports.setIn = setIn;
7
+
8
+ function _inheritsLoose(subClass, superClass) { subClass.prototype = Object.create(superClass.prototype); subClass.prototype.constructor = subClass; _setPrototypeOf(subClass, superClass); }
9
+
10
+ function _wrapNativeSuper(Class) { var _cache = typeof Map === "function" ? new Map() : undefined; _wrapNativeSuper = function _wrapNativeSuper(Class) { if (Class === null || !_isNativeFunction(Class)) return Class; if (typeof Class !== "function") { throw new TypeError("Super expression must either be null or a function"); } if (typeof _cache !== "undefined") { if (_cache.has(Class)) return _cache.get(Class); _cache.set(Class, Wrapper); } function Wrapper() { return _construct(Class, arguments, _getPrototypeOf(this).constructor); } Wrapper.prototype = Object.create(Class.prototype, { constructor: { value: Wrapper, enumerable: false, writable: true, configurable: true } }); return _setPrototypeOf(Wrapper, Class); }; return _wrapNativeSuper(Class); }
11
+
12
+ function _construct(Parent, args, Class) { if (_isNativeReflectConstruct()) { _construct = Reflect.construct; } else { _construct = function _construct(Parent, args, Class) { var a = [null]; a.push.apply(a, args); var Constructor = Function.bind.apply(Parent, a); var instance = new Constructor(); if (Class) _setPrototypeOf(instance, Class.prototype); return instance; }; } return _construct.apply(null, arguments); }
13
+
14
+ function _isNativeReflectConstruct() { if (typeof Reflect === "undefined" || !Reflect.construct) return false; if (Reflect.construct.sham) return false; if (typeof Proxy === "function") return true; try { Boolean.prototype.valueOf.call(Reflect.construct(Boolean, [], function () {})); return true; } catch (e) { return false; } }
15
+
16
+ function _isNativeFunction(fn) { return Function.toString.call(fn).indexOf("[native code]") !== -1; }
17
+
18
+ function _setPrototypeOf(o, p) { _setPrototypeOf = Object.setPrototypeOf || function _setPrototypeOf(o, p) { o.__proto__ = p; return o; }; return _setPrototypeOf(o, p); }
19
+
20
+ function _getPrototypeOf(o) { _getPrototypeOf = Object.setPrototypeOf ? Object.getPrototypeOf : function _getPrototypeOf(o) { return o.__proto__ || Object.getPrototypeOf(o); }; return _getPrototypeOf(o); }
21
+
22
+ // These were taken from Scour.js
23
+ // Then, modified to respect the id=0 keypath
24
+ var isSearchable = /^[\da-zA-Z\-_=.]+$/;
25
+
26
+ var KeyPathError = /*#__PURE__*/function (_Error) {
27
+ _inheritsLoose(KeyPathError, _Error);
28
+
29
+ function KeyPathError(message) {
30
+ var _this;
31
+
32
+ _this = _Error.call(this, message) || this;
33
+ _this.name = 'KeyPathError';
34
+ return _this;
35
+ }
36
+
37
+ return KeyPathError;
38
+ }( /*#__PURE__*/_wrapNativeSuper(Error));
39
+
40
+ exports.KeyPathError = KeyPathError;
41
+
42
+ function getIn(obj, path) {
43
+ var keyPath = normalizeKeyPath(path);
44
+ var result = obj;
45
+
46
+ for (var i = 0; i < keyPath.length; i++) {
47
+ var key = keyPath[i];
48
+
49
+ if (!result) {
50
+ break;
51
+ }
52
+
53
+ result = atKey(result, key);
54
+ }
55
+
56
+ return result;
57
+ }
58
+
59
+ function clone(object) {
60
+ return Array.isArray(object) ? [].slice.call(object) : Object.assign({}, object);
61
+ }
62
+
63
+ function getKey(node, key) {
64
+ if (Array.isArray(node) && isNaN(key)) {
65
+ var key_parts = Array.from(key.split('='));
66
+ var attr = key_parts[0];
67
+ var id = key_parts[1];
68
+ var i, child;
69
+
70
+ if (!id) {
71
+ return key;
72
+ }
73
+
74
+ for (i = 0; i < node.length; i++) {
75
+ child = node[i];
76
+
77
+ if (child[attr].toString() === id) {
78
+ break;
79
+ }
80
+ }
81
+
82
+ return i;
83
+ } else {
84
+ return key;
85
+ }
86
+ }
87
+
88
+ function isArray(ary) {
89
+ return Array.isArray(ary);
90
+ }
91
+
92
+ function isObject(obj) {
93
+ return !isArray(obj) && obj === Object(obj);
94
+ }
95
+
96
+ function atKey(node, key) {
97
+ var id, attr;
98
+
99
+ if (isSearchable.test(key)) {
100
+ // eslint-disable-next-line
101
+ ;
102
+
103
+ var _Array$from = Array.from(key.split('='));
104
+
105
+ attr = _Array$from[0];
106
+ id = _Array$from[1];
107
+ }
108
+
109
+ if (!isArray(node) && !isObject(node)) {
110
+ throw new KeyPathError("Expected to traverse an Array or Obj, got " + JSON.stringify(node));
111
+ }
112
+
113
+ if (isObject(node) && id) {
114
+ throw new KeyPathError("Expected to find an Array when using the key: " + key);
115
+ }
116
+
117
+ if (isObject(node) && !Object.prototype.hasOwnProperty.call(node, key)) {
118
+ throw new KeyPathError("Expected to find key: " + key + " in object " + JSON.stringify(node));
119
+ }
120
+
121
+ if (Array.isArray(node) && id) {
122
+ var child;
123
+
124
+ for (var i = 0; i < node.length; i++) {
125
+ child = node[i];
126
+
127
+ if (child[attr].toString() === id) {
128
+ break;
129
+ }
130
+ }
131
+
132
+ if (child[attr].toString() === id) {
133
+ return child;
134
+ } else {
135
+ return undefined;
136
+ }
137
+ } else {
138
+ return node[key];
139
+ }
140
+ }
141
+
142
+ function normalizeKeyPath(path) {
143
+ if (typeof path === 'string') {
144
+ path = path.replace(/ /g, '');
145
+
146
+ if (path === '') {
147
+ return [];
148
+ }
149
+
150
+ return path.split('.');
151
+ } else {
152
+ return path;
153
+ }
154
+ }
155
+
156
+ function setIn(object, keypath, value) {
157
+ keypath = normalizeKeyPath(keypath);
158
+ var results = {};
159
+ var parents = {};
160
+ var i;
161
+ parents[0] = object;
162
+
163
+ for (i = 0; i < keypath.length; i++) {
164
+ parents[i + 1] = atKey(parents[i], keypath[i]);
165
+ }
166
+
167
+ results[keypath.length] = value;
168
+
169
+ for (i = keypath.length - 1; i >= 0; i--) {
170
+ results[i] = clone(parents[i]);
171
+ var key = getKey(results[i], keypath[i]);
172
+ results[i][key] = results[i + 1];
173
+ }
174
+
175
+ return results[0];
176
+ }
package/utils/index.js ADDED
@@ -0,0 +1,59 @@
1
+ "use strict";
2
+
3
+ exports.__esModule = true;
4
+
5
+ var _helpers = require("./helpers");
6
+
7
+ Object.keys(_helpers).forEach(function (key) {
8
+ if (key === "default" || key === "__esModule") return;
9
+ if (key in exports && exports[key] === _helpers[key]) return;
10
+ exports[key] = _helpers[key];
11
+ });
12
+
13
+ var _immutability = require("./immutability");
14
+
15
+ Object.keys(_immutability).forEach(function (key) {
16
+ if (key === "default" || key === "__esModule") return;
17
+ if (key in exports && exports[key] === _immutability[key]) return;
18
+ exports[key] = _immutability[key];
19
+ });
20
+
21
+ var _react = require("./react");
22
+
23
+ Object.keys(_react).forEach(function (key) {
24
+ if (key === "default" || key === "__esModule") return;
25
+ if (key in exports && exports[key] === _react[key]) return;
26
+ exports[key] = _react[key];
27
+ });
28
+
29
+ var _request = require("./request");
30
+
31
+ Object.keys(_request).forEach(function (key) {
32
+ if (key === "default" || key === "__esModule") return;
33
+ if (key in exports && exports[key] === _request[key]) return;
34
+ exports[key] = _request[key];
35
+ });
36
+
37
+ var _ujs = require("./ujs");
38
+
39
+ Object.keys(_ujs).forEach(function (key) {
40
+ if (key === "default" || key === "__esModule") return;
41
+ if (key in exports && exports[key] === _ujs[key]) return;
42
+ exports[key] = _ujs[key];
43
+ });
44
+
45
+ var _url = require("./url");
46
+
47
+ Object.keys(_url).forEach(function (key) {
48
+ if (key === "default" || key === "__esModule") return;
49
+ if (key in exports && exports[key] === _url[key]) return;
50
+ exports[key] = _url[key];
51
+ });
52
+
53
+ var _window = require("./window");
54
+
55
+ Object.keys(_window).forEach(function (key) {
56
+ if (key === "default" || key === "__esModule") return;
57
+ if (key in exports && exports[key] === _window[key]) return;
58
+ exports[key] = _window[key];
59
+ });
package/utils/react.js ADDED
@@ -0,0 +1,52 @@
1
+ "use strict";
2
+
3
+ exports.__esModule = true;
4
+ exports.mapDispatchToPropsIncludingVisitAndRemote = exports.mapDispatchToProps = void 0;
5
+ exports.mapStateToProps = mapStateToProps;
6
+
7
+ var _action_creators = require("../action_creators");
8
+
9
+ var _url = require("./url");
10
+
11
+ function mapStateToProps(state, ownProps) {
12
+ if (state === void 0) {
13
+ state = {
14
+ pages: {},
15
+ superglue: {}
16
+ };
17
+ }
18
+
19
+ var pageKey = ownProps.pageKey;
20
+ var params = ownProps;
21
+ var csrfToken = state.superglue.csrfToken;
22
+ pageKey = (0, _url.urlToPageKey)(pageKey);
23
+
24
+ var _ref = state.pages[pageKey] || {
25
+ data: {},
26
+ flash: {},
27
+ fragments: []
28
+ },
29
+ data = _ref.data,
30
+ flash = _ref.flash,
31
+ fragments = _ref.fragments;
32
+
33
+ return Object.assign({}, data, params, {
34
+ pageKey: pageKey,
35
+ csrfToken: csrfToken,
36
+ flash: flash,
37
+ fragments: fragments
38
+ });
39
+ }
40
+
41
+ var mapDispatchToProps = {
42
+ saveAndProcessPage: _action_creators.saveAndProcessPage,
43
+ copyPage: _action_creators.copyPage
44
+ };
45
+ exports.mapDispatchToProps = mapDispatchToProps;
46
+ var mapDispatchToPropsIncludingVisitAndRemote = {
47
+ visit: _action_creators.visit,
48
+ remote: _action_creators.remote,
49
+ saveAndProcessPage: _action_creators.saveAndProcessPage,
50
+ copyPage: _action_creators.copyPage
51
+ };
52
+ exports.mapDispatchToPropsIncludingVisitAndRemote = mapDispatchToPropsIncludingVisitAndRemote;