@jpmorganchase/elemental-dev-portal 1.0.0 → 2.1.0
Sign up to get free protection for your applications and to get access to all the features.
- package/components/BranchSelector/BranchSelector.d.ts +7 -0
- package/components/BranchSelector/BranchSelector.spec.d.ts +15 -0
- package/components/BranchSelector/BranchSelector.stories.d.ts +29 -0
- package/components/DevPortalProvider/index.d.ts +7 -0
- package/components/Forbidden.d.ts +1 -0
- package/components/Loading.d.ts +1 -0
- package/components/NodeContent/NodeContent.d.ts +17 -0
- package/components/NodeContent/NodeContent.spec.d.ts +1 -0
- package/components/NodeContent/NodeContent.stories.d.ts +43 -0
- package/components/NotFound.d.ts +1 -0
- package/components/Search/Search.d.ts +13 -0
- package/components/Search/Search.stories.d.ts +33 -0
- package/components/TableOfContents/TableOfContents.d.ts +12 -0
- package/components/TableOfContents/TableOfContents.stories.d.ts +36 -0
- package/components/UpgradeToStarter.d.ts +1 -0
- package/consts.d.ts +28 -0
- package/containers/StoplightProject.d.ts +13 -0
- package/containers/StoplightProject.spec.d.ts +1 -0
- package/containers/StoplightProject.stories.d.ts +45 -0
- package/handlers/getBranches.d.ts +6 -0
- package/handlers/getNodeContent.d.ts +12 -0
- package/handlers/getNodes.d.ts +9 -0
- package/handlers/getTableOfContents.d.ts +7 -0
- package/handlers/getWorkspace.d.ts +6 -0
- package/hooks/useGetBranches.d.ts +3 -0
- package/hooks/useGetNodeContent.d.ts +5 -0
- package/hooks/useGetNodes.d.ts +7 -0
- package/hooks/useGetTableOfContents.d.ts +4 -0
- package/hooks/useGetWorkspace.d.ts +3 -0
- package/index.esm.js +682 -0
- package/index.js +726 -0
- package/index.mjs +682 -0
- package/package.json +17 -70
- package/styles.min.css +1 -0
- package/types.d.ts +65 -0
- package/version.d.ts +1 -0
- package/web-components/components.d.ts +1 -0
- package/web-components/index.d.ts +1 -0
- package/web-components.min.js +2 -0
- package/web-components.min.js.LICENSE.txt +187 -0
- package/.storybook/main.js +0 -1
- package/.storybook/manager.js +0 -1
- package/.storybook/preview.jsx +0 -46
- package/jest.config.js +0 -10
- package/src/__fixtures__/branches.json +0 -26
- package/src/__fixtures__/node-content.json +0 -257
- package/src/__fixtures__/table-of-contents.json +0 -144
- package/src/components/BranchSelector/BranchSelector.spec.tsx +0 -63
- package/src/components/BranchSelector/BranchSelector.stories.tsx +0 -41
- package/src/components/BranchSelector/BranchSelector.tsx +0 -50
- package/src/components/DevPortalProvider/index.tsx +0 -19
- package/src/components/Forbidden.tsx +0 -11
- package/src/components/Loading.tsx +0 -9
- package/src/components/NodeContent/NodeContent.spec.tsx +0 -54
- package/src/components/NodeContent/NodeContent.stories.tsx +0 -60
- package/src/components/NodeContent/NodeContent.tsx +0 -171
- package/src/components/NotFound.tsx +0 -11
- package/src/components/Search/Search.stories.tsx +0 -73
- package/src/components/Search/Search.tsx +0 -133
- package/src/components/TableOfContents/TableOfContents.stories.tsx +0 -54
- package/src/components/TableOfContents/TableOfContents.tsx +0 -51
- package/src/components/UpgradeToStarter.tsx +0 -22
- package/src/consts.ts +0 -32
- package/src/containers/StoplightProject.spec.tsx +0 -78
- package/src/containers/StoplightProject.stories.tsx +0 -28
- package/src/containers/StoplightProject.tsx +0 -213
- package/src/handlers/__tests__/getBranches.test.ts +0 -30
- package/src/handlers/__tests__/getNodeContent.test.ts +0 -35
- package/src/handlers/__tests__/getNodes.test.ts +0 -38
- package/src/handlers/__tests__/getTableOfContents.test.ts +0 -34
- package/src/handlers/__tests__/getWorkspace.test.ts +0 -30
- package/src/handlers/getBranches.ts +0 -27
- package/src/handlers/getNodeContent.ts +0 -53
- package/src/handlers/getNodes.ts +0 -69
- package/src/handlers/getTableOfContents.ts +0 -30
- package/src/handlers/getWorkspace.ts +0 -27
- package/src/hooks/__tests__/dataFetching.spec.tsx +0 -42
- package/src/hooks/useGetBranches.ts +0 -17
- package/src/hooks/useGetNodeContent.ts +0 -24
- package/src/hooks/useGetNodes.ts +0 -34
- package/src/hooks/useGetTableOfContents.ts +0 -15
- package/src/hooks/useGetWorkspace.tsx +0 -13
- package/src/styles.css +0 -1
- package/src/types.ts +0 -81
- package/src/version.ts +0 -2
- package/src/web-components/__stories__/StoplightProject.stories.tsx +0 -33
- package/src/web-components/components.ts +0 -17
- package/src/web-components/index.ts +0 -3
- package/tsconfig.build.json +0 -18
- package/tsconfig.json +0 -7
- package/web-components.config.js +0 -1
- /package/{src/components/BranchSelector/index.tsx → components/BranchSelector/index.d.ts} +0 -0
- /package/{src/components/NodeContent/index.tsx → components/NodeContent/index.d.ts} +0 -0
- /package/{src/components/Search/index.tsx → components/Search/index.d.ts} +0 -0
- /package/{src/components/TableOfContents/index.tsx → components/TableOfContents/index.d.ts} +0 -0
- /package/{src/index.ts → index.d.ts} +0 -0
package/index.mjs
ADDED
@@ -0,0 +1,682 @@
|
|
1
|
+
import { Menu, FieldButton, Modal, Input, Box, Icon, ListBox, ListBoxItem, Flex, VStack, Heading } from '@stoplight/mosaic';
|
2
|
+
import * as React from 'react';
|
3
|
+
import React__default, { useRef, useEffect, useMemo, useCallback, useState } from 'react';
|
4
|
+
import { withPersistenceBoundary, withQueryClientProvider, withMosaicProvider, MarkdownComponentsProvider, MockingProvider, Docs, withStyles, NodeTypeIconDefs, NodeTypeColors, TableOfContents as TableOfContents$1, PoweredByLink, useRouter, RouterTypeContext, findFirstNode, ReactRouterMarkdownLink, SidebarLayout } from '@jpmorganchase/elemental-core';
|
5
|
+
import { resolve, dirname } from '@stoplight/path';
|
6
|
+
import { NodeType } from '@stoplight/types';
|
7
|
+
import flow from 'lodash/flow.js';
|
8
|
+
import { Route, useParams, useHistory, Redirect, Link } from 'react-router-dom';
|
9
|
+
import { useQuery } from 'react-query';
|
10
|
+
|
11
|
+
const BranchSelector = ({ branchSlug, branches, onChange }) => {
|
12
|
+
const currentBranch = branches.find(branch => (!branchSlug ? branch.is_default : branch.slug === branchSlug));
|
13
|
+
const handleChange = React.useCallback((selectedSlug) => {
|
14
|
+
const selectedBranch = branches.find(branch => branch.slug === selectedSlug);
|
15
|
+
if (selectedBranch) {
|
16
|
+
onChange(selectedBranch);
|
17
|
+
}
|
18
|
+
}, [onChange, branches]);
|
19
|
+
return (React.createElement(Menu, { "aria-label": "Versions", placement: "bottom left", closeOnPress: true, matchTriggerWidth: true, renderTrigger: ({ isOpen }) => (React.createElement(FieldButton, { w: "full", icon: "layer-group", px: 4, h: "md", active: isOpen, borderR: 0, roundedR: "none" }, (currentBranch === null || currentBranch === void 0 ? void 0 : currentBranch.name) || (currentBranch === null || currentBranch === void 0 ? void 0 : currentBranch.slug) || 'Choose a version')), items: [
|
20
|
+
{
|
21
|
+
type: 'option_group',
|
22
|
+
title: 'Versions',
|
23
|
+
onChange: handleChange,
|
24
|
+
value: (currentBranch === null || currentBranch === void 0 ? void 0 : currentBranch.slug) || '',
|
25
|
+
children: branches.map(branch => ({
|
26
|
+
label: branch.name || branch.slug,
|
27
|
+
value: branch.slug,
|
28
|
+
meta: branch.is_default ? 'Default' : undefined,
|
29
|
+
})),
|
30
|
+
},
|
31
|
+
] }));
|
32
|
+
};
|
33
|
+
|
34
|
+
const PlatformContext = React.createContext({ platformUrl: 'https://stoplight.io' });
|
35
|
+
const PlatformProvider = ({ platformUrl = 'https://stoplight.io', platformAuthToken, children, }) => {
|
36
|
+
return React.createElement(PlatformContext.Provider, { value: { platformUrl, platformAuthToken } }, children);
|
37
|
+
};
|
38
|
+
const DevPortalProvider = withPersistenceBoundary(withQueryClientProvider(withMosaicProvider(PlatformProvider)));
|
39
|
+
|
40
|
+
const NodeContent = ({ node, Link, hideMocking, refResolver, tryItCorsProxy, tryItCredentialsPolicy, nodeHasChanged, compact, hideTryIt, hideTryItPanel, hideExport, onExportRequest, }) => {
|
41
|
+
var _a, _b, _c, _d;
|
42
|
+
return (React.createElement(NodeLinkContext.Provider, { value: [node, Link] },
|
43
|
+
React.createElement(MarkdownComponentsProvider, { value: { a: LinkComponent } },
|
44
|
+
React.createElement(MockingProvider, { mockUrl: node.links.mock_url, hideMocking: hideMocking },
|
45
|
+
React.createElement(Docs, { nodeType: node.type, nodeData: node.data, nodeTitle: node.title, layoutOptions: {
|
46
|
+
compact,
|
47
|
+
hideTryIt: hideTryIt,
|
48
|
+
hideTryItPanel: hideTryItPanel,
|
49
|
+
hideExport: hideExport ||
|
50
|
+
((_b = (_a = node.links.export_url) !== null && _a !== void 0 ? _a : node.links.export_original_file_url) !== null && _b !== void 0 ? _b : node.links.export_bundled_file_url) ===
|
51
|
+
undefined,
|
52
|
+
}, useNodeForRefResolving: true, refResolver: refResolver, tryItCorsProxy: tryItCorsProxy, exportProps: [NodeType.HttpService, NodeType.Model].includes(node.type)
|
53
|
+
? {
|
54
|
+
original: onExportRequest
|
55
|
+
? { onPress: () => onExportRequest('original') }
|
56
|
+
: { href: (_c = node.links.export_original_file_url) !== null && _c !== void 0 ? _c : node.links.export_url },
|
57
|
+
bundled: onExportRequest
|
58
|
+
? { onPress: () => onExportRequest('bundled') }
|
59
|
+
: { href: (_d = node.links.export_bundled_file_url) !== null && _d !== void 0 ? _d : getBundledUrl(node.links.export_url) },
|
60
|
+
}
|
61
|
+
: undefined, tryItCredentialsPolicy: tryItCredentialsPolicy, nodeHasChanged: nodeHasChanged })))));
|
62
|
+
};
|
63
|
+
const NodeLinkContext = React.createContext(undefined);
|
64
|
+
const externalRegex = new RegExp('^(?:[a-z]+:)?//', 'i');
|
65
|
+
const LinkComponent = ({ children, href, title }) => {
|
66
|
+
const ctx = React.useContext(NodeLinkContext);
|
67
|
+
if (href && externalRegex.test(href)) {
|
68
|
+
return (React.createElement("a", { href: href, target: "_blank", rel: "noreferrer", title: title ? title : undefined }, children));
|
69
|
+
}
|
70
|
+
if (href && ctx) {
|
71
|
+
const [node, Link] = ctx;
|
72
|
+
const { fileUri } = getNodeUriParts(node.uri);
|
73
|
+
const { fileUri: hrefFileUri } = getNodeUriParts(href);
|
74
|
+
let resolvedUri;
|
75
|
+
if (hrefFileUri) {
|
76
|
+
resolvedUri = resolve(dirname(fileUri), href);
|
77
|
+
}
|
78
|
+
else {
|
79
|
+
resolvedUri = resolve(fileUri, href);
|
80
|
+
}
|
81
|
+
const [resolvedUriWithoutAnchor, hash] = resolvedUri.split('#');
|
82
|
+
const decodedUrl = decodeURIComponent(href);
|
83
|
+
const decodedResolvedUriWithoutAnchor = decodeURIComponent(resolvedUriWithoutAnchor);
|
84
|
+
const edge = node.outbound_edges.find(edge => edge.uri === decodedUrl || edge.uri === decodedResolvedUriWithoutAnchor);
|
85
|
+
if (edge) {
|
86
|
+
return React.createElement(Link, { to: `${edge.slug}${hash ? `#${hash}` : ''}` }, children);
|
87
|
+
}
|
88
|
+
}
|
89
|
+
return React.createElement("a", { href: href }, children);
|
90
|
+
};
|
91
|
+
function getBundledUrl(url) {
|
92
|
+
if (url === undefined)
|
93
|
+
return undefined;
|
94
|
+
const bundledUrl = new URL(url);
|
95
|
+
const searchParams = new URLSearchParams(bundledUrl.search);
|
96
|
+
searchParams.append('deref', 'optimizedBundle');
|
97
|
+
bundledUrl.search = searchParams.toString();
|
98
|
+
return bundledUrl.toString();
|
99
|
+
}
|
100
|
+
const getNodeUriParts = (uri) => {
|
101
|
+
const parts = uri.split(/(\.yaml|\.yml|\.json|\.md)/);
|
102
|
+
if (parts.length === 1) {
|
103
|
+
return { fileUri: '', pointer: parts[0] || '' };
|
104
|
+
}
|
105
|
+
const fileUri = `${parts[0] || ''}${parts[1] || ''}`;
|
106
|
+
return { fileUri, pointer: parts[2] || '' };
|
107
|
+
};
|
108
|
+
|
109
|
+
/*!
|
110
|
+
* Font Awesome Free 6.1.1 by @fontawesome - https://fontawesome.com
|
111
|
+
* License - https://fontawesome.com/license/free (Icons: CC BY 4.0, Fonts: SIL OFL 1.1, Code: MIT License)
|
112
|
+
* Copyright 2022 Fonticons, Inc.
|
113
|
+
*/
|
114
|
+
var faCircleNotch = {
|
115
|
+
prefix: 'fas',
|
116
|
+
iconName: 'circle-notch',
|
117
|
+
icon: [512, 512, [], "f1ce", "M222.7 32.15C227.7 49.08 218.1 66.9 201.1 71.94C121.8 95.55 64 169.1 64 255.1C64 362 149.1 447.1 256 447.1C362 447.1 448 362 448 255.1C448 169.1 390.2 95.55 310.9 71.94C293.9 66.9 284.3 49.08 289.3 32.15C294.4 15.21 312.2 5.562 329.1 10.6C434.9 42.07 512 139.1 512 255.1C512 397.4 397.4 511.1 256 511.1C114.6 511.1 0 397.4 0 255.1C0 139.1 77.15 42.07 182.9 10.6C199.8 5.562 217.6 15.21 222.7 32.15V32.15z"]
|
118
|
+
};
|
119
|
+
var faMagnifyingGlass = {
|
120
|
+
prefix: 'fas',
|
121
|
+
iconName: 'magnifying-glass',
|
122
|
+
icon: [512, 512, [128269, "search"], "f002", "M500.3 443.7l-119.7-119.7c27.22-40.41 40.65-90.9 33.46-144.7C401.8 87.79 326.8 13.32 235.2 1.723C99.01-15.51-15.51 99.01 1.724 235.2c11.6 91.64 86.08 166.7 177.6 178.9c53.8 7.189 104.3-6.236 144.7-33.46l119.7 119.7c15.62 15.62 40.95 15.62 56.57 0C515.9 484.7 515.9 459.3 500.3 443.7zM79.1 208c0-70.58 57.42-128 128-128s128 57.42 128 128c0 70.58-57.42 128-128 128S79.1 278.6 79.1 208z"]
|
123
|
+
};
|
124
|
+
var faSearch = faMagnifyingGlass;
|
125
|
+
var faSpinner = {
|
126
|
+
prefix: 'fas',
|
127
|
+
iconName: 'spinner',
|
128
|
+
icon: [512, 512, [], "f110", "M304 48C304 74.51 282.5 96 256 96C229.5 96 208 74.51 208 48C208 21.49 229.5 0 256 0C282.5 0 304 21.49 304 48zM304 464C304 490.5 282.5 512 256 512C229.5 512 208 490.5 208 464C208 437.5 229.5 416 256 416C282.5 416 304 437.5 304 464zM0 256C0 229.5 21.49 208 48 208C74.51 208 96 229.5 96 256C96 282.5 74.51 304 48 304C21.49 304 0 282.5 0 256zM512 256C512 282.5 490.5 304 464 304C437.5 304 416 282.5 416 256C416 229.5 437.5 208 464 208C490.5 208 512 229.5 512 256zM74.98 437C56.23 418.3 56.23 387.9 74.98 369.1C93.73 350.4 124.1 350.4 142.9 369.1C161.6 387.9 161.6 418.3 142.9 437C124.1 455.8 93.73 455.8 74.98 437V437zM142.9 142.9C124.1 161.6 93.73 161.6 74.98 142.9C56.24 124.1 56.24 93.73 74.98 74.98C93.73 56.23 124.1 56.23 142.9 74.98C161.6 93.73 161.6 124.1 142.9 142.9zM369.1 369.1C387.9 350.4 418.3 350.4 437 369.1C455.8 387.9 455.8 418.3 437 437C418.3 455.8 387.9 455.8 369.1 437C350.4 418.3 350.4 387.9 369.1 369.1V369.1z"]
|
129
|
+
};
|
130
|
+
|
131
|
+
const SearchImpl = ({ isLoading, search, searchResults, isOpen, onClose, onClick, onSearch }) => {
|
132
|
+
const listBoxRef = React.useRef(null);
|
133
|
+
const onChange = React.useCallback(e => onSearch(e.currentTarget.value), [onSearch]);
|
134
|
+
const onKeyDown = React.useCallback(e => {
|
135
|
+
var _a;
|
136
|
+
if (e.key === 'ArrowDown') {
|
137
|
+
e.preventDefault();
|
138
|
+
(_a = listBoxRef.current) === null || _a === void 0 ? void 0 : _a.focus();
|
139
|
+
}
|
140
|
+
}, []);
|
141
|
+
const onSelectionChange = React.useCallback(keys => {
|
142
|
+
const selectedId = keys.values().next().value;
|
143
|
+
const selectedResult = searchResults === null || searchResults === void 0 ? void 0 : searchResults.find(searchResult => `${searchResult.id}-${searchResult.project_id}` === selectedId);
|
144
|
+
if (selectedResult) {
|
145
|
+
onClick(selectedResult);
|
146
|
+
}
|
147
|
+
}, [searchResults, onClick]);
|
148
|
+
return (React.createElement(Modal, { renderHeader: () => (React.createElement(Input, { appearance: "minimal", borderB: true, size: "lg", icon: React.createElement(Box, { as: Icon, ml: 1, icon: isLoading ? faSpinner : faSearch, spin: isLoading }), autoFocus: true, placeholder: "Search...", value: search, onChange: onChange, onKeyDown: onKeyDown })), isOpen: !!isOpen, onClose: onClose }, searchResults && searchResults.length > 0 ? (React.createElement(ListBox, { ref: listBoxRef, "aria-label": "Search", overflowY: "auto", h: 80, m: -5, items: searchResults, selectionMode: "single", onSelectionChange: onSelectionChange }, (searchResult) => {
|
149
|
+
var _a, _b;
|
150
|
+
return (React.createElement(ListBoxItem, { key: `${searchResult.id}-${searchResult.project_id}`, textValue: searchResult.title },
|
151
|
+
React.createElement(Box, { p: 3, borderB: true },
|
152
|
+
React.createElement(Flex, { align: "center" },
|
153
|
+
React.createElement(Box, { as: Icon, w: 4, icon: NodeTypeIconDefs[searchResult.type], style: { color: NodeTypeColors[searchResult.type] } }),
|
154
|
+
React.createElement(Box, { flex: 1, fontSize: "lg", dangerouslySetInnerHTML: { __html: (_a = searchResult.highlighted.name) !== null && _a !== void 0 ? _a : '' }, fontWeight: "medium", textOverflow: "overflow-ellipsis", mx: 2 }),
|
155
|
+
React.createElement(Box, { fontSize: "sm", color: "muted" }, searchResult.project_name)),
|
156
|
+
React.createElement(Box, { dangerouslySetInnerHTML: { __html: (_b = searchResult.highlighted.summary) !== null && _b !== void 0 ? _b : '' }, color: "muted", fontSize: "sm", mt: 1, ml: 6 }))));
|
157
|
+
})) : (React.createElement(Flex, { w: "full", h: 80, align: "center", justify: "center", m: -5 }, "No search results"))));
|
158
|
+
};
|
159
|
+
const Search = flow(withStyles, withPersistenceBoundary, withMosaicProvider, withQueryClientProvider)(SearchImpl);
|
160
|
+
|
161
|
+
/*! *****************************************************************************
|
162
|
+
Copyright (c) Microsoft Corporation.
|
163
|
+
|
164
|
+
Permission to use, copy, modify, and/or distribute this software for any
|
165
|
+
purpose with or without fee is hereby granted.
|
166
|
+
|
167
|
+
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
|
168
|
+
REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
|
169
|
+
AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
|
170
|
+
INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
|
171
|
+
LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
|
172
|
+
OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
173
|
+
PERFORMANCE OF THIS SOFTWARE.
|
174
|
+
***************************************************************************** */
|
175
|
+
|
176
|
+
function __rest(s, e) {
|
177
|
+
var t = {};
|
178
|
+
for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)
|
179
|
+
t[p] = s[p];
|
180
|
+
if (s != null && typeof Object.getOwnPropertySymbols === "function")
|
181
|
+
for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {
|
182
|
+
if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))
|
183
|
+
t[p[i]] = s[p[i]];
|
184
|
+
}
|
185
|
+
return t;
|
186
|
+
}
|
187
|
+
|
188
|
+
function __awaiter(thisArg, _arguments, P, generator) {
|
189
|
+
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
190
|
+
return new (P || (P = Promise))(function (resolve, reject) {
|
191
|
+
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
192
|
+
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
193
|
+
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
194
|
+
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
195
|
+
});
|
196
|
+
}
|
197
|
+
|
198
|
+
const TableOfContents = (_a) => {
|
199
|
+
var { tableOfContents, activeId, Link, collapseTableOfContents = false, externalScrollbar, onLinkClick } = _a, boxProps = __rest(_a, ["tableOfContents", "activeId", "Link", "collapseTableOfContents", "externalScrollbar", "onLinkClick"]);
|
200
|
+
return (React.createElement(Flex, Object.assign({ bg: "canvas-100" }, boxProps, { flexDirection: "col", maxH: "full" }),
|
201
|
+
React.createElement(Flex, { flexGrow: true, flexShrink: true, overflowY: "auto" },
|
202
|
+
React.createElement(TableOfContents$1, { tree: tableOfContents.items, activeId: activeId, Link: Link, maxDepthOpenByDefault: collapseTableOfContents ? 0 : 1, externalScrollbar: externalScrollbar, onLinkClick: onLinkClick })),
|
203
|
+
tableOfContents.hide_powered_by ? null : (React.createElement(PoweredByLink, { source: activeId, pathname: typeof window !== 'undefined' ? window.location.pathname : '', packageType: "elements-dev-portal" }))));
|
204
|
+
};
|
205
|
+
|
206
|
+
const ROOT_CACHE_KEY = '@stoplight/elements-dev-portal/client-query';
|
207
|
+
const devPortalCacheKeys = {
|
208
|
+
all: [ROOT_CACHE_KEY],
|
209
|
+
projects: () => [ROOT_CACHE_KEY, 'projects'],
|
210
|
+
project: (projectId) => [...devPortalCacheKeys.projects(), projectId],
|
211
|
+
projectsList: () => [...devPortalCacheKeys.projects(), 'list'],
|
212
|
+
projectDetails: (projectId) => [...devPortalCacheKeys.project(projectId), 'details'],
|
213
|
+
branches: (projectId) => [...devPortalCacheKeys.project(projectId), 'branches'],
|
214
|
+
branch: (projectId, branch) => [...devPortalCacheKeys.branches(projectId), branch],
|
215
|
+
branchesList: (projectId) => [...devPortalCacheKeys.branches(projectId), 'list'],
|
216
|
+
branchDetails: (projectId, branch) => [...devPortalCacheKeys.branch(projectId, branch), 'details'],
|
217
|
+
branchTOC: (projectId, branch) => [...devPortalCacheKeys.branch(projectId, branch), 'toc'],
|
218
|
+
branchNodes: (projectId, branch) => [...devPortalCacheKeys.branch(projectId, branch), 'nodes'],
|
219
|
+
branchNode: (projectId, branch, node) => [...devPortalCacheKeys.branchNodes(projectId, branch), node],
|
220
|
+
branchNodesList: (projectId, branch) => [...devPortalCacheKeys.branchNodes(projectId, branch), 'list'],
|
221
|
+
branchNodeDetails: (projectId, branch, node) => [...devPortalCacheKeys.branchNode(projectId, branch, node), 'details'],
|
222
|
+
search: () => [...devPortalCacheKeys.all, 'search'],
|
223
|
+
searchNodes: (filters) => [
|
224
|
+
...devPortalCacheKeys.search(),
|
225
|
+
filters,
|
226
|
+
],
|
227
|
+
};
|
228
|
+
|
229
|
+
const Forbidden = () => (React__default.createElement(Flex, { align: "center", justify: "center", flexGrow: true },
|
230
|
+
React__default.createElement(VStack, { spacing: 4, align: "center" },
|
231
|
+
React__default.createElement(Heading, { size: 1 }, "Forbidden"),
|
232
|
+
React__default.createElement(Box, { as: "p" }, "You don't have permission to access this resource"))));
|
233
|
+
|
234
|
+
const Loading = () => (React__default.createElement(Flex, { justify: "center", alignItems: "center", w: "full", minH: "screen", color: "muted" },
|
235
|
+
React__default.createElement(Icon, { icon: faCircleNotch, size: "3x", spin: true })));
|
236
|
+
|
237
|
+
const NotFound = () => (React__default.createElement(Flex, { align: "center", justify: "center", flexGrow: true },
|
238
|
+
React__default.createElement(VStack, { spacing: 4, align: "center" },
|
239
|
+
React__default.createElement(Heading, { size: 1 }, "Not Found"),
|
240
|
+
React__default.createElement(Box, { as: "p" }, "Could not find what you are looking for"))));
|
241
|
+
|
242
|
+
const UpgradeToStarter = () => (React__default.createElement(Flex, { as: "a", href: "https://stoplight.io/pricing/", target: "_blank", rel: "noreferrer noopener", justify: "center", alignItems: "center", w: "full", minH: "screen", color: "muted", flexDirection: "col" },
|
243
|
+
React__default.createElement(Icon, { icon: ['fas', 'exclamation-triangle'], size: "4x" }),
|
244
|
+
React__default.createElement(Box, { pt: 3 }, "Please upgrade your Stoplight Workspace to the Starter Plan to use Elements Dev Portal in production.")));
|
245
|
+
|
246
|
+
const appVersion = '2.1.0';
|
247
|
+
|
248
|
+
class ResponseError extends Error {
|
249
|
+
constructor(message, responseCode) {
|
250
|
+
super(message);
|
251
|
+
this.name = 'ResponseError';
|
252
|
+
this.code = responseCode;
|
253
|
+
}
|
254
|
+
}
|
255
|
+
const getNodeContent = ({ nodeSlug, projectId, branchSlug, platformUrl = 'https://stoplight.io', platformAuthToken, }) => __awaiter(void 0, void 0, void 0, function* () {
|
256
|
+
const encodedNodeSlug = encodeURIComponent(nodeSlug);
|
257
|
+
const encodedProjectId = encodeURIComponent(projectId);
|
258
|
+
const encodedBranchSlug = branchSlug ? encodeURIComponent(branchSlug) : '';
|
259
|
+
const branchQuery = encodedBranchSlug ? `?branch=${encodedBranchSlug}` : '';
|
260
|
+
const response = yield fetch(`${platformUrl}/api/v1/projects/${encodedProjectId}/nodes/${encodedNodeSlug}${branchQuery}`, {
|
261
|
+
headers: Object.assign({ 'Stoplight-Elements-Version': appVersion }, (platformAuthToken && { Authorization: `Bearer ${platformAuthToken}` })),
|
262
|
+
});
|
263
|
+
const data = yield response.json();
|
264
|
+
if (!response.ok) {
|
265
|
+
if (response.status === 402) {
|
266
|
+
throw new ResponseError('Payment Required', response.status);
|
267
|
+
}
|
268
|
+
else if (response.status === 403) {
|
269
|
+
throw new ResponseError('Forbidden', response.status);
|
270
|
+
}
|
271
|
+
else {
|
272
|
+
throw new ResponseError('Something went wrong', response.status);
|
273
|
+
}
|
274
|
+
}
|
275
|
+
return data;
|
276
|
+
});
|
277
|
+
|
278
|
+
const getBranches = ({ projectId, platformUrl = 'https://stoplight.io', platformAuthToken, }) => __awaiter(void 0, void 0, void 0, function* () {
|
279
|
+
const encodedProjectId = encodeURIComponent(projectId);
|
280
|
+
const response = yield fetch(`${platformUrl}/api/v1/projects/${encodedProjectId}/branches`, {
|
281
|
+
headers: Object.assign({ 'Stoplight-Elements-Version': appVersion }, (platformAuthToken && { Authorization: `Bearer ${platformAuthToken}` })),
|
282
|
+
});
|
283
|
+
const data = yield response.json();
|
284
|
+
if (!response.ok) {
|
285
|
+
throw new Error(data);
|
286
|
+
}
|
287
|
+
return data;
|
288
|
+
});
|
289
|
+
|
290
|
+
function useGetBranches({ projectId }) {
|
291
|
+
const { platformUrl, platformAuthToken } = React.useContext(PlatformContext);
|
292
|
+
return useQuery([...devPortalCacheKeys.branchesList(projectId), platformUrl, platformAuthToken], () => getBranches({ projectId, platformUrl, platformAuthToken }), {
|
293
|
+
enabled: projectId ? true : false,
|
294
|
+
});
|
295
|
+
}
|
296
|
+
|
297
|
+
function useGetNodeContent({ nodeSlug, projectId, branchSlug, }) {
|
298
|
+
const { platformUrl, platformAuthToken } = React.useContext(PlatformContext);
|
299
|
+
return useQuery([...devPortalCacheKeys.branchNodeDetails(projectId, branchSlug !== null && branchSlug !== void 0 ? branchSlug : '', nodeSlug), platformUrl, platformAuthToken], () => getNodeContent({ nodeSlug, projectId, branchSlug, platformUrl, platformAuthToken }), { enabled: nodeSlug && projectId ? true : false });
|
300
|
+
}
|
301
|
+
|
302
|
+
const getTableOfContents = ({ projectId, branchSlug, platformUrl = 'https://stoplight.io', platformAuthToken, }) => __awaiter(void 0, void 0, void 0, function* () {
|
303
|
+
const encodedProjectId = encodeURIComponent(projectId);
|
304
|
+
const encodedBranchSlug = branchSlug ? encodeURIComponent(branchSlug) : '';
|
305
|
+
const branchQuery = encodedBranchSlug ? `?branch=${encodedBranchSlug}` : '';
|
306
|
+
const response = yield fetch(`${platformUrl}/api/v1/projects/${encodedProjectId}/table-of-contents${branchQuery}`, {
|
307
|
+
headers: Object.assign({ 'Stoplight-Elements-Version': appVersion }, (platformAuthToken && { Authorization: `Bearer ${platformAuthToken}` })),
|
308
|
+
});
|
309
|
+
const data = yield response.json();
|
310
|
+
if (!response.ok) {
|
311
|
+
throw new Error(data);
|
312
|
+
}
|
313
|
+
return data;
|
314
|
+
});
|
315
|
+
|
316
|
+
function useGetTableOfContents({ projectId, branchSlug }) {
|
317
|
+
const { platformUrl, platformAuthToken } = React.useContext(PlatformContext);
|
318
|
+
return useQuery([...devPortalCacheKeys.branchTOC(projectId, branchSlug !== null && branchSlug !== void 0 ? branchSlug : ''), platformUrl, platformAuthToken], () => getTableOfContents({ projectId, branchSlug, platformUrl, platformAuthToken }), { enabled: projectId ? true : false });
|
319
|
+
}
|
320
|
+
|
321
|
+
const StoplightProjectImpl = ({ projectId, hideTryIt, hideMocking, hideExport, collapseTableOfContents = false, tryItCredentialsPolicy, tryItCorsProxy, }) => {
|
322
|
+
const { branchSlug = '', nodeSlug = '' } = useParams();
|
323
|
+
const history = useHistory();
|
324
|
+
const { data: tableOfContents, isFetched: isTocFetched } = useGetTableOfContents({ projectId, branchSlug });
|
325
|
+
const { data: branches } = useGetBranches({ projectId });
|
326
|
+
const { data: node, isLoading: isLoadingNode, isError, error: nodeError, } = useGetNodeContent({
|
327
|
+
nodeSlug,
|
328
|
+
projectId,
|
329
|
+
branchSlug,
|
330
|
+
});
|
331
|
+
const container = React.useRef(null);
|
332
|
+
if (!nodeSlug && isTocFetched && (tableOfContents === null || tableOfContents === void 0 ? void 0 : tableOfContents.items)) {
|
333
|
+
const firstNode = findFirstNode(tableOfContents.items);
|
334
|
+
if (firstNode) {
|
335
|
+
return React.createElement(Redirect, { to: branchSlug ? `/branches/${branchSlug}/${firstNode.slug}` : `/${firstNode.slug}` });
|
336
|
+
}
|
337
|
+
}
|
338
|
+
let elem;
|
339
|
+
if (isLoadingNode || !isTocFetched) {
|
340
|
+
elem = React.createElement(Loading, null);
|
341
|
+
}
|
342
|
+
else if (isError) {
|
343
|
+
if (nodeError instanceof ResponseError) {
|
344
|
+
if (nodeError.code === 402) {
|
345
|
+
elem = React.createElement(UpgradeToStarter, null);
|
346
|
+
}
|
347
|
+
else if (nodeError.code === 403) {
|
348
|
+
elem = React.createElement(Forbidden, null);
|
349
|
+
}
|
350
|
+
else {
|
351
|
+
elem = React.createElement(NotFound, null);
|
352
|
+
}
|
353
|
+
}
|
354
|
+
else {
|
355
|
+
elem = React.createElement(NotFound, null);
|
356
|
+
}
|
357
|
+
}
|
358
|
+
else if (!node) {
|
359
|
+
elem = React.createElement(NotFound, null);
|
360
|
+
}
|
361
|
+
else if ((node === null || node === void 0 ? void 0 : node.slug) && nodeSlug !== node.slug) {
|
362
|
+
return React.createElement(Redirect, { to: branchSlug ? `/branches/${branchSlug}/${node.slug}` : `/${node.slug}` });
|
363
|
+
}
|
364
|
+
else {
|
365
|
+
elem = (React.createElement(NodeContent, { node: node, Link: ReactRouterMarkdownLink, hideTryIt: hideTryIt, hideMocking: hideMocking, hideExport: hideExport, tryItCredentialsPolicy: tryItCredentialsPolicy, tryItCorsProxy: tryItCorsProxy }));
|
366
|
+
}
|
367
|
+
const handleTocClick = () => {
|
368
|
+
if (container.current) {
|
369
|
+
container.current.scrollIntoView();
|
370
|
+
}
|
371
|
+
};
|
372
|
+
return (React.createElement(SidebarLayout, { ref: container, sidebar: React.createElement(React.Fragment, null,
|
373
|
+
branches && branches.length > 1 ? (React.createElement(BranchSelector, { branchSlug: branchSlug, branches: branches, onChange: branch => history.push(branch.is_default ? `/${nodeSlug}` : `/branches/${branch.slug}/${nodeSlug}`) })) : null,
|
374
|
+
tableOfContents ? (React.createElement(TableOfContents, { activeId: (node === null || node === void 0 ? void 0 : node.id) || (nodeSlug === null || nodeSlug === void 0 ? void 0 : nodeSlug.split('-')[0]) || '', tableOfContents: tableOfContents, Link: Link, collapseTableOfContents: collapseTableOfContents, onLinkClick: handleTocClick })) : null) }, elem));
|
375
|
+
};
|
376
|
+
const StoplightProjectRouter = (_a) => {
|
377
|
+
var { platformUrl, basePath = '/', staticRouterPath = '', router = 'history' } = _a, props = __rest(_a, ["platformUrl", "basePath", "staticRouterPath", "router"]);
|
378
|
+
const { Router, routerProps } = useRouter(router, basePath, staticRouterPath);
|
379
|
+
return (React.createElement(DevPortalProvider, { platformUrl: platformUrl },
|
380
|
+
React.createElement(RouterTypeContext.Provider, { value: router },
|
381
|
+
React.createElement(Router, Object.assign({}, routerProps, { key: basePath }),
|
382
|
+
React.createElement(Route, { path: "/branches/:branchSlug/:nodeSlug", exact: true },
|
383
|
+
React.createElement(StoplightProjectImpl, Object.assign({}, props))),
|
384
|
+
React.createElement(Route, { path: "/:nodeSlug", exact: true },
|
385
|
+
React.createElement(StoplightProjectImpl, Object.assign({}, props))),
|
386
|
+
React.createElement(Route, { path: "/", exact: true },
|
387
|
+
React.createElement(StoplightProjectImpl, Object.assign({}, props)))))));
|
388
|
+
};
|
389
|
+
const StoplightProject = withStyles(StoplightProjectRouter);
|
390
|
+
|
391
|
+
const getNodes = ({ workspaceId, branchSlug, projectIds, search, platformUrl = 'https://stoplight.io', platformAuthToken, }) => __awaiter(void 0, void 0, void 0, function* () {
|
392
|
+
const queryParams = [];
|
393
|
+
let fetchedWorkspaceId = workspaceId || '';
|
394
|
+
if (!workspaceId && (projectIds === null || projectIds === void 0 ? void 0 : projectIds.length)) {
|
395
|
+
const encodedProjectId = encodeURIComponent(projectIds[0]);
|
396
|
+
const response = yield fetch(`${platformUrl}/api/v1/projects/${encodedProjectId}`, {
|
397
|
+
headers: Object.assign({ 'Stoplight-Elements-Version': appVersion }, (platformAuthToken && { Authorization: `Bearer ${platformAuthToken}` })),
|
398
|
+
});
|
399
|
+
const data = yield response.json();
|
400
|
+
fetchedWorkspaceId = data.workspace.id;
|
401
|
+
}
|
402
|
+
if (projectIds && projectIds.length) {
|
403
|
+
queryParams.push(...projectIds.map((projectId, index) => {
|
404
|
+
const encodedProjectId = encodeURIComponent(projectId);
|
405
|
+
return `project_ids[${index}]=${encodedProjectId}`;
|
406
|
+
}));
|
407
|
+
}
|
408
|
+
if (search) {
|
409
|
+
const encodedSearch = encodeURIComponent(search);
|
410
|
+
queryParams.push(`search=${encodedSearch}`);
|
411
|
+
}
|
412
|
+
if (branchSlug) {
|
413
|
+
const encodedBranchSlug = encodeURIComponent(branchSlug);
|
414
|
+
queryParams.push(`branch=${encodedBranchSlug}`);
|
415
|
+
}
|
416
|
+
const query = queryParams.length ? `?${queryParams.join('&')}` : '';
|
417
|
+
const encodedWorkspaceId = encodeURIComponent(fetchedWorkspaceId);
|
418
|
+
const response = yield fetch(`${platformUrl}/api/v1/workspaces/${encodedWorkspaceId}/nodes${query}`, {
|
419
|
+
headers: Object.assign({ 'Stoplight-Elements-Version': appVersion }, (platformAuthToken && { Authorization: `Bearer ${platformAuthToken}` })),
|
420
|
+
});
|
421
|
+
const data = yield response.json();
|
422
|
+
if (!response.ok) {
|
423
|
+
throw new Error(data);
|
424
|
+
}
|
425
|
+
return data;
|
426
|
+
});
|
427
|
+
|
428
|
+
const getWorkspace = ({ projectIds, platformUrl = 'https://stoplight.io', platformAuthToken, }) => __awaiter(void 0, void 0, void 0, function* () {
|
429
|
+
const encodedProjectId = encodeURIComponent(projectIds[0]);
|
430
|
+
const response = yield fetch(`${platformUrl}/api/v1/projects/${encodedProjectId}`, {
|
431
|
+
headers: Object.assign({ 'Stoplight-Elements-Version': appVersion }, (platformAuthToken && { Authorization: `Bearer ${platformAuthToken}` })),
|
432
|
+
});
|
433
|
+
const data = yield response.json();
|
434
|
+
if (!response.ok) {
|
435
|
+
throw new Error(data);
|
436
|
+
}
|
437
|
+
return data;
|
438
|
+
});
|
439
|
+
|
440
|
+
/**
|
441
|
+
* Creates a debounced function that delays invoking `func` until after `wait`
|
442
|
+
* milliseconds have elapsed since the last time the debounced function was
|
443
|
+
* invoked, or until the next browser frame is drawn. The debounced function
|
444
|
+
* comes with a `cancel` method to cancel delayed `func` invocations and a
|
445
|
+
* `flush` method to immediately invoke them. Provide `options` to indicate
|
446
|
+
* whether `func` should be invoked on the leading and/or trailing edge of the
|
447
|
+
* `wait` timeout. The `func` is invoked with the last arguments provided to the
|
448
|
+
* debounced function. Subsequent calls to the debounced function return the
|
449
|
+
* result of the last `func` invocation.
|
450
|
+
*
|
451
|
+
* **Note:** If `leading` and `trailing` options are `true`, `func` is
|
452
|
+
* invoked on the trailing edge of the timeout only if the debounced function
|
453
|
+
* is invoked more than once during the `wait` timeout.
|
454
|
+
*
|
455
|
+
* If `wait` is `0` and `leading` is `false`, `func` invocation is deferred
|
456
|
+
* until the next tick, similar to `setTimeout` with a timeout of `0`.
|
457
|
+
*
|
458
|
+
* If `wait` is omitted in an environment with `requestAnimationFrame`, `func`
|
459
|
+
* invocation will be deferred until the next frame is drawn (typically about
|
460
|
+
* 16ms).
|
461
|
+
*
|
462
|
+
* See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/)
|
463
|
+
* for details over the differences between `debounce` and `throttle`.
|
464
|
+
*
|
465
|
+
* @category Function
|
466
|
+
* @param {Function} func The function to debounce.
|
467
|
+
* @param {number} [wait=0]
|
468
|
+
* The number of milliseconds to delay; if omitted, `requestAnimationFrame` is
|
469
|
+
* used (if available, otherwise it will be setTimeout(...,0)).
|
470
|
+
* @param {Object} [options={}] The options object.
|
471
|
+
* Specify invoking on the leading edge of the timeout.
|
472
|
+
* @param {boolean} [options.leading=false]
|
473
|
+
* The maximum time `func` is allowed to be delayed before it's invoked.
|
474
|
+
* @param {number} [options.maxWait]
|
475
|
+
* Specify invoking on the trailing edge of the timeout.
|
476
|
+
* @param {boolean} [options.trailing=true]
|
477
|
+
* @returns {Function} Returns the new debounced function.
|
478
|
+
* @example
|
479
|
+
*
|
480
|
+
* // Avoid costly calculations while the window size is in flux.
|
481
|
+
* const resizeHandler = useDebouncedCallback(calculateLayout, 150);
|
482
|
+
* window.addEventListener('resize', resizeHandler)
|
483
|
+
*
|
484
|
+
* // Invoke `sendMail` when clicked, debouncing subsequent calls.
|
485
|
+
* const clickHandler = useDebouncedCallback(sendMail, 300, {
|
486
|
+
* leading: true,
|
487
|
+
* trailing: false,
|
488
|
+
* })
|
489
|
+
* <button onClick={clickHandler}>click me</button>
|
490
|
+
*
|
491
|
+
* // Ensure `batchLog` is invoked once after 1 second of debounced calls.
|
492
|
+
* const debounced = useDebouncedCallback(batchLog, 250, { 'maxWait': 1000 })
|
493
|
+
* const source = new EventSource('/stream')
|
494
|
+
* source.addEventListener('message', debounced)
|
495
|
+
*
|
496
|
+
* // Cancel the trailing debounced invocation.
|
497
|
+
* window.addEventListener('popstate', debounced.cancel)
|
498
|
+
*
|
499
|
+
* // Check for pending invocations.
|
500
|
+
* const status = debounced.pending() ? "Pending..." : "Ready"
|
501
|
+
*/
|
502
|
+
function useDebouncedCallback(func, wait, options) {
|
503
|
+
var _this = this;
|
504
|
+
var lastCallTime = useRef(null);
|
505
|
+
var lastInvokeTime = useRef(0);
|
506
|
+
var timerId = useRef(null);
|
507
|
+
var lastArgs = useRef([]);
|
508
|
+
var lastThis = useRef();
|
509
|
+
var result = useRef();
|
510
|
+
var funcRef = useRef(func);
|
511
|
+
var mounted = useRef(true);
|
512
|
+
funcRef.current = func;
|
513
|
+
// Bypass `requestAnimationFrame` by explicitly setting `wait=0`.
|
514
|
+
var useRAF = !wait && wait !== 0 && typeof window !== 'undefined';
|
515
|
+
if (typeof func !== 'function') {
|
516
|
+
throw new TypeError('Expected a function');
|
517
|
+
}
|
518
|
+
wait = +wait || 0;
|
519
|
+
options = options || {};
|
520
|
+
var leading = !!options.leading;
|
521
|
+
var trailing = 'trailing' in options ? !!options.trailing : true; // `true` by default
|
522
|
+
var maxing = 'maxWait' in options;
|
523
|
+
var maxWait = maxing ? Math.max(+options.maxWait || 0, wait) : null;
|
524
|
+
useEffect(function () {
|
525
|
+
mounted.current = true;
|
526
|
+
return function () {
|
527
|
+
mounted.current = false;
|
528
|
+
};
|
529
|
+
}, []);
|
530
|
+
// You may have a question, why we have so many code under the useMemo definition.
|
531
|
+
//
|
532
|
+
// This was made as we want to escape from useCallback hell and
|
533
|
+
// not to initialize a number of functions each time useDebouncedCallback is called.
|
534
|
+
//
|
535
|
+
// It means that we have less garbage for our GC calls which improves performance.
|
536
|
+
// Also, it makes this library smaller.
|
537
|
+
//
|
538
|
+
// And the last reason, that the code without lots of useCallback with deps is easier to read.
|
539
|
+
// You have only one place for that.
|
540
|
+
var debounced = useMemo(function () {
|
541
|
+
var invokeFunc = function (time) {
|
542
|
+
var args = lastArgs.current;
|
543
|
+
var thisArg = lastThis.current;
|
544
|
+
lastArgs.current = lastThis.current = null;
|
545
|
+
lastInvokeTime.current = time;
|
546
|
+
return (result.current = funcRef.current.apply(thisArg, args));
|
547
|
+
};
|
548
|
+
var startTimer = function (pendingFunc, wait) {
|
549
|
+
if (useRAF)
|
550
|
+
cancelAnimationFrame(timerId.current);
|
551
|
+
timerId.current = useRAF ? requestAnimationFrame(pendingFunc) : setTimeout(pendingFunc, wait);
|
552
|
+
};
|
553
|
+
var shouldInvoke = function (time) {
|
554
|
+
if (!mounted.current)
|
555
|
+
return false;
|
556
|
+
var timeSinceLastCall = time - lastCallTime.current;
|
557
|
+
var timeSinceLastInvoke = time - lastInvokeTime.current;
|
558
|
+
// Either this is the first call, activity has stopped and we're at the
|
559
|
+
// trailing edge, the system time has gone backwards and we're treating
|
560
|
+
// it as the trailing edge, or we've hit the `maxWait` limit.
|
561
|
+
return (!lastCallTime.current ||
|
562
|
+
timeSinceLastCall >= wait ||
|
563
|
+
timeSinceLastCall < 0 ||
|
564
|
+
(maxing && timeSinceLastInvoke >= maxWait));
|
565
|
+
};
|
566
|
+
var trailingEdge = function (time) {
|
567
|
+
timerId.current = null;
|
568
|
+
// Only invoke if we have `lastArgs` which means `func` has been
|
569
|
+
// debounced at least once.
|
570
|
+
if (trailing && lastArgs.current) {
|
571
|
+
return invokeFunc(time);
|
572
|
+
}
|
573
|
+
lastArgs.current = lastThis.current = null;
|
574
|
+
return result.current;
|
575
|
+
};
|
576
|
+
var timerExpired = function () {
|
577
|
+
var time = Date.now();
|
578
|
+
if (shouldInvoke(time)) {
|
579
|
+
return trailingEdge(time);
|
580
|
+
}
|
581
|
+
// https://github.com/xnimorz/use-debounce/issues/97
|
582
|
+
if (!mounted.current) {
|
583
|
+
return;
|
584
|
+
}
|
585
|
+
// Remaining wait calculation
|
586
|
+
var timeSinceLastCall = time - lastCallTime.current;
|
587
|
+
var timeSinceLastInvoke = time - lastInvokeTime.current;
|
588
|
+
var timeWaiting = wait - timeSinceLastCall;
|
589
|
+
var remainingWait = maxing ? Math.min(timeWaiting, maxWait - timeSinceLastInvoke) : timeWaiting;
|
590
|
+
// Restart the timer
|
591
|
+
startTimer(timerExpired, remainingWait);
|
592
|
+
};
|
593
|
+
var func = function () {
|
594
|
+
var args = [];
|
595
|
+
for (var _i = 0; _i < arguments.length; _i++) {
|
596
|
+
args[_i] = arguments[_i];
|
597
|
+
}
|
598
|
+
var time = Date.now();
|
599
|
+
var isInvoking = shouldInvoke(time);
|
600
|
+
lastArgs.current = args;
|
601
|
+
lastThis.current = _this;
|
602
|
+
lastCallTime.current = time;
|
603
|
+
if (isInvoking) {
|
604
|
+
if (!timerId.current && mounted.current) {
|
605
|
+
// Reset any `maxWait` timer.
|
606
|
+
lastInvokeTime.current = lastCallTime.current;
|
607
|
+
// Start the timer for the trailing edge.
|
608
|
+
startTimer(timerExpired, wait);
|
609
|
+
// Invoke the leading edge.
|
610
|
+
return leading ? invokeFunc(lastCallTime.current) : result.current;
|
611
|
+
}
|
612
|
+
if (maxing) {
|
613
|
+
// Handle invocations in a tight loop.
|
614
|
+
startTimer(timerExpired, wait);
|
615
|
+
return invokeFunc(lastCallTime.current);
|
616
|
+
}
|
617
|
+
}
|
618
|
+
if (!timerId.current) {
|
619
|
+
startTimer(timerExpired, wait);
|
620
|
+
}
|
621
|
+
return result.current;
|
622
|
+
};
|
623
|
+
func.cancel = function () {
|
624
|
+
if (timerId.current) {
|
625
|
+
useRAF ? cancelAnimationFrame(timerId.current) : clearTimeout(timerId.current);
|
626
|
+
}
|
627
|
+
lastInvokeTime.current = 0;
|
628
|
+
lastArgs.current = lastCallTime.current = lastThis.current = timerId.current = null;
|
629
|
+
};
|
630
|
+
func.isPending = function () {
|
631
|
+
return !!timerId.current;
|
632
|
+
};
|
633
|
+
func.flush = function () {
|
634
|
+
return !timerId.current ? result.current : trailingEdge(Date.now());
|
635
|
+
};
|
636
|
+
return func;
|
637
|
+
}, [leading, maxing, wait, maxWait, trailing, useRAF]);
|
638
|
+
return debounced;
|
639
|
+
}
|
640
|
+
|
641
|
+
function valueEquality(left, right) {
|
642
|
+
return left === right;
|
643
|
+
}
|
644
|
+
function adjustFunctionValueOfSetState(value) {
|
645
|
+
return typeof value === 'function' ? function () { return value; } : value;
|
646
|
+
}
|
647
|
+
function useStateIgnoreCallback(initialState) {
|
648
|
+
var _a = useState(adjustFunctionValueOfSetState(initialState)), state = _a[0], setState = _a[1];
|
649
|
+
var setStateIgnoreCallback = useCallback(function (value) { return setState(adjustFunctionValueOfSetState(value)); }, []);
|
650
|
+
return [state, setStateIgnoreCallback];
|
651
|
+
}
|
652
|
+
function useDebounce(value, delay, options) {
|
653
|
+
var eq = (options && options.equalityFn) || valueEquality;
|
654
|
+
var _a = useStateIgnoreCallback(value), state = _a[0], dispatch = _a[1];
|
655
|
+
var debounced = useDebouncedCallback(useCallback(function (value) { return dispatch(value); }, [dispatch]), delay, options);
|
656
|
+
var previousValue = useRef(value);
|
657
|
+
useEffect(function () {
|
658
|
+
// We need to use this condition otherwise we will run debounce timer for the first render (including maxWait option)
|
659
|
+
if (!eq(previousValue.current, value)) {
|
660
|
+
debounced(value);
|
661
|
+
previousValue.current = value;
|
662
|
+
}
|
663
|
+
}, [value, debounced, eq]);
|
664
|
+
return [state, { cancel: debounced.cancel, isPending: debounced.isPending, flush: debounced.flush }];
|
665
|
+
}
|
666
|
+
|
667
|
+
function useGetNodes({ search, workspaceId, projectIds, branch, pause, }) {
|
668
|
+
const { platformUrl, platformAuthToken } = React.useContext(PlatformContext);
|
669
|
+
const [debounceSearch] = useDebounce(search, 500);
|
670
|
+
return useQuery([
|
671
|
+
...devPortalCacheKeys.searchNodes({ projectIds, branchSlug: branch, workspaceId, search: debounceSearch }),
|
672
|
+
platformUrl,
|
673
|
+
platformAuthToken,
|
674
|
+
], () => getNodes({ workspaceId, projectIds, branchSlug: branch, search: debounceSearch, platformUrl, platformAuthToken }), { enabled: !pause, keepPreviousData: true });
|
675
|
+
}
|
676
|
+
|
677
|
+
function useGetWorkspace({ projectIds }) {
|
678
|
+
const { platformUrl, platformAuthToken } = React.useContext(PlatformContext);
|
679
|
+
return useQuery([...devPortalCacheKeys.searchNodes({ projectIds }), platformUrl, platformAuthToken], () => getWorkspace({ projectIds, platformUrl, platformAuthToken }));
|
680
|
+
}
|
681
|
+
|
682
|
+
export { BranchSelector, DevPortalProvider, NodeContent, ResponseError, Search, StoplightProject, TableOfContents, devPortalCacheKeys, getBranches, getNodeContent, getNodes, getTableOfContents, getWorkspace, useGetBranches, useGetNodeContent, useGetNodes, useGetTableOfContents, useGetWorkspace };
|