@sitecore-jss/sitecore-jss-nextjs 22.6.0-canary.9 → 22.7.0-canary.1
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.
- package/dist/cjs/components/Link.js +1 -0
- package/dist/cjs/components/RichText.js +17 -5
- package/dist/cjs/editing/editing-data-service.js +7 -1
- package/dist/cjs/index.js +3 -2
- package/dist/cjs/middleware/middleware.js +7 -4
- package/dist/cjs/middleware/personalize-middleware.js +1 -1
- package/dist/cjs/middleware/redirects-middleware.js +30 -15
- package/dist/esm/components/Link.js +1 -0
- package/dist/esm/components/RichText.js +15 -3
- package/dist/esm/editing/editing-data-service.js +7 -1
- package/dist/esm/index.js +1 -1
- package/dist/esm/middleware/middleware.js +6 -3
- package/dist/esm/middleware/personalize-middleware.js +2 -2
- package/dist/esm/middleware/redirects-middleware.js +31 -16
- package/package.json +10 -9
- package/types/components/RichText.d.ts +7 -1
- package/types/index.d.ts +1 -1
- package/types/middleware/middleware.d.ts +3 -2
|
@@ -67,6 +67,7 @@ exports.Link = (0, react_1.forwardRef)((props, ref) => {
|
|
|
67
67
|
const isFileUrl = FILE_EXTENSION_MATCHER.test(href);
|
|
68
68
|
// determine if a link is a route or not. File extensions are not routes and should not be pre-fetched.
|
|
69
69
|
if (isMatching && !isFileUrl) {
|
|
70
|
+
delete htmlLinkProps.emptyFieldEditingComponent;
|
|
70
71
|
return (react_1.default.createElement(link_1.default, Object.assign({ href: { pathname: href, query: querystring, hash: anchor }, key: "link", locale: false, title: value.title, target: value.target, className: value.class, prefetch: props.prefetch }, htmlLinkProps, { ref: ref }),
|
|
71
72
|
text,
|
|
72
73
|
children));
|
|
@@ -37,12 +37,12 @@ var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
|
37
37
|
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
38
38
|
};
|
|
39
39
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
40
|
-
exports.RichText = void 0;
|
|
40
|
+
exports.RichText = exports.prefetched = void 0;
|
|
41
41
|
const react_1 = __importStar(require("react"));
|
|
42
42
|
const prop_types_1 = __importDefault(require("prop-types"));
|
|
43
43
|
const router_1 = require("next/router");
|
|
44
44
|
const sitecore_jss_react_1 = require("@sitecore-jss/sitecore-jss-react");
|
|
45
|
-
|
|
45
|
+
exports.prefetched = {};
|
|
46
46
|
const RichText = (props) => {
|
|
47
47
|
const { internalLinksSelector = 'a[href^="/"]', prefetchLinks = true, editable = true } = props, rest = __rest(props, ["internalLinksSelector", "prefetchLinks", "editable"]);
|
|
48
48
|
const hasText = props.field && props.field.value;
|
|
@@ -50,7 +50,7 @@ const RichText = (props) => {
|
|
|
50
50
|
const router = (0, router_1.useRouter)();
|
|
51
51
|
const richTextRef = (0, react_1.useRef)(null);
|
|
52
52
|
(0, react_1.useEffect)(() => {
|
|
53
|
-
// NOT IN
|
|
53
|
+
// NOT IN EDIT MODE
|
|
54
54
|
if (hasText && !isEditing) {
|
|
55
55
|
initializeLinks();
|
|
56
56
|
}
|
|
@@ -71,9 +71,21 @@ const RichText = (props) => {
|
|
|
71
71
|
internalLinks.forEach((link) => {
|
|
72
72
|
if (link.target === '_blank')
|
|
73
73
|
return;
|
|
74
|
-
|
|
74
|
+
const prefetch = () => {
|
|
75
75
|
router.prefetch(link.pathname, undefined, { locale: false });
|
|
76
|
-
prefetched[link.pathname] = true;
|
|
76
|
+
exports.prefetched[link.pathname] = true;
|
|
77
|
+
};
|
|
78
|
+
if (!exports.prefetched[link.pathname] && prefetchLinks !== false) {
|
|
79
|
+
if (prefetchLinks === true) {
|
|
80
|
+
prefetch();
|
|
81
|
+
}
|
|
82
|
+
if (prefetchLinks === 'hover') {
|
|
83
|
+
const mouseOverHandler = () => {
|
|
84
|
+
prefetch();
|
|
85
|
+
link.removeEventListener('mouseover', mouseOverHandler);
|
|
86
|
+
};
|
|
87
|
+
link.addEventListener('mouseover', mouseOverHandler, false);
|
|
88
|
+
}
|
|
77
89
|
}
|
|
78
90
|
link.addEventListener('click', routeHandler, false);
|
|
79
91
|
});
|
|
@@ -108,7 +108,13 @@ class ServerlessEditingDataService {
|
|
|
108
108
|
params,
|
|
109
109
|
};
|
|
110
110
|
sitecore_jss_1.debug.editing('storing editing data for %o: %o', previewData, data);
|
|
111
|
-
return this.dataFetcher
|
|
111
|
+
return this.dataFetcher
|
|
112
|
+
.put(url, data, {
|
|
113
|
+
headers: {
|
|
114
|
+
'Content-Type': 'application/json',
|
|
115
|
+
},
|
|
116
|
+
})
|
|
117
|
+
.then(() => {
|
|
112
118
|
return previewData;
|
|
113
119
|
});
|
|
114
120
|
});
|
package/dist/cjs/index.js
CHANGED
|
@@ -23,8 +23,8 @@ var __importStar = (this && this.__importStar) || function (mod) {
|
|
|
23
23
|
return result;
|
|
24
24
|
};
|
|
25
25
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
26
|
-
exports.
|
|
27
|
-
exports.EditingScripts = exports.withEmptyFieldEditingComponent = exports.withFieldMetadata = exports.withDatasourceCheck = exports.withPlaceholder = exports.withEditorChromes = exports.useSitecoreContext = exports.withSitecoreContext = exports.SitecoreContextReactContext = exports.SitecoreContext = exports.VisitorIdentification = exports.DefaultEmptyFieldEditingComponentText = exports.DefaultEmptyFieldEditingComponentImage = exports.File = exports.getComponentLibraryStylesheetLinks = exports.BYOCComponent = void 0;
|
|
26
|
+
exports.FEaaSComponent = exports.EditFrame = exports.DateField = exports.Text = exports.Image = exports.Form = exports.ComponentBuilder = exports.BYOCWrapper = exports.FEaaSWrapper = exports.NextImage = exports.Placeholder = exports.RichText = exports.Link = exports.useComponentProps = exports.ComponentPropsContext = exports.ComponentPropsReactContext = exports.normalizeSiteRewrite = exports.getSiteRewriteData = exports.getSiteRewrite = exports.GraphQLSiteInfoService = exports.SiteResolver = exports.GraphQLRobotsService = exports.GraphQLErrorPagesService = exports.GraphQLSitemapXmlService = exports.MultisiteGraphQLSitemapService = exports.GraphQLSitemapService = exports.DisconnectedSitemapService = exports.ComponentPropsService = exports.CdpHelper = exports.normalizePersonalizedRewrite = exports.getGroomedVariantIds = exports.getPersonalizedRewriteData = exports.getPersonalizedRewrite = exports.personalizeLayout = exports.RestDictionaryService = exports.GraphQLDictionaryService = exports.trackingApi = exports.mediaApi = exports.EditMode = exports.getContentStylesheetLink = exports.getFieldValue = exports.getChildPlaceholder = exports.RestLayoutService = exports.GraphQLLayoutService = exports.LayoutServicePageState = exports.MemoryCacheClient = exports.debug = exports.enableDebug = exports.NativeDataFetcher = exports.constants = void 0;
|
|
27
|
+
exports.EditingScripts = exports.withEmptyFieldEditingComponent = exports.withFieldMetadata = exports.withDatasourceCheck = exports.withPlaceholder = exports.withEditorChromes = exports.useSitecoreContext = exports.withSitecoreContext = exports.SitecoreContextReactContext = exports.SitecoreContext = exports.VisitorIdentification = exports.DefaultEmptyFieldEditingComponentText = exports.DefaultEmptyFieldEditingComponentImage = exports.File = exports.getComponentLibraryStylesheetLinks = exports.BYOCComponent = exports.fetchFEaaSComponentServerProps = void 0;
|
|
28
28
|
var sitecore_jss_1 = require("@sitecore-jss/sitecore-jss");
|
|
29
29
|
Object.defineProperty(exports, "constants", { enumerable: true, get: function () { return sitecore_jss_1.constants; } });
|
|
30
30
|
Object.defineProperty(exports, "NativeDataFetcher", { enumerable: true, get: function () { return sitecore_jss_1.NativeDataFetcher; } });
|
|
@@ -89,6 +89,7 @@ exports.BYOCWrapper = BYOCWrapper;
|
|
|
89
89
|
var ComponentBuilder_1 = require("./ComponentBuilder");
|
|
90
90
|
Object.defineProperty(exports, "ComponentBuilder", { enumerable: true, get: function () { return ComponentBuilder_1.ComponentBuilder; } });
|
|
91
91
|
var sitecore_jss_react_1 = require("@sitecore-jss/sitecore-jss-react");
|
|
92
|
+
Object.defineProperty(exports, "Form", { enumerable: true, get: function () { return sitecore_jss_react_1.Form; } });
|
|
92
93
|
Object.defineProperty(exports, "Image", { enumerable: true, get: function () { return sitecore_jss_react_1.Image; } });
|
|
93
94
|
Object.defineProperty(exports, "Text", { enumerable: true, get: function () { return sitecore_jss_react_1.Text; } });
|
|
94
95
|
Object.defineProperty(exports, "DateField", { enumerable: true, get: function () { return sitecore_jss_react_1.DateField; } });
|
|
@@ -1,12 +1,12 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.MiddlewareBase = void 0;
|
|
3
|
+
exports.MiddlewareBase = exports.REWRITE_HEADER_NAME = void 0;
|
|
4
4
|
const server_1 = require("next/server");
|
|
5
|
+
exports.REWRITE_HEADER_NAME = 'x-sc-rewrite';
|
|
5
6
|
class MiddlewareBase {
|
|
6
7
|
constructor(config) {
|
|
7
8
|
this.config = config;
|
|
8
9
|
this.SITE_SYMBOL = 'sc_site';
|
|
9
|
-
this.REWRITE_HEADER_NAME = 'x-sc-rewrite';
|
|
10
10
|
this.defaultHostname = config.defaultHostname || 'localhost';
|
|
11
11
|
}
|
|
12
12
|
/**
|
|
@@ -84,14 +84,17 @@ class MiddlewareBase {
|
|
|
84
84
|
* @param {string} rewritePath the destionation path
|
|
85
85
|
* @param {NextRequest} req the current request
|
|
86
86
|
* @param {NextResponse} res the current response
|
|
87
|
+
* @param {boolean} [skipHeader] don't write 'x-sc-rewrite' header
|
|
87
88
|
*/
|
|
88
|
-
rewrite(rewritePath, req, res) {
|
|
89
|
+
rewrite(rewritePath, req, res, skipHeader) {
|
|
89
90
|
// Note an absolute URL is required: https://nextjs.org/docs/messages/middleware-relative-urls
|
|
90
91
|
const rewriteUrl = req.nextUrl.clone();
|
|
91
92
|
rewriteUrl.pathname = rewritePath;
|
|
92
93
|
const response = server_1.NextResponse.rewrite(rewriteUrl, res);
|
|
93
94
|
// Share rewrite path with following executed middlewares
|
|
94
|
-
|
|
95
|
+
if (!skipHeader) {
|
|
96
|
+
response.headers.set(exports.REWRITE_HEADER_NAME, rewritePath);
|
|
97
|
+
}
|
|
95
98
|
return response;
|
|
96
99
|
}
|
|
97
100
|
}
|
|
@@ -102,7 +102,7 @@ class PersonalizeMiddleware extends middleware_1.MiddlewareBase {
|
|
|
102
102
|
return response;
|
|
103
103
|
}
|
|
104
104
|
// Path can be rewritten by previously executed middleware
|
|
105
|
-
const basePath = (res === null || res === void 0 ? void 0 : res.headers.get(
|
|
105
|
+
const basePath = (res === null || res === void 0 ? void 0 : res.headers.get(middleware_1.REWRITE_HEADER_NAME)) || pathname;
|
|
106
106
|
// Rewrite to persononalized path
|
|
107
107
|
const rewritePath = (0, personalize_1.getPersonalizedRewrite)(basePath, identifiedVariantIds);
|
|
108
108
|
response = this.rewrite(rewritePath, req, response);
|
|
@@ -62,23 +62,35 @@ class RedirectsMiddleware extends middleware_1.MiddlewareBase {
|
|
|
62
62
|
*/
|
|
63
63
|
getExistsRedirect(req, siteName) {
|
|
64
64
|
return __awaiter(this, void 0, void 0, function* () {
|
|
65
|
-
const { pathname:
|
|
66
|
-
const
|
|
65
|
+
const { pathname: incomingURL, search: incomingQS = '' } = this.normalizeUrl(req.nextUrl.clone());
|
|
66
|
+
const locale = this.getLanguage(req);
|
|
67
|
+
const normalizedPath = incomingURL.replace(/\/*$/gi, '');
|
|
67
68
|
const redirects = yield this.redirectsService.fetchRedirects(siteName);
|
|
68
69
|
const language = this.getLanguage(req);
|
|
69
70
|
const modifyRedirects = structuredClone(redirects);
|
|
70
71
|
let matchedQueryString;
|
|
72
|
+
const localePath = `/${locale.toLowerCase()}${normalizedPath}`;
|
|
71
73
|
return modifyRedirects.length
|
|
72
74
|
? modifyRedirects.find((redirect) => {
|
|
73
|
-
|
|
75
|
+
// process static URL (non-regex) rules
|
|
74
76
|
if ((0, utils_1.isRegexOrUrl)(redirect.pattern) === 'url') {
|
|
75
|
-
const
|
|
77
|
+
const urlArray = redirect.pattern.endsWith('/')
|
|
76
78
|
? redirect.pattern.slice(0, -1).split('?')
|
|
77
79
|
: redirect.pattern.split('?');
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
80
|
+
const patternQS = urlArray[1];
|
|
81
|
+
let patternPath = urlArray[0];
|
|
82
|
+
// nextjs routes are case-sensitive, but locales should be compared case-insensitively
|
|
83
|
+
const patternParts = patternPath.split('/');
|
|
84
|
+
const maybeLocale = patternParts[1].toLowerCase();
|
|
85
|
+
// case insensitive lookup of locales
|
|
86
|
+
if (new RegExp(this.locales.join('|'), 'i').test(maybeLocale)) {
|
|
87
|
+
patternPath = patternPath.replace(`/${patternParts[1]}`, `/${maybeLocale}`);
|
|
88
|
+
}
|
|
89
|
+
return ((patternPath === localePath || patternPath === normalizedPath) &&
|
|
90
|
+
(!patternQS ||
|
|
91
|
+
(0, utils_1.areURLSearchParamsEqual)(new URLSearchParams(patternQS), new URLSearchParams(incomingQS))));
|
|
81
92
|
}
|
|
93
|
+
// process regex rules
|
|
82
94
|
// Modify the redirect pattern to ignore the language prefix in the path
|
|
83
95
|
// And escapes non-special "?" characters in a string or regex.
|
|
84
96
|
redirect.pattern = (0, utils_1.escapeNonSpecialQuestionMarks)(redirect.pattern.replace(new RegExp(`^[^]?/${language}/`, 'gi'), ''));
|
|
@@ -88,16 +100,17 @@ class RedirectsMiddleware extends middleware_1.MiddlewareBase {
|
|
|
88
100
|
.replace(/^\^\/|\/\$$/g, '') // Removes unnecessary start (^) and end ($) anchors
|
|
89
101
|
.replace(/^\^|\$$/g, '') // Further cleans up anchors
|
|
90
102
|
.replace(/\$\/gi$/g, '')}[\/]?$/i`; // Ensures the pattern allows an optional trailing slash
|
|
103
|
+
// Redirect pattern matches the full incoming URL with query string present
|
|
91
104
|
matchedQueryString = [
|
|
92
|
-
(0, regex_parser_1.default)(redirect.pattern).test(`${
|
|
93
|
-
(0, regex_parser_1.default)(redirect.pattern).test(
|
|
105
|
+
(0, regex_parser_1.default)(redirect.pattern).test(`${localePath}${incomingQS}`),
|
|
106
|
+
(0, regex_parser_1.default)(redirect.pattern).test(`${normalizedPath}${incomingQS}`),
|
|
94
107
|
].some(Boolean)
|
|
95
|
-
?
|
|
108
|
+
? incomingQS
|
|
96
109
|
: undefined;
|
|
97
110
|
// Save the matched query string (if found) into the redirect object
|
|
98
111
|
redirect.matchedQueryString = matchedQueryString || '';
|
|
99
|
-
return (!!((0, regex_parser_1.default)(redirect.pattern).test(
|
|
100
|
-
(0, regex_parser_1.default)(redirect.pattern).test(
|
|
112
|
+
return (!!((0, regex_parser_1.default)(redirect.pattern).test(`/${req.nextUrl.locale}${incomingURL}`) ||
|
|
113
|
+
(0, regex_parser_1.default)(redirect.pattern).test(incomingURL) ||
|
|
101
114
|
matchedQueryString) && (redirect.locale ? redirect.locale.toLowerCase() === locale.toLowerCase() : true));
|
|
102
115
|
})
|
|
103
116
|
: undefined;
|
|
@@ -145,6 +158,7 @@ class RedirectsMiddleware extends middleware_1.MiddlewareBase {
|
|
|
145
158
|
sitecore_jss_1.debug.redirects('skipped (redirect does not exist)');
|
|
146
159
|
return response;
|
|
147
160
|
}
|
|
161
|
+
sitecore_jss_1.debug.redirects('Matched redirect rule: %o', { existsRedirect });
|
|
148
162
|
// Find context site language and replace token
|
|
149
163
|
if (REGEXP_CONTEXT_SITE_LANG.test(existsRedirect.target) &&
|
|
150
164
|
!(REGEXP_ABSOLUTE_URL.test(existsRedirect.target) &&
|
|
@@ -191,7 +205,7 @@ class RedirectsMiddleware extends middleware_1.MiddlewareBase {
|
|
|
191
205
|
return this.createRedirectResponse(url, response, 302, 'Found');
|
|
192
206
|
}
|
|
193
207
|
case site_1.REDIRECT_TYPE_SERVER_TRANSFER: {
|
|
194
|
-
return this.rewrite(url.href, req, response);
|
|
208
|
+
return this.rewrite(url.href, req, response, true);
|
|
195
209
|
}
|
|
196
210
|
default:
|
|
197
211
|
return response;
|
|
@@ -241,9 +255,9 @@ class RedirectsMiddleware extends middleware_1.MiddlewareBase {
|
|
|
241
255
|
return false;
|
|
242
256
|
})
|
|
243
257
|
.join('&');
|
|
244
|
-
const newUrl = new URL(`${url.pathname}?${newQueryString}`, url.origin);
|
|
258
|
+
const newUrl = new URL(`${url.pathname.toLowerCase()}?${newQueryString}`, url.origin);
|
|
245
259
|
url.search = newUrl.search;
|
|
246
|
-
url.pathname = newUrl.pathname;
|
|
260
|
+
url.pathname = newUrl.pathname.toLowerCase();
|
|
247
261
|
url.href = newUrl.href;
|
|
248
262
|
return url;
|
|
249
263
|
}
|
|
@@ -264,6 +278,7 @@ class RedirectsMiddleware extends middleware_1.MiddlewareBase {
|
|
|
264
278
|
if (res === null || res === void 0 ? void 0 : res.headers) {
|
|
265
279
|
redirect.headers.delete('x-middleware-next');
|
|
266
280
|
redirect.headers.delete('x-middleware-rewrite');
|
|
281
|
+
redirect.headers.delete(middleware_1.REWRITE_HEADER_NAME);
|
|
267
282
|
}
|
|
268
283
|
return redirect;
|
|
269
284
|
}
|
|
@@ -38,6 +38,7 @@ export const Link = forwardRef((props, ref) => {
|
|
|
38
38
|
const isFileUrl = FILE_EXTENSION_MATCHER.test(href);
|
|
39
39
|
// determine if a link is a route or not. File extensions are not routes and should not be pre-fetched.
|
|
40
40
|
if (isMatching && !isFileUrl) {
|
|
41
|
+
delete htmlLinkProps.emptyFieldEditingComponent;
|
|
41
42
|
return (React.createElement(NextLink, Object.assign({ href: { pathname: href, query: querystring, hash: anchor }, key: "link", locale: false, title: value.title, target: value.target, className: value.class, prefetch: props.prefetch }, htmlLinkProps, { ref: ref }),
|
|
42
43
|
text,
|
|
43
44
|
children));
|
|
@@ -13,7 +13,7 @@ import React, { useEffect, useRef } from 'react';
|
|
|
13
13
|
import PropTypes from 'prop-types';
|
|
14
14
|
import { useRouter } from 'next/router';
|
|
15
15
|
import { RichText as ReactRichText, RichTextPropTypes, } from '@sitecore-jss/sitecore-jss-react';
|
|
16
|
-
const prefetched = {};
|
|
16
|
+
export const prefetched = {};
|
|
17
17
|
export const RichText = (props) => {
|
|
18
18
|
const { internalLinksSelector = 'a[href^="/"]', prefetchLinks = true, editable = true } = props, rest = __rest(props, ["internalLinksSelector", "prefetchLinks", "editable"]);
|
|
19
19
|
const hasText = props.field && props.field.value;
|
|
@@ -21,7 +21,7 @@ export const RichText = (props) => {
|
|
|
21
21
|
const router = useRouter();
|
|
22
22
|
const richTextRef = useRef(null);
|
|
23
23
|
useEffect(() => {
|
|
24
|
-
// NOT IN
|
|
24
|
+
// NOT IN EDIT MODE
|
|
25
25
|
if (hasText && !isEditing) {
|
|
26
26
|
initializeLinks();
|
|
27
27
|
}
|
|
@@ -42,9 +42,21 @@ export const RichText = (props) => {
|
|
|
42
42
|
internalLinks.forEach((link) => {
|
|
43
43
|
if (link.target === '_blank')
|
|
44
44
|
return;
|
|
45
|
-
|
|
45
|
+
const prefetch = () => {
|
|
46
46
|
router.prefetch(link.pathname, undefined, { locale: false });
|
|
47
47
|
prefetched[link.pathname] = true;
|
|
48
|
+
};
|
|
49
|
+
if (!prefetched[link.pathname] && prefetchLinks !== false) {
|
|
50
|
+
if (prefetchLinks === true) {
|
|
51
|
+
prefetch();
|
|
52
|
+
}
|
|
53
|
+
if (prefetchLinks === 'hover') {
|
|
54
|
+
const mouseOverHandler = () => {
|
|
55
|
+
prefetch();
|
|
56
|
+
link.removeEventListener('mouseover', mouseOverHandler);
|
|
57
|
+
};
|
|
58
|
+
link.addEventListener('mouseover', mouseOverHandler, false);
|
|
59
|
+
}
|
|
48
60
|
}
|
|
49
61
|
link.addEventListener('click', routeHandler, false);
|
|
50
62
|
});
|
|
@@ -103,7 +103,13 @@ export class ServerlessEditingDataService {
|
|
|
103
103
|
params,
|
|
104
104
|
};
|
|
105
105
|
debug.editing('storing editing data for %o: %o', previewData, data);
|
|
106
|
-
return this.dataFetcher
|
|
106
|
+
return this.dataFetcher
|
|
107
|
+
.put(url, data, {
|
|
108
|
+
headers: {
|
|
109
|
+
'Content-Type': 'application/json',
|
|
110
|
+
},
|
|
111
|
+
})
|
|
112
|
+
.then(() => {
|
|
107
113
|
return previewData;
|
|
108
114
|
});
|
|
109
115
|
});
|
package/dist/esm/index.js
CHANGED
|
@@ -19,4 +19,4 @@ import * as BYOCWrapper from './components/BYOCWrapper';
|
|
|
19
19
|
export { FEaaSWrapper };
|
|
20
20
|
export { BYOCWrapper };
|
|
21
21
|
export { ComponentBuilder } from './ComponentBuilder';
|
|
22
|
-
export { Image, Text, DateField, EditFrame, FEaaSComponent, fetchFEaaSComponentServerProps, BYOCComponent, getComponentLibraryStylesheetLinks, File, DefaultEmptyFieldEditingComponentImage, DefaultEmptyFieldEditingComponentText, VisitorIdentification, SitecoreContext, SitecoreContextReactContext, withSitecoreContext, useSitecoreContext, withEditorChromes, withPlaceholder, withDatasourceCheck, withFieldMetadata, withEmptyFieldEditingComponent, EditingScripts, } from '@sitecore-jss/sitecore-jss-react';
|
|
22
|
+
export { Form, Image, Text, DateField, EditFrame, FEaaSComponent, fetchFEaaSComponentServerProps, BYOCComponent, getComponentLibraryStylesheetLinks, File, DefaultEmptyFieldEditingComponentImage, DefaultEmptyFieldEditingComponentText, VisitorIdentification, SitecoreContext, SitecoreContextReactContext, withSitecoreContext, useSitecoreContext, withEditorChromes, withPlaceholder, withDatasourceCheck, withFieldMetadata, withEmptyFieldEditingComponent, EditingScripts, } from '@sitecore-jss/sitecore-jss-react';
|
|
@@ -1,9 +1,9 @@
|
|
|
1
1
|
import { NextResponse } from 'next/server';
|
|
2
|
+
export const REWRITE_HEADER_NAME = 'x-sc-rewrite';
|
|
2
3
|
export class MiddlewareBase {
|
|
3
4
|
constructor(config) {
|
|
4
5
|
this.config = config;
|
|
5
6
|
this.SITE_SYMBOL = 'sc_site';
|
|
6
|
-
this.REWRITE_HEADER_NAME = 'x-sc-rewrite';
|
|
7
7
|
this.defaultHostname = config.defaultHostname || 'localhost';
|
|
8
8
|
}
|
|
9
9
|
/**
|
|
@@ -81,14 +81,17 @@ export class MiddlewareBase {
|
|
|
81
81
|
* @param {string} rewritePath the destionation path
|
|
82
82
|
* @param {NextRequest} req the current request
|
|
83
83
|
* @param {NextResponse} res the current response
|
|
84
|
+
* @param {boolean} [skipHeader] don't write 'x-sc-rewrite' header
|
|
84
85
|
*/
|
|
85
|
-
rewrite(rewritePath, req, res) {
|
|
86
|
+
rewrite(rewritePath, req, res, skipHeader) {
|
|
86
87
|
// Note an absolute URL is required: https://nextjs.org/docs/messages/middleware-relative-urls
|
|
87
88
|
const rewriteUrl = req.nextUrl.clone();
|
|
88
89
|
rewriteUrl.pathname = rewritePath;
|
|
89
90
|
const response = NextResponse.rewrite(rewriteUrl, res);
|
|
90
91
|
// Share rewrite path with following executed middlewares
|
|
91
|
-
|
|
92
|
+
if (!skipHeader) {
|
|
93
|
+
response.headers.set(REWRITE_HEADER_NAME, rewritePath);
|
|
94
|
+
}
|
|
92
95
|
return response;
|
|
93
96
|
}
|
|
94
97
|
}
|
|
@@ -10,7 +10,7 @@ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, ge
|
|
|
10
10
|
import { NextResponse } from 'next/server';
|
|
11
11
|
import { GraphQLPersonalizeService, getPersonalizedRewrite, CdpHelper, DEFAULT_VARIANT, } from '@sitecore-jss/sitecore-jss/personalize';
|
|
12
12
|
import { debug } from '@sitecore-jss/sitecore-jss';
|
|
13
|
-
import { MiddlewareBase } from './middleware';
|
|
13
|
+
import { MiddlewareBase, REWRITE_HEADER_NAME } from './middleware';
|
|
14
14
|
import { CloudSDK } from '@sitecore-cloudsdk/core/server';
|
|
15
15
|
import { personalize } from '@sitecore-cloudsdk/personalize/server';
|
|
16
16
|
/**
|
|
@@ -99,7 +99,7 @@ export class PersonalizeMiddleware extends MiddlewareBase {
|
|
|
99
99
|
return response;
|
|
100
100
|
}
|
|
101
101
|
// Path can be rewritten by previously executed middleware
|
|
102
|
-
const basePath = (res === null || res === void 0 ? void 0 : res.headers.get(
|
|
102
|
+
const basePath = (res === null || res === void 0 ? void 0 : res.headers.get(REWRITE_HEADER_NAME)) || pathname;
|
|
103
103
|
// Rewrite to persononalized path
|
|
104
104
|
const rewritePath = getPersonalizedRewrite(basePath, identifiedVariantIds);
|
|
105
105
|
response = this.rewrite(rewritePath, req, response);
|
|
@@ -12,7 +12,7 @@ import { GraphQLRedirectsService, REDIRECT_TYPE_301, REDIRECT_TYPE_302, REDIRECT
|
|
|
12
12
|
import { areURLSearchParamsEqual, escapeNonSpecialQuestionMarks, isRegexOrUrl, mergeURLSearchParams, } from '@sitecore-jss/sitecore-jss/utils';
|
|
13
13
|
import { NextResponse } from 'next/server';
|
|
14
14
|
import regexParser from 'regex-parser';
|
|
15
|
-
import { MiddlewareBase } from './middleware';
|
|
15
|
+
import { MiddlewareBase, REWRITE_HEADER_NAME } from './middleware';
|
|
16
16
|
const REGEXP_CONTEXT_SITE_LANG = new RegExp(/\$siteLang/, 'i');
|
|
17
17
|
const REGEXP_ABSOLUTE_URL = new RegExp('^(?:[a-z]+:)?//', 'i');
|
|
18
18
|
/**
|
|
@@ -56,23 +56,35 @@ export class RedirectsMiddleware extends MiddlewareBase {
|
|
|
56
56
|
*/
|
|
57
57
|
getExistsRedirect(req, siteName) {
|
|
58
58
|
return __awaiter(this, void 0, void 0, function* () {
|
|
59
|
-
const { pathname:
|
|
60
|
-
const
|
|
59
|
+
const { pathname: incomingURL, search: incomingQS = '' } = this.normalizeUrl(req.nextUrl.clone());
|
|
60
|
+
const locale = this.getLanguage(req);
|
|
61
|
+
const normalizedPath = incomingURL.replace(/\/*$/gi, '');
|
|
61
62
|
const redirects = yield this.redirectsService.fetchRedirects(siteName);
|
|
62
63
|
const language = this.getLanguage(req);
|
|
63
64
|
const modifyRedirects = structuredClone(redirects);
|
|
64
65
|
let matchedQueryString;
|
|
66
|
+
const localePath = `/${locale.toLowerCase()}${normalizedPath}`;
|
|
65
67
|
return modifyRedirects.length
|
|
66
68
|
? modifyRedirects.find((redirect) => {
|
|
67
|
-
|
|
69
|
+
// process static URL (non-regex) rules
|
|
68
70
|
if (isRegexOrUrl(redirect.pattern) === 'url') {
|
|
69
|
-
const
|
|
71
|
+
const urlArray = redirect.pattern.endsWith('/')
|
|
70
72
|
? redirect.pattern.slice(0, -1).split('?')
|
|
71
73
|
: redirect.pattern.split('?');
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
74
|
+
const patternQS = urlArray[1];
|
|
75
|
+
let patternPath = urlArray[0];
|
|
76
|
+
// nextjs routes are case-sensitive, but locales should be compared case-insensitively
|
|
77
|
+
const patternParts = patternPath.split('/');
|
|
78
|
+
const maybeLocale = patternParts[1].toLowerCase();
|
|
79
|
+
// case insensitive lookup of locales
|
|
80
|
+
if (new RegExp(this.locales.join('|'), 'i').test(maybeLocale)) {
|
|
81
|
+
patternPath = patternPath.replace(`/${patternParts[1]}`, `/${maybeLocale}`);
|
|
82
|
+
}
|
|
83
|
+
return ((patternPath === localePath || patternPath === normalizedPath) &&
|
|
84
|
+
(!patternQS ||
|
|
85
|
+
areURLSearchParamsEqual(new URLSearchParams(patternQS), new URLSearchParams(incomingQS))));
|
|
75
86
|
}
|
|
87
|
+
// process regex rules
|
|
76
88
|
// Modify the redirect pattern to ignore the language prefix in the path
|
|
77
89
|
// And escapes non-special "?" characters in a string or regex.
|
|
78
90
|
redirect.pattern = escapeNonSpecialQuestionMarks(redirect.pattern.replace(new RegExp(`^[^]?/${language}/`, 'gi'), ''));
|
|
@@ -82,16 +94,17 @@ export class RedirectsMiddleware extends MiddlewareBase {
|
|
|
82
94
|
.replace(/^\^\/|\/\$$/g, '') // Removes unnecessary start (^) and end ($) anchors
|
|
83
95
|
.replace(/^\^|\$$/g, '') // Further cleans up anchors
|
|
84
96
|
.replace(/\$\/gi$/g, '')}[\/]?$/i`; // Ensures the pattern allows an optional trailing slash
|
|
97
|
+
// Redirect pattern matches the full incoming URL with query string present
|
|
85
98
|
matchedQueryString = [
|
|
86
|
-
regexParser(redirect.pattern).test(`${
|
|
87
|
-
regexParser(redirect.pattern).test(
|
|
99
|
+
regexParser(redirect.pattern).test(`${localePath}${incomingQS}`),
|
|
100
|
+
regexParser(redirect.pattern).test(`${normalizedPath}${incomingQS}`),
|
|
88
101
|
].some(Boolean)
|
|
89
|
-
?
|
|
102
|
+
? incomingQS
|
|
90
103
|
: undefined;
|
|
91
104
|
// Save the matched query string (if found) into the redirect object
|
|
92
105
|
redirect.matchedQueryString = matchedQueryString || '';
|
|
93
|
-
return (!!(regexParser(redirect.pattern).test(
|
|
94
|
-
regexParser(redirect.pattern).test(
|
|
106
|
+
return (!!(regexParser(redirect.pattern).test(`/${req.nextUrl.locale}${incomingURL}`) ||
|
|
107
|
+
regexParser(redirect.pattern).test(incomingURL) ||
|
|
95
108
|
matchedQueryString) && (redirect.locale ? redirect.locale.toLowerCase() === locale.toLowerCase() : true));
|
|
96
109
|
})
|
|
97
110
|
: undefined;
|
|
@@ -139,6 +152,7 @@ export class RedirectsMiddleware extends MiddlewareBase {
|
|
|
139
152
|
debug.redirects('skipped (redirect does not exist)');
|
|
140
153
|
return response;
|
|
141
154
|
}
|
|
155
|
+
debug.redirects('Matched redirect rule: %o', { existsRedirect });
|
|
142
156
|
// Find context site language and replace token
|
|
143
157
|
if (REGEXP_CONTEXT_SITE_LANG.test(existsRedirect.target) &&
|
|
144
158
|
!(REGEXP_ABSOLUTE_URL.test(existsRedirect.target) &&
|
|
@@ -185,7 +199,7 @@ export class RedirectsMiddleware extends MiddlewareBase {
|
|
|
185
199
|
return this.createRedirectResponse(url, response, 302, 'Found');
|
|
186
200
|
}
|
|
187
201
|
case REDIRECT_TYPE_SERVER_TRANSFER: {
|
|
188
|
-
return this.rewrite(url.href, req, response);
|
|
202
|
+
return this.rewrite(url.href, req, response, true);
|
|
189
203
|
}
|
|
190
204
|
default:
|
|
191
205
|
return response;
|
|
@@ -235,9 +249,9 @@ export class RedirectsMiddleware extends MiddlewareBase {
|
|
|
235
249
|
return false;
|
|
236
250
|
})
|
|
237
251
|
.join('&');
|
|
238
|
-
const newUrl = new URL(`${url.pathname}?${newQueryString}`, url.origin);
|
|
252
|
+
const newUrl = new URL(`${url.pathname.toLowerCase()}?${newQueryString}`, url.origin);
|
|
239
253
|
url.search = newUrl.search;
|
|
240
|
-
url.pathname = newUrl.pathname;
|
|
254
|
+
url.pathname = newUrl.pathname.toLowerCase();
|
|
241
255
|
url.href = newUrl.href;
|
|
242
256
|
return url;
|
|
243
257
|
}
|
|
@@ -258,6 +272,7 @@ export class RedirectsMiddleware extends MiddlewareBase {
|
|
|
258
272
|
if (res === null || res === void 0 ? void 0 : res.headers) {
|
|
259
273
|
redirect.headers.delete('x-middleware-next');
|
|
260
274
|
redirect.headers.delete('x-middleware-rewrite');
|
|
275
|
+
redirect.headers.delete(REWRITE_HEADER_NAME);
|
|
261
276
|
}
|
|
262
277
|
return redirect;
|
|
263
278
|
}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@sitecore-jss/sitecore-jss-nextjs",
|
|
3
|
-
"version": "22.
|
|
3
|
+
"version": "22.7.0-canary.1",
|
|
4
4
|
"main": "dist/cjs/index.js",
|
|
5
5
|
"module": "dist/esm/index.js",
|
|
6
6
|
"sideEffects": false,
|
|
@@ -29,8 +29,8 @@
|
|
|
29
29
|
"url": "https://github.com/sitecore/jss/issues"
|
|
30
30
|
},
|
|
31
31
|
"devDependencies": {
|
|
32
|
-
"@sitecore-cloudsdk/core": "^0.
|
|
33
|
-
"@sitecore-cloudsdk/personalize": "^0.
|
|
32
|
+
"@sitecore-cloudsdk/core": "^0.5.0",
|
|
33
|
+
"@sitecore-cloudsdk/personalize": "^0.5.0",
|
|
34
34
|
"@types/chai": "^4.3.4",
|
|
35
35
|
"@types/chai-as-promised": "^7.1.5",
|
|
36
36
|
"@types/chai-string": "^1.4.2",
|
|
@@ -66,16 +66,17 @@
|
|
|
66
66
|
"typescript": "~5.4.0"
|
|
67
67
|
},
|
|
68
68
|
"peerDependencies": {
|
|
69
|
-
"@sitecore-cloudsdk/core": "^0.
|
|
70
|
-
"@sitecore-cloudsdk/
|
|
69
|
+
"@sitecore-cloudsdk/core": "^0.5.0",
|
|
70
|
+
"@sitecore-cloudsdk/events": "^0.5.0",
|
|
71
|
+
"@sitecore-cloudsdk/personalize": "^0.5.0",
|
|
71
72
|
"next": "^14.2.18",
|
|
72
73
|
"react": "^18.2.0",
|
|
73
74
|
"react-dom": "^18.2.0"
|
|
74
75
|
},
|
|
75
76
|
"dependencies": {
|
|
76
|
-
"@sitecore-jss/sitecore-jss": "
|
|
77
|
-
"@sitecore-jss/sitecore-jss-dev-tools": "
|
|
78
|
-
"@sitecore-jss/sitecore-jss-react": "
|
|
77
|
+
"@sitecore-jss/sitecore-jss": "22.7.0-canary.1",
|
|
78
|
+
"@sitecore-jss/sitecore-jss-dev-tools": "22.7.0-canary.1",
|
|
79
|
+
"@sitecore-jss/sitecore-jss-react": "22.7.0-canary.1",
|
|
79
80
|
"@vercel/kv": "^0.2.1",
|
|
80
81
|
"prop-types": "^15.8.1",
|
|
81
82
|
"regex-parser": "^2.2.11",
|
|
@@ -83,7 +84,7 @@
|
|
|
83
84
|
},
|
|
84
85
|
"description": "",
|
|
85
86
|
"types": "types/index.d.ts",
|
|
86
|
-
"gitHead": "
|
|
87
|
+
"gitHead": "31b27434eca1cdfcc402ad567b7ecff70670f675",
|
|
87
88
|
"files": [
|
|
88
89
|
"dist",
|
|
89
90
|
"types",
|
|
@@ -11,9 +11,15 @@ export type RichTextProps = ReactRichTextProps & {
|
|
|
11
11
|
/**
|
|
12
12
|
* Controls the prefetch of internal links. This can be beneficial if you have RichText fields
|
|
13
13
|
* with large numbers of internal links in them.
|
|
14
|
+
* - `true` (default): The full route & its data will be prefetched.
|
|
15
|
+
* - `hover`: Prefetching will happen on hover.
|
|
16
|
+
* - `false`: Prefetching will not happen.
|
|
14
17
|
* @default true
|
|
15
18
|
*/
|
|
16
|
-
prefetchLinks?: boolean;
|
|
19
|
+
prefetchLinks?: boolean | 'hover';
|
|
20
|
+
};
|
|
21
|
+
export declare const prefetched: {
|
|
22
|
+
[cacheKey: string]: boolean;
|
|
17
23
|
};
|
|
18
24
|
export declare const RichText: {
|
|
19
25
|
(props: RichTextProps): JSX.Element;
|
package/types/index.d.ts
CHANGED
|
@@ -22,4 +22,4 @@ import * as BYOCWrapper from './components/BYOCWrapper';
|
|
|
22
22
|
export { FEaaSWrapper };
|
|
23
23
|
export { BYOCWrapper };
|
|
24
24
|
export { ComponentBuilder, ComponentBuilderConfig } from './ComponentBuilder';
|
|
25
|
-
export { ComponentFactory, Image, ImageField, ImageFieldValue, ImageProps, LinkField, LinkFieldValue, Text, TextField, DateField, EditFrame, FEaaSComponent, FEaaSComponentProps, FEaaSComponentParams, fetchFEaaSComponentServerProps, BYOCComponentParams, BYOCComponent, BYOCComponentProps, getComponentLibraryStylesheetLinks, File, FileField, RichTextField, DefaultEmptyFieldEditingComponentImage, DefaultEmptyFieldEditingComponentText, VisitorIdentification, PlaceholderComponentProps, SitecoreContext, SitecoreContextState, SitecoreContextValue, SitecoreContextReactContext, withSitecoreContext, useSitecoreContext, withEditorChromes, withPlaceholder, withDatasourceCheck, ImageSizeParameters, WithSitecoreContextOptions, WithSitecoreContextProps, WithSitecoreContextHocProps, withFieldMetadata, withEmptyFieldEditingComponent, EditingScripts, } from '@sitecore-jss/sitecore-jss-react';
|
|
25
|
+
export { ComponentFactory, Form, Image, ImageField, ImageFieldValue, ImageProps, LinkField, LinkFieldValue, Text, TextField, DateField, EditFrame, FEaaSComponent, FEaaSComponentProps, FEaaSComponentParams, fetchFEaaSComponentServerProps, BYOCComponentParams, BYOCComponent, BYOCComponentProps, getComponentLibraryStylesheetLinks, File, FileField, RichTextField, DefaultEmptyFieldEditingComponentImage, DefaultEmptyFieldEditingComponentText, VisitorIdentification, PlaceholderComponentProps, SitecoreContext, SitecoreContextState, SitecoreContextValue, SitecoreContextReactContext, withSitecoreContext, useSitecoreContext, withEditorChromes, withPlaceholder, withDatasourceCheck, ImageSizeParameters, WithSitecoreContextOptions, WithSitecoreContextProps, WithSitecoreContextHocProps, withFieldMetadata, withEmptyFieldEditingComponent, EditingScripts, } from '@sitecore-jss/sitecore-jss-react';
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { SiteInfo, SiteResolver } from '@sitecore-jss/sitecore-jss/site';
|
|
2
2
|
import { NextRequest, NextResponse } from 'next/server';
|
|
3
|
+
export declare const REWRITE_HEADER_NAME = "x-sc-rewrite";
|
|
3
4
|
export type MiddlewareBaseConfig = {
|
|
4
5
|
/**
|
|
5
6
|
* function, determines if middleware should be turned off, based on cookie, header, or other considerations
|
|
@@ -28,7 +29,6 @@ export type MiddlewareBaseConfig = {
|
|
|
28
29
|
export declare abstract class MiddlewareBase {
|
|
29
30
|
protected config: MiddlewareBaseConfig;
|
|
30
31
|
protected SITE_SYMBOL: string;
|
|
31
|
-
protected REWRITE_HEADER_NAME: string;
|
|
32
32
|
protected defaultHostname: string;
|
|
33
33
|
constructor(config: MiddlewareBaseConfig);
|
|
34
34
|
/**
|
|
@@ -77,6 +77,7 @@ export declare abstract class MiddlewareBase {
|
|
|
77
77
|
* @param {string} rewritePath the destionation path
|
|
78
78
|
* @param {NextRequest} req the current request
|
|
79
79
|
* @param {NextResponse} res the current response
|
|
80
|
+
* @param {boolean} [skipHeader] don't write 'x-sc-rewrite' header
|
|
80
81
|
*/
|
|
81
|
-
protected rewrite(rewritePath: string, req: NextRequest, res: NextResponse): NextResponse;
|
|
82
|
+
protected rewrite(rewritePath: string, req: NextRequest, res: NextResponse, skipHeader?: boolean): NextResponse;
|
|
82
83
|
}
|