@ionic/react-router 8.8.4-dev.11775078622.1402ffa2 → 8.8.4-nightly.20260403

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/index.js CHANGED
@@ -1,1301 +1,176 @@
1
1
  import { __rest } from 'tslib';
2
- import React, { useRef, useState, useEffect, useCallback } from 'react';
3
- import { Route, matchPath as matchPath$1, Routes, Navigate, UNSAFE_RouteContext, matchRoutes, useLocation, useNavigate, BrowserRouter, useNavigationType, HashRouter } from 'react-router-dom';
4
- import { IonRoute, ViewStacks, generateId, ViewLifeCycleManager, StackContext, RouteManagerContext, getConfig, LocationHistory, NavManager } from '@ionic/react';
5
- import { MemoryRouter, useLocation as useLocation$1, useNavigationType as useNavigationType$1 } from 'react-router';
2
+ import { createBrowserHistory, createHashHistory } from 'history';
3
+ import React from 'react';
4
+ import { withRouter, Router } from 'react-router-dom';
5
+ import { ViewStacks, generateId, IonRoute, ViewLifeCycleManager, StackContext, RouteManagerContext, getConfig, LocationHistory, NavManager } from '@ionic/react';
6
+ import { Route, matchPath as matchPath$1, Router as Router$1 } from 'react-router';
6
7
 
7
- const IonRouteInner = ({ path, index, caseSensitive, element }) => {
8
- return React.createElement(Route, { path: path, index: index, caseSensitive: caseSensitive, element: element });
9
- };
10
-
11
- /**
12
- * The matchPath function is used only for matching paths, not rendering components or elements.
13
- * @see https://reactrouter.com/v6/utils/match-path
14
- */
15
- const matchPath = ({ pathname, componentProps }) => {
16
- var _a, _b;
17
- const { path, index } = componentProps, restProps = __rest(componentProps, ["path", "index"]);
18
- // Handle index routes - they match when pathname is empty or just "/"
19
- if (index && !path) {
20
- if (pathname === '' || pathname === '/') {
21
- return {
22
- params: {},
23
- pathname: pathname,
24
- pathnameBase: pathname || '/',
25
- pattern: {
26
- path: '',
27
- caseSensitive: false,
28
- end: true,
29
- },
30
- };
31
- }
32
- return null;
33
- }
34
- // Handle empty path routes - they match when pathname is also empty or just "/"
35
- if (path === '' || path === undefined) {
36
- if (pathname === '' || pathname === '/') {
37
- return {
38
- params: {},
39
- pathname: pathname,
40
- pathnameBase: pathname || '/',
41
- pattern: {
42
- path: '',
43
- caseSensitive: (_a = restProps.caseSensitive) !== null && _a !== void 0 ? _a : false,
44
- end: (_b = restProps.end) !== null && _b !== void 0 ? _b : true,
45
- },
46
- };
47
- }
48
- return null;
49
- }
50
- // For relative paths (don't start with '/'), normalize both path and pathname for matching
51
- if (!path.startsWith('/')) {
52
- const matchOptions = Object.assign({ path: `/${path}` }, restProps);
53
- if ((matchOptions === null || matchOptions === void 0 ? void 0 : matchOptions.end) === undefined) {
54
- matchOptions.end = !path.endsWith('*');
55
- }
56
- const normalizedPathname = pathname.startsWith('/') ? pathname : `/${pathname}`;
57
- const match = matchPath$1(matchOptions, normalizedPathname);
58
- if (match) {
59
- // Adjust the match to remove the leading '/' we added
60
- return Object.assign(Object.assign({}, match), { pathname: pathname, pathnameBase: match.pathnameBase === '/' ? '/' : match.pathnameBase.slice(1), pattern: Object.assign(Object.assign({}, match.pattern), { path: path }) });
61
- }
62
- return null;
63
- }
64
- // For absolute paths, use React Router's matcher directly.
65
- // React Router v6 routes default to `end: true` unless the pattern
66
- // explicitly opts into wildcards with `*`. Mirror that behaviour so
67
- // matching parity stays aligned with <Route>.
68
- const matchOptions = Object.assign({ path }, restProps);
69
- if ((matchOptions === null || matchOptions === void 0 ? void 0 : matchOptions.end) === undefined) {
70
- matchOptions.end = !path.endsWith('*');
71
- }
72
- return matchPath$1(matchOptions, pathname);
73
- };
74
- /**
75
- * Determines the portion of a pathname that a given route pattern should match against.
76
- * For absolute route patterns we return the full pathname. For relative patterns we
77
- * strip off the already-matched parent segments so React Router receives the remainder.
78
- */
79
- const derivePathnameToMatch = (fullPathname, routePath) => {
80
- var _a;
81
- // For absolute or empty routes, use the full pathname as-is
82
- if (!routePath || routePath === '' || routePath.startsWith('/')) {
83
- return fullPathname;
84
- }
85
- const trimmedPath = fullPathname.startsWith('/') ? fullPathname.slice(1) : fullPathname;
86
- if (!trimmedPath) {
87
- // For root-level relative routes (pathname is "/" and routePath is relative),
88
- // return the full pathname so matchPath can normalize both.
89
- // This allows routes like <Route path="foo/*" .../> at root level to work correctly.
90
- return fullPathname;
91
- }
92
- const fullSegments = trimmedPath.split('/').filter(Boolean);
93
- if (fullSegments.length === 0) {
94
- return '';
95
- }
96
- const routeSegments = routePath.split('/').filter(Boolean);
97
- if (routeSegments.length === 0) {
98
- return trimmedPath;
99
- }
100
- const wildcardIndex = routeSegments.findIndex((segment) => segment === '*' || segment === '**');
101
- if (wildcardIndex >= 0) {
102
- const baseSegments = routeSegments.slice(0, wildcardIndex);
103
- if (baseSegments.length === 0) {
104
- return trimmedPath;
105
- }
106
- const startIndex = fullSegments.findIndex((_, idx) => baseSegments.every((seg, segIdx) => {
107
- const target = fullSegments[idx + segIdx];
108
- if (!target) {
109
- return false;
110
- }
111
- if (seg.startsWith(':')) {
112
- return true;
8
+ class IonRouteInner extends React.PureComponent {
9
+ render() {
10
+ return (React.createElement(Route, Object.assign({ path: this.props.path, exact: this.props.exact, render: this.props.render }, (this.props.computedMatch !== undefined
11
+ ? {
12
+ computedMatch: this.props.computedMatch,
113
13
  }
114
- return target === seg;
115
- }));
116
- if (startIndex >= 0) {
117
- return fullSegments.slice(startIndex).join('/');
118
- }
14
+ : {}))));
119
15
  }
120
- if (routeSegments.length <= fullSegments.length) {
121
- return fullSegments.slice(fullSegments.length - routeSegments.length).join('/');
122
- }
123
- return (_a = fullSegments[fullSegments.length - 1]) !== null && _a !== void 0 ? _a : trimmedPath;
124
- };
16
+ }
125
17
 
126
18
  /**
127
- * Finds the longest common prefix among an array of paths.
128
- * Used to determine the scope of an outlet with absolute routes.
129
- *
130
- * @param paths An array of absolute path strings.
131
- * @returns The common prefix shared by all paths.
132
- */
133
- const computeCommonPrefix = (paths) => {
134
- if (paths.length === 0)
135
- return '';
136
- if (paths.length === 1) {
137
- // For a single path, extract the directory-like prefix
138
- // e.g., /dynamic-routes/home -> /dynamic-routes
139
- const segments = paths[0].split('/').filter(Boolean);
140
- if (segments.length > 1) {
141
- return '/' + segments.slice(0, -1).join('/');
142
- }
143
- return '/' + segments[0];
144
- }
145
- // Split all paths into segments
146
- const segmentArrays = paths.map((p) => p.split('/').filter(Boolean));
147
- const minLength = Math.min(...segmentArrays.map((s) => s.length));
148
- const commonSegments = [];
149
- for (let i = 0; i < minLength; i++) {
150
- const segment = segmentArrays[0][i];
151
- // Skip segments with route parameters or wildcards
152
- if (segment.includes(':') || segment.includes('*')) {
153
- break;
154
- }
155
- const allMatch = segmentArrays.every((s) => s[i] === segment);
156
- if (allMatch) {
157
- commonSegments.push(segment);
158
- }
159
- else {
160
- break;
161
- }
162
- }
163
- return commonSegments.length > 0 ? '/' + commonSegments.join('/') : '';
164
- };
165
- /**
166
- * Checks if a pathname falls within the scope of a mount path using
167
- * segment-aware comparison. Prevents false positives like "/tabs-secondary"
168
- * matching mount path "/tabs".
169
- */
170
- const isPathnameInScope = (pathname, mountPath) => {
171
- if (mountPath === '/')
172
- return true;
173
- return pathname === mountPath || pathname.startsWith(mountPath + '/');
174
- };
175
- /**
176
- * Checks if a route path is a "splat-only" route (just `*` or `/*`).
177
- */
178
- const isSplatOnlyRoute = (routePath) => {
179
- return routePath === '*' || routePath === '/*';
180
- };
181
- /**
182
- * Checks if a route has an embedded wildcard (e.g., "tab1/*" but not "*" or "/*").
183
- */
184
- const hasEmbeddedWildcard = (routePath) => {
185
- return !!routePath && routePath.includes('*') && !isSplatOnlyRoute(routePath);
186
- };
187
- /**
188
- * Checks if a route with an embedded wildcard matches a pathname.
189
- */
190
- const matchesEmbeddedWildcardRoute = (route, pathname) => {
191
- const routePath = route.props.path;
192
- if (!hasEmbeddedWildcard(routePath)) {
193
- return false;
194
- }
195
- return !!matchPath({ pathname, componentProps: route.props });
196
- };
197
- /**
198
- * Checks if a route is a specific match (not wildcard-only or index).
19
+ * @see https://v5.reactrouter.com/web/api/matchPath
199
20
  */
200
- const isSpecificRouteMatch = (route, remainingPath) => {
201
- const routePath = route.props.path;
202
- if (route.props.index || isSplatOnlyRoute(routePath)) {
21
+ const matchPath = ({ pathname, componentProps, }) => {
22
+ const { exact, component } = componentProps;
23
+ const path = componentProps.path || componentProps.from;
24
+ /***
25
+ * The props to match against, they are identical
26
+ * to the matching props `Route` accepts. It could also be a string
27
+ * or an array of strings as shortcut for `{ path }`.
28
+ */
29
+ const matchProps = {
30
+ exact,
31
+ path,
32
+ component,
33
+ };
34
+ const match = matchPath$1(pathname, matchProps);
35
+ if (!match) {
203
36
  return false;
204
37
  }
205
- return !!matchPath({ pathname: remainingPath, componentProps: route.props });
206
- };
207
- /**
208
- * Analyzes route children to determine their characteristics.
209
- *
210
- * @param routeChildren The route children to analyze.
211
- * @returns Analysis of the route characteristics.
212
- */
213
- const analyzeRouteChildren = (routeChildren) => {
214
- const hasRelativeRoutes = routeChildren.some((route) => {
215
- const path = route.props.path;
216
- return path && !path.startsWith('/') && path !== '*';
217
- });
218
- const hasIndexRoute = routeChildren.some((route) => route.props.index);
219
- const hasWildcardRoute = routeChildren.some((route) => {
220
- const routePath = route.props.path;
221
- return routePath === '*' || routePath === '/*';
222
- });
223
- return { hasRelativeRoutes, hasIndexRoute, hasWildcardRoute, routeChildren };
224
- };
225
- /**
226
- * Checks if any route matches as a specific (non-wildcard, non-index) route.
227
- */
228
- const findSpecificMatch = (routeChildren, remainingPath) => {
229
- return routeChildren.some((route) => isSpecificRouteMatch(route, remainingPath) || matchesEmbeddedWildcardRoute(route, remainingPath));
230
- };
231
- /**
232
- * Returns the first route that matches as a specific (non-wildcard, non-index) route.
233
- */
234
- const findFirstSpecificMatchingRoute = (routeChildren, remainingPath) => {
235
- return routeChildren.find((route) => isSpecificRouteMatch(route, remainingPath) || matchesEmbeddedWildcardRoute(route, remainingPath));
236
- };
237
- /**
238
- * Checks if any specific route could plausibly match the remaining path.
239
- * Used to determine if we should fall back to a wildcard match.
240
- *
241
- * Uses exact first-segment matching: the remaining path's first segment
242
- * must exactly equal a route's first segment to block the wildcard.
243
- * The outlet's mount path is always known from React Router's RouteContext,
244
- * so no heuristic-based discovery is needed.
245
- */
246
- const couldSpecificRouteMatch = (routeChildren, remainingPath) => {
247
- const remainingFirstSegment = remainingPath.split('/')[0];
248
- return routeChildren.some((route) => {
249
- const routePath = route.props.path;
250
- if (!routePath || routePath === '*' || routePath === '/*')
251
- return false;
252
- if (route.props.index)
253
- return false;
254
- const routeFirstSegment = routePath.split('/')[0].replace(/[*:]/g, '');
255
- if (!routeFirstSegment)
256
- return false;
257
- return routeFirstSegment === remainingFirstSegment;
258
- });
259
- };
260
- /**
261
- * Determines the best parent path from the available matches.
262
- * Priority: specific > wildcard > index
263
- */
264
- const selectBestMatch = (specificMatch, wildcardMatch, indexMatch) => {
265
- var _a;
266
- return (_a = specificMatch !== null && specificMatch !== void 0 ? specificMatch : wildcardMatch) !== null && _a !== void 0 ? _a : indexMatch;
267
- };
268
- /**
269
- * Handles outlets with only absolute routes by computing their common prefix.
270
- */
271
- const computeAbsoluteRoutesParentPath = (routeChildren, currentPathname, outletMountPath) => {
272
- const absolutePathRoutes = routeChildren.filter((route) => {
273
- const path = route.props.path;
274
- return path && path.startsWith('/');
275
- });
276
- if (absolutePathRoutes.length === 0) {
277
- return undefined;
278
- }
279
- const absolutePaths = absolutePathRoutes.map((r) => r.props.path);
280
- const commonPrefix = computeCommonPrefix(absolutePaths);
281
- if (!commonPrefix || commonPrefix === '/') {
282
- return undefined;
283
- }
284
- const newOutletMountPath = outletMountPath || commonPrefix;
285
- if (!currentPathname.startsWith(commonPrefix)) {
286
- return { parentPath: undefined, outletMountPath: newOutletMountPath };
287
- }
288
- return { parentPath: commonPrefix, outletMountPath: newOutletMountPath };
289
- };
290
- /**
291
- * Computes the parent path for a nested outlet based on the current pathname
292
- * and the outlet's route configuration.
293
- *
294
- * When the mount path is known (seeded from React Router's RouteContext), the
295
- * parent path is simply the mount path — no iterative discovery needed. The
296
- * iterative fallback only runs for outlets where RouteContext doesn't provide
297
- * a parent match (typically root-level outlets on first render).
298
- *
299
- * @param options The options for computing the parent path.
300
- * @returns The computed parent path result.
301
- */
302
- const computeParentPath = (options) => {
303
- const { currentPathname, outletMountPath, routeChildren, hasRelativeRoutes, hasIndexRoute, hasWildcardRoute } = options;
304
- // If pathname is outside the established mount path scope, skip computation.
305
- // Use segment-aware comparison: /tabs-secondary must NOT match /tabs scope.
306
- if (outletMountPath && !isPathnameInScope(currentPathname, outletMountPath)) {
307
- return { parentPath: undefined, outletMountPath };
308
- }
309
- // Fast path: when the mount path is known (from React Router's RouteContext),
310
- // the parent path IS the mount path. The iterative segment-by-segment discovery
311
- // below was needed when the mount depth had to be guessed from URL structure,
312
- // but with RouteContext we already know exactly where this outlet is mounted.
313
- if (outletMountPath && (hasRelativeRoutes || hasIndexRoute)) {
314
- return { parentPath: outletMountPath, outletMountPath };
315
- }
316
- // Fallback: mount path not yet known. Iterate through path segments to discover
317
- // the correct parent depth. This only runs on first render of outlets where
318
- // RouteContext doesn't provide a parent match (typically root-level outlets,
319
- // which usually have absolute routes and take the absolute routes path below).
320
- if (!outletMountPath && (hasRelativeRoutes || hasIndexRoute) && currentPathname.includes('/')) {
321
- const segments = currentPathname.split('/').filter(Boolean);
322
- if (segments.length >= 1) {
323
- let firstSpecificMatch;
324
- let firstWildcardMatch;
325
- let indexMatchAtMount;
326
- for (let i = 1; i <= segments.length; i++) {
327
- const parentPath = '/' + segments.slice(0, i).join('/');
328
- const remainingPath = segments.slice(i).join('/');
329
- // Check for specific route match (highest priority)
330
- if (!firstSpecificMatch && findSpecificMatch(routeChildren, remainingPath)) {
331
- // Don't let empty/default path routes (path="" or undefined) drive
332
- // the parent deeper than a wildcard match. An empty path route matching
333
- // when remainingPath is "" just means all segments were consumed.
334
- if (firstWildcardMatch) {
335
- const matchingRoute = findFirstSpecificMatchingRoute(routeChildren, remainingPath);
336
- if (matchingRoute) {
337
- const matchingPath = matchingRoute.props.path;
338
- if (!matchingPath || matchingPath === '') {
339
- continue;
340
- }
341
- }
342
- }
343
- firstSpecificMatch = parentPath;
344
- break;
345
- }
346
- // Check for wildcard match (only if remaining path is non-empty)
347
- const hasNonEmptyRemaining = remainingPath !== '' && remainingPath !== '/';
348
- if (!firstWildcardMatch && hasNonEmptyRemaining && hasWildcardRoute) {
349
- if (!couldSpecificRouteMatch(routeChildren, remainingPath)) {
350
- firstWildcardMatch = parentPath;
351
- }
352
- }
353
- // Check for index route match
354
- if ((remainingPath === '' || remainingPath === '/') && hasIndexRoute) {
355
- indexMatchAtMount = parentPath;
356
- }
357
- }
358
- // Fallback: check root level for embedded wildcard routes (e.g., "tab1/*")
359
- if (!firstSpecificMatch) {
360
- const fullRemainingPath = segments.join('/');
361
- if (routeChildren.some((route) => matchesEmbeddedWildcardRoute(route, fullRemainingPath))) {
362
- firstSpecificMatch = '/';
363
- }
364
- }
365
- const bestPath = selectBestMatch(firstSpecificMatch, firstWildcardMatch, indexMatchAtMount);
366
- return { parentPath: bestPath, outletMountPath: bestPath };
367
- }
368
- }
369
- // Handle outlets with only absolute routes
370
- if (!hasRelativeRoutes && !hasIndexRoute) {
371
- const result = computeAbsoluteRoutesParentPath(routeChildren, currentPathname, outletMountPath);
372
- if (result) {
373
- return result;
374
- }
375
- }
376
- return { parentPath: outletMountPath, outletMountPath };
377
- };
378
-
379
- /**
380
- * Ensures the given path has a leading slash.
381
- *
382
- * @param value The path string to normalize.
383
- * @returns The path with a leading slash.
384
- */
385
- const ensureLeadingSlash = (value) => {
386
- if (value === '') {
387
- return '/';
388
- }
389
- return value.startsWith('/') ? value : `/${value}`;
390
- };
391
- /**
392
- * Strips the trailing slash from a path, unless it's the root path.
393
- *
394
- * @param value The path string to normalize.
395
- * @returns The path without a trailing slash.
396
- */
397
- const stripTrailingSlash = (value) => {
398
- return value.length > 1 && value.endsWith('/') ? value.slice(0, -1) : value;
399
- };
400
- /**
401
- * Normalizes a pathname for comparison by ensuring a leading slash
402
- * and removing trailing slashes.
403
- *
404
- * @param value The pathname to normalize, can be undefined.
405
- * @returns A normalized pathname string.
406
- */
407
- const normalizePathnameForComparison = (value) => {
408
- if (!value || value === '') {
409
- return '/';
410
- }
411
- const withLeadingSlash = ensureLeadingSlash(value);
412
- return stripTrailingSlash(withLeadingSlash);
413
- };
414
-
415
- /**
416
- * Extracts the children from a Routes wrapper component.
417
- * The use of `<Routes />` is encouraged with React Router v6.
418
- *
419
- * @param node The React node to extract Routes children from.
420
- * @returns The children of the Routes component, or undefined if not found.
421
- */
422
- const getRoutesChildren = (node) => {
423
- let routesNode;
424
- React.Children.forEach(node, (child) => {
425
- if (child.type === Routes) {
426
- routesNode = child;
427
- }
428
- });
429
- if (routesNode) {
430
- // The children of the `<Routes />` component are most likely
431
- // (and should be) the `<Route />` components.
432
- return routesNode.props.children;
433
- }
434
- return undefined;
435
- };
436
- /**
437
- * Extracts Route children from a node (either directly or from a Routes wrapper).
438
- *
439
- * @param children The children to extract routes from.
440
- * @returns An array of Route elements.
441
- */
442
- const extractRouteChildren = (children) => {
443
- var _a;
444
- const routesChildren = (_a = getRoutesChildren(children)) !== null && _a !== void 0 ? _a : children;
445
- return React.Children.toArray(routesChildren).filter((child) => React.isValidElement(child) && (child.type === Route || child.type === IonRoute));
446
- };
447
- /**
448
- * Checks if a React element is a Navigate component (redirect).
449
- *
450
- * @param element The element to check.
451
- * @returns True if the element is a Navigate component.
452
- */
453
- const isNavigateElement = (element) => {
454
- return (React.isValidElement(element) &&
455
- (element.type === Navigate || (typeof element.type === 'function' && element.type.name === 'Navigate')));
456
- };
457
-
458
- /**
459
- * Compares two routes by specificity for sorting (most specific first).
460
- *
461
- * Sort order:
462
- * 1. Index routes come first
463
- * 2. Wildcard-only routes (* or /*) come last
464
- * 3. Exact matches (no wildcards/params) before wildcard/param routes
465
- * 4. Among routes with same status, longer paths are more specific
466
- */
467
- const compareRouteSpecificity = (a, b) => {
468
- // Index routes come first
469
- if (a.index && !b.index)
470
- return -1;
471
- if (!a.index && b.index)
472
- return 1;
473
- // Wildcard-only routes (* or /*) should come last
474
- const aIsWildcardOnly = a.path === '*' || a.path === '/*';
475
- const bIsWildcardOnly = b.path === '*' || b.path === '/*';
476
- if (!aIsWildcardOnly && bIsWildcardOnly)
477
- return -1;
478
- if (aIsWildcardOnly && !bIsWildcardOnly)
479
- return 1;
480
- // Exact matches (no wildcards/params) come before wildcard/param routes
481
- const aHasWildcard = a.path.includes('*') || a.path.includes(':');
482
- const bHasWildcard = b.path.includes('*') || b.path.includes(':');
483
- if (!aHasWildcard && bHasWildcard)
484
- return -1;
485
- if (aHasWildcard && !bHasWildcard)
486
- return 1;
487
- // Among routes with same wildcard status, longer paths are more specific
488
- if (a.path.length !== b.path.length) {
489
- return b.path.length - a.path.length;
490
- }
491
- return 0;
492
- };
493
- /**
494
- * Sorts view items by route specificity (most specific first).
495
- *
496
- * Sort order aligns with findViewItemByPath in ReactRouterViewStack.tsx:
497
- * 1. Index routes come first
498
- * 2. Wildcard-only routes (* or /*) come last
499
- * 3. Exact matches (no wildcards/params) come before wildcard/param routes
500
- * 4. Among routes with same wildcard status, longer paths are more specific
501
- *
502
- * @param views The view items to sort.
503
- * @returns A new sorted array of view items.
504
- */
505
- const sortViewsBySpecificity = (views) => {
506
- return [...views].sort((a, b) => {
507
- var _a, _b, _c, _d, _e, _f, _g, _h;
508
- return compareRouteSpecificity({ path: ((_b = (_a = a.routeData) === null || _a === void 0 ? void 0 : _a.childProps) === null || _b === void 0 ? void 0 : _b.path) || '', index: !!((_d = (_c = a.routeData) === null || _c === void 0 ? void 0 : _c.childProps) === null || _d === void 0 ? void 0 : _d.index) }, { path: ((_f = (_e = b.routeData) === null || _e === void 0 ? void 0 : _e.childProps) === null || _f === void 0 ? void 0 : _f.path) || '', index: !!((_h = (_g = b.routeData) === null || _g === void 0 ? void 0 : _g.childProps) === null || _h === void 0 ? void 0 : _h.index) });
509
- });
38
+ return match;
510
39
  };
511
40
 
512
- /**
513
- * `ReactRouterViewStack` is a custom navigation manager used in Ionic React
514
- * apps to map React Router route elements (such as `<IonRoute>`) to "view
515
- * items" that Ionic can manage in a view stack. This is critical to maintain
516
- * Ionic’s animation, lifecycle, and history behavior across views.
517
- */
518
- /**
519
- * Delay in milliseconds before removing a Navigate view item after a redirect.
520
- * This ensures the redirect navigation completes before the view is removed.
521
- */
522
- const NAVIGATE_REDIRECT_DELAY_MS = 100;
523
- /**
524
- * Delay in milliseconds before cleaning up a view without an IonPage element.
525
- * This double-checks that the view is truly not needed before removal.
526
- */
527
- const VIEW_CLEANUP_DELAY_MS = 200;
528
- /**
529
- * Computes the absolute pathnameBase for a route element based on its type.
530
- * Handles relative paths, index routes, and splat routes differently.
531
- */
532
- const computeAbsolutePathnameBase = (routeElement, routeMatch, parentPathnameBase, routeInfoPathname) => {
533
- const routePath = routeElement.props.path;
534
- const isRelativePath = routePath && !routePath.startsWith('/');
535
- const isIndexRoute = !!routeElement.props.index;
536
- const isSplatOnlyRoute = routePath === '*' || routePath === '/*';
537
- if (isSplatOnlyRoute) {
538
- // Splat routes should NOT contribute their matched portion to pathnameBase
539
- // This aligns with React Router v7's v7_relativeSplatPath behavior
540
- return parentPathnameBase;
541
- }
542
- if (isRelativePath && (routeMatch === null || routeMatch === void 0 ? void 0 : routeMatch.pathnameBase)) {
543
- const relativeBase = routeMatch.pathnameBase.startsWith('/')
544
- ? routeMatch.pathnameBase.slice(1)
545
- : routeMatch.pathnameBase;
546
- return parentPathnameBase === '/' ? `/${relativeBase}` : `${parentPathnameBase}/${relativeBase}`;
547
- }
548
- if (isIndexRoute) {
549
- return parentPathnameBase;
550
- }
551
- return (routeMatch === null || routeMatch === void 0 ? void 0 : routeMatch.pathnameBase) || routeInfoPathname;
552
- };
553
- /**
554
- * Gets fallback params from view items in other outlets when parent context is empty.
555
- * This handles cases where React context propagation doesn't work as expected.
556
- */
557
- const getFallbackParamsFromViewItems = (allViewItems, currentOutletId, currentPathname) => {
558
- var _a;
559
- const matchingViews = [];
560
- for (const otherViewItem of allViewItems) {
561
- if (otherViewItem.outletId === currentOutletId)
562
- continue;
563
- const otherMatch = (_a = otherViewItem.routeData) === null || _a === void 0 ? void 0 : _a.match;
564
- if ((otherMatch === null || otherMatch === void 0 ? void 0 : otherMatch.params) && Object.keys(otherMatch.params).length > 0) {
565
- const matchedPathname = otherMatch.pathnameBase || otherMatch.pathname;
566
- if (matchedPathname && currentPathname.startsWith(matchedPathname)) {
567
- matchingViews.push({
568
- params: otherMatch.params,
569
- pathLength: matchedPathname.length,
570
- });
571
- }
572
- }
573
- }
574
- // Sort ascending by path length so more-specific (longer) paths are applied
575
- // last and their params take priority over less-specific ones.
576
- matchingViews.sort((a, b) => a.pathLength - b.pathLength);
577
- const params = {};
578
- for (const view of matchingViews) {
579
- Object.assign(params, view.params);
580
- }
581
- return params;
582
- };
583
- /**
584
- * Builds the matches array for RouteContext.
585
- */
586
- const buildContextMatches = (parentMatches, combinedParams, routeMatch, routeInfoPathname, absolutePathnameBase, viewItem, routeElement, componentElement) => {
587
- return [
588
- ...parentMatches,
589
- {
590
- params: combinedParams,
591
- pathname: (routeMatch === null || routeMatch === void 0 ? void 0 : routeMatch.pathname) || routeInfoPathname,
592
- pathnameBase: absolutePathnameBase,
593
- route: {
594
- id: viewItem.id,
595
- path: routeElement.props.path,
596
- element: componentElement,
597
- index: !!routeElement.props.index,
598
- caseSensitive: routeElement.props.caseSensitive,
599
- hasErrorBoundary: false,
600
- },
601
- },
602
- ];
603
- };
604
- const createDefaultMatch = (fullPathname, routeProps) => {
605
- var _a, _b;
606
- const isIndexRoute = !!routeProps.index;
607
- const patternPath = (_a = routeProps.path) !== null && _a !== void 0 ? _a : '';
608
- const pathnameBase = fullPathname === '' ? '/' : fullPathname;
609
- const computedEnd = routeProps.end !== undefined ? routeProps.end : patternPath !== '' ? !patternPath.endsWith('*') : true;
610
- return {
611
- params: {},
612
- pathname: isIndexRoute ? '' : fullPathname,
613
- pathnameBase,
614
- pattern: {
615
- path: patternPath,
616
- caseSensitive: (_b = routeProps.caseSensitive) !== null && _b !== void 0 ? _b : false,
617
- end: isIndexRoute ? true : computedEnd,
618
- },
619
- };
620
- };
621
- const computeRelativeToParent = (pathname, parentPath) => {
622
- if (!parentPath)
623
- return null;
624
- const normalizedParent = normalizePathnameForComparison(parentPath);
625
- const normalizedPathname = normalizePathnameForComparison(pathname);
626
- if (normalizedPathname === normalizedParent) {
627
- return '';
628
- }
629
- const withSlash = normalizedParent === '/' ? '/' : normalizedParent + '/';
630
- if (normalizedPathname.startsWith(withSlash)) {
631
- return normalizedPathname.slice(withSlash.length);
632
- }
633
- return null;
634
- };
635
- const resolveIndexRouteMatch = (viewItem, pathname, parentPath) => {
636
- var _a, _b, _c;
637
- if (!((_b = (_a = viewItem.routeData) === null || _a === void 0 ? void 0 : _a.childProps) === null || _b === void 0 ? void 0 : _b.index)) {
638
- return null;
639
- }
640
- // Prefer computing against the parent path when available to align with RRv6 semantics
641
- const relative = computeRelativeToParent(pathname, parentPath);
642
- if (relative !== null) {
643
- // Index routes match only when there is no remaining path
644
- if (relative === '' || relative === '/') {
645
- return createDefaultMatch(parentPath || pathname, viewItem.routeData.childProps);
646
- }
647
- return null;
648
- }
649
- // Fallback: use previously computed match base for equality check
650
- const previousMatch = (_c = viewItem.routeData) === null || _c === void 0 ? void 0 : _c.match;
651
- if (!previousMatch) {
652
- return null;
653
- }
654
- const normalizedPathname = normalizePathnameForComparison(pathname);
655
- const normalizedBase = normalizePathnameForComparison(previousMatch.pathnameBase || previousMatch.pathname || '');
656
- return normalizedPathname === normalizedBase ? previousMatch : null;
657
- };
658
41
  class ReactRouterViewStack extends ViewStacks {
659
42
  constructor() {
660
43
  super();
661
- /**
662
- * Stores the computed parent path for each outlet.
663
- * Used by findViewItemByPath to correctly evaluate index route matches
664
- * without requiring the outlet's React element or route children.
665
- */
666
- this.outletParentPaths = new Map();
667
- /**
668
- * Stores the computed mount path for each outlet.
669
- * Fed back into computeParentPath on subsequent calls to stabilize
670
- * the parent path computation across navigations (mirrors StackManager.outletMountPath).
671
- */
672
- this.outletMountPaths = new Map();
673
- /**
674
- * Creates a new view item for the given outlet and react route element.
675
- * Associates route props with the matched route path for further lookups.
676
- */
677
- this.createViewItem = (outletId, reactElement, routeInfo, page) => {
678
- var _a, _b;
679
- const routePath = reactElement.props.path || '';
680
- // Check if we already have a view item for this exact route that we can reuse
681
- // Include wildcard routes like tabs/* since they should be reused
682
- // Also check unmounted items that might have been preserved for browser navigation
683
- const existingViewItem = this.getViewItemsForOutlet(outletId).find((v) => {
684
- var _a, _b, _c, _d, _e, _f, _g, _h;
685
- const existingRouteProps = (_b = (_a = v.reactElement) === null || _a === void 0 ? void 0 : _a.props) !== null && _b !== void 0 ? _b : {};
686
- const existingPath = existingRouteProps.path || '';
687
- const existingElement = existingRouteProps.element;
688
- const newElement = reactElement.props.element;
689
- const existingIsIndexRoute = !!existingRouteProps.index;
690
- const newIsIndexRoute = !!reactElement.props.index;
691
- // For Navigate components, match by destination
692
- const existingIsNavigate = React.isValidElement(existingElement) && existingElement.type === Navigate;
693
- const newIsNavigate = React.isValidElement(newElement) && newElement.type === Navigate;
694
- if (existingIsNavigate && newIsNavigate) {
695
- const existingTo = (_c = existingElement.props) === null || _c === void 0 ? void 0 : _c.to;
696
- const newTo = (_d = newElement.props) === null || _d === void 0 ? void 0 : _d.to;
697
- if (existingTo === newTo) {
698
- return true;
699
- }
700
- }
701
- if (existingIsIndexRoute && newIsIndexRoute) {
702
- return true;
703
- }
704
- // Reuse view items with the same path
705
- // Special case: reuse tabs/* and other specific wildcard routes
706
- // Don't reuse index routes (empty path) or generic catch-all wildcards (*)
707
- if (existingPath === routePath && existingPath !== '' && existingPath !== '*') {
708
- // Parameterized routes need pathname matching to ensure /details/1 and /details/2
709
- // get separate view items. For wildcard routes (e.g., user/:userId/*), compare
710
- // pathnameBase to allow child path changes while preserving the parent view.
711
- const hasParams = routePath.includes(':');
712
- const isWildcard = routePath.includes('*');
713
- if (hasParams) {
714
- if (isWildcard) {
715
- const existingPathnameBase = (_f = (_e = v.routeData) === null || _e === void 0 ? void 0 : _e.match) === null || _f === void 0 ? void 0 : _f.pathnameBase;
716
- const newMatch = matchComponent$1(reactElement, routeInfo.pathname, false, this.outletParentPaths.get(outletId));
717
- const newPathnameBase = newMatch === null || newMatch === void 0 ? void 0 : newMatch.pathnameBase;
718
- if (existingPathnameBase !== newPathnameBase) {
719
- return false;
720
- }
721
- }
722
- else {
723
- const existingPathname = (_h = (_g = v.routeData) === null || _g === void 0 ? void 0 : _g.match) === null || _h === void 0 ? void 0 : _h.pathname;
724
- if (existingPathname !== routeInfo.pathname) {
725
- return false;
726
- }
727
- }
728
- }
729
- return true;
730
- }
731
- // Also reuse specific wildcard routes like tabs/*
732
- if (existingPath === routePath && existingPath.endsWith('/*') && existingPath !== '/*') {
733
- return true;
734
- }
735
- return false;
736
- });
737
- if (existingViewItem) {
738
- // Update and ensure the existing view item is properly configured
739
- existingViewItem.reactElement = reactElement;
740
- existingViewItem.mount = true;
741
- existingViewItem.ionPageElement = page || existingViewItem.ionPageElement;
742
- const updatedMatch = matchComponent$1(reactElement, routeInfo.pathname, false, this.outletParentPaths.get(outletId)) ||
743
- ((_a = existingViewItem.routeData) === null || _a === void 0 ? void 0 : _a.match) ||
744
- createDefaultMatch(routeInfo.pathname, reactElement.props);
745
- existingViewItem.routeData = {
746
- match: updatedMatch,
747
- childProps: reactElement.props,
748
- lastPathname: (_b = existingViewItem.routeData) === null || _b === void 0 ? void 0 : _b.lastPathname, // Preserve navigation history
749
- };
750
- return existingViewItem;
751
- }
752
- const id = `${outletId}-${generateId(outletId)}`;
753
- const viewItem = {
754
- id,
755
- outletId,
756
- ionPageElement: page,
757
- reactElement,
758
- mount: true,
759
- ionRoute: true,
760
- };
761
- if (reactElement.type === IonRoute) {
762
- viewItem.disableIonPageManagement = reactElement.props.disableIonPageManagement;
763
- }
764
- const initialMatch = matchComponent$1(reactElement, routeInfo.pathname, true, this.outletParentPaths.get(outletId)) ||
765
- createDefaultMatch(routeInfo.pathname, reactElement.props);
766
- viewItem.routeData = {
767
- match: initialMatch,
768
- childProps: reactElement.props,
769
- };
770
- this.add(viewItem);
771
- return viewItem;
44
+ this.createViewItem = this.createViewItem.bind(this);
45
+ this.findViewItemByRouteInfo = this.findViewItemByRouteInfo.bind(this);
46
+ this.findLeavingViewItemByRouteInfo = this.findLeavingViewItemByRouteInfo.bind(this);
47
+ this.getChildrenToRender = this.getChildrenToRender.bind(this);
48
+ this.findViewItemByPathname = this.findViewItemByPathname.bind(this);
49
+ }
50
+ createViewItem(outletId, reactElement, routeInfo, page) {
51
+ const viewItem = {
52
+ id: generateId('viewItem'),
53
+ outletId,
54
+ ionPageElement: page,
55
+ reactElement,
56
+ mount: true,
57
+ ionRoute: false,
772
58
  };
773
- /**
774
- * Renders a ViewLifeCycleManager for the given view item.
775
- * Handles cleanup if the view no longer matches.
776
- *
777
- * - Deactivates view if it no longer matches the current route
778
- * - Wraps the route element in <Routes> to support nested routing and ensure remounting
779
- * - Adds a unique key to <Routes> so React Router remounts routes when switching
780
- */
781
- this.renderViewItem = (viewItem, routeInfo, parentPath, reRender) => {
782
- var _a, _b, _c, _d, _e, _f, _g, _h;
783
- const routePath = viewItem.reactElement.props.path || '';
784
- let match = matchComponent$1(viewItem.reactElement, routeInfo.pathname, false, parentPath);
785
- if (!match) {
786
- const indexMatch = resolveIndexRouteMatch(viewItem, routeInfo.pathname, parentPath);
787
- if (indexMatch) {
788
- match = indexMatch;
789
- }
790
- }
791
- // For parameterized routes, check if this is a navigation to a different path instance
792
- // In that case, we should NOT reuse this view - a new view should be created
793
- const isParameterRoute = routePath.includes(':');
794
- const previousMatch = (_a = viewItem.routeData) === null || _a === void 0 ? void 0 : _a.match;
795
- const isSamePath = (match === null || match === void 0 ? void 0 : match.pathname) === (previousMatch === null || previousMatch === void 0 ? void 0 : previousMatch.pathname);
796
- // Flag to indicate this view should not be reused for this different parameterized path
797
- const shouldSkipForDifferentParam = isParameterRoute && match && previousMatch && !isSamePath;
798
- // Don't deactivate views automatically - let the StackManager handle view lifecycle
799
- // This preserves views in the stack for navigation history like native apps
800
- // Views will be hidden/shown by the StackManager's transition logic instead of being unmounted
801
- // Special handling for Navigate components - they should unmount after redirecting
802
- const elementComponent = (_c = (_b = viewItem.reactElement) === null || _b === void 0 ? void 0 : _b.props) === null || _c === void 0 ? void 0 : _c.element;
803
- const isNavigateComponent = isNavigateElement(elementComponent);
804
- if (isNavigateComponent) {
805
- // Navigate components should only be mounted when they match
806
- // Once they redirect (no longer match), they should be removed completely
807
- // IMPORTANT: For index routes, we need to check indexMatch too since matchComponent
808
- // may not properly match index routes without explicit parent path context
809
- const indexMatch = ((_e = (_d = viewItem.routeData) === null || _d === void 0 ? void 0 : _d.childProps) === null || _e === void 0 ? void 0 : _e.index)
810
- ? resolveIndexRouteMatch(viewItem, routeInfo.pathname, parentPath)
811
- : null;
812
- const hasValidMatch = match || indexMatch;
813
- if (!hasValidMatch && viewItem.mount) {
814
- viewItem.mount = false;
815
- // Schedule removal of the Navigate view item after a short delay
816
- // This ensures the redirect completes before removal
817
- setTimeout(() => {
818
- this.remove(viewItem);
819
- reRender === null || reRender === void 0 ? void 0 : reRender();
820
- }, NAVIGATE_REDIRECT_DELAY_MS);
821
- }
822
- }
823
- // Components that don't have IonPage elements and no longer match should be cleaned up
824
- // BUT we need to be careful not to remove them if they're part of browser navigation history
825
- // This handles components that perform immediate actions like programmatic navigation
826
- // EXCEPTION: Navigate components should ALWAYS remain mounted until they redirect
827
- // since they need to be rendered to trigger the navigation
828
- if (!match && viewItem.mount && !viewItem.ionPageElement && !isNavigateComponent) {
829
- // Check if this view item should be preserved for browser navigation
830
- // We'll keep it if it was recently active (within the last navigation)
831
- const shouldPreserve = viewItem.routeData.lastPathname === routeInfo.pathname ||
832
- ((_f = viewItem.routeData.match) === null || _f === void 0 ? void 0 : _f.pathname) === routeInfo.lastPathname;
833
- if (!shouldPreserve) {
834
- // This view item doesn't match and doesn't have an IonPage
835
- // It's likely a utility component that performs an action and navigates away
836
- viewItem.mount = false;
837
- // Schedule removal to allow it to be recreated on next navigation
838
- setTimeout(() => {
839
- // Double-check before removing - the view might be needed again
840
- const stillNotNeeded = !viewItem.mount && !viewItem.ionPageElement;
841
- if (stillNotNeeded) {
842
- this.remove(viewItem);
843
- reRender === null || reRender === void 0 ? void 0 : reRender();
844
- }
845
- }, VIEW_CLEANUP_DELAY_MS);
846
- }
847
- else {
848
- // Preserve it but unmount it for now
849
- viewItem.mount = false;
850
- }
851
- }
852
- // Reactivate view if it matches but was previously deactivated
853
- // Don't reactivate if this is a parameterized route navigating to a different path instance
854
- // Don't reactivate catch-all wildcard routes — they are created fresh by createViewItem
855
- const isCatchAllWildcard = routePath === '*' || routePath === '/*';
856
- if (match && !viewItem.mount && !shouldSkipForDifferentParam && !isCatchAllWildcard) {
857
- viewItem.mount = true;
858
- viewItem.routeData.match = match;
859
- }
860
- // Deactivate wildcard (catch-all) and empty-path (default) routes when a more-specific route matches.
861
- // This prevents "Not found" or fallback pages from showing alongside valid routes.
862
- if (routePath === '*' || routePath === '') {
863
- // Check if any other view in this outlet has a match for the current route
864
- const outletViews = this.getViewItemsForOutlet(viewItem.outletId);
865
- // When parent path context is available, compute the relative pathname once
866
- // outside the loop since both routeInfo.pathname and parentPath are invariant.
867
- const relativePathname = parentPath
868
- ? computeRelativeToParent(routeInfo.pathname, parentPath)
869
- : null;
870
- let hasSpecificMatch = outletViews.some((v) => {
871
- var _a, _b;
872
- if (v.id === viewItem.id)
873
- return false; // Skip self
874
- const vRoutePath = ((_b = (_a = v.reactElement) === null || _a === void 0 ? void 0 : _a.props) === null || _b === void 0 ? void 0 : _b.path) || '';
875
- if (vRoutePath === '*' || vRoutePath === '')
876
- return false; // Skip other wildcard/empty routes
877
- // When parent path context is available and the route is relative, use
878
- // parent-path-aware matching. This avoids false positives from
879
- // derivePathnameToMatch's tail-slice heuristic, which can incorrectly
880
- // match route literals that appear at the wrong position in the pathname.
881
- // Example: pathname /parent/extra/details/99 with route details/:id —
882
- // the tail-slice extracts ["details","99"] producing a false match.
883
- if (parentPath && vRoutePath && !vRoutePath.startsWith('/')) {
884
- if (relativePathname === null) {
885
- return false; // Pathname is outside this outlet's parent scope
886
- }
887
- return !!matchPath({
888
- pathname: relativePathname,
889
- componentProps: v.reactElement.props,
890
- });
891
- }
892
- // Fallback to matchComponent when no parent path context is available
893
- const vMatch = v.reactElement ? matchComponent$1(v.reactElement, routeInfo.pathname) : null;
894
- return !!vMatch;
895
- });
896
- // For catch-all * routes, also deactivate when the pathname matches the outlet's
897
- // parent path exactly. This means there are no remaining segments for the wildcard
898
- // to catch, so the empty-path or index route should handle it instead.
899
- if (!hasSpecificMatch && routePath === '*') {
900
- const outletParentPath = this.outletParentPaths.get(viewItem.outletId);
901
- if (outletParentPath) {
902
- const normalizedParent = normalizePathnameForComparison(outletParentPath);
903
- const normalizedPathname = normalizePathnameForComparison(routeInfo.pathname);
904
- if (normalizedPathname === normalizedParent) {
905
- // Check if there's an empty-path or index view item that should handle this
906
- const hasDefaultRoute = outletViews.some((v) => {
907
- var _a, _b, _c, _d;
908
- if (v.id === viewItem.id)
909
- return false;
910
- const vRoutePath = (_b = (_a = v.reactElement) === null || _a === void 0 ? void 0 : _a.props) === null || _b === void 0 ? void 0 : _b.path;
911
- return vRoutePath === '' || vRoutePath === undefined || !!((_d = (_c = v.routeData) === null || _c === void 0 ? void 0 : _c.childProps) === null || _d === void 0 ? void 0 : _d.index);
912
- });
913
- if (hasDefaultRoute) {
914
- hasSpecificMatch = true;
915
- }
916
- }
917
- }
918
- }
919
- if (hasSpecificMatch) {
920
- viewItem.mount = false;
921
- if (viewItem.ionPageElement) {
922
- viewItem.ionPageElement.classList.add('ion-page-hidden');
923
- viewItem.ionPageElement.setAttribute('aria-hidden', 'true');
924
- }
925
- }
926
- }
927
- const routeElement = React.cloneElement(viewItem.reactElement);
928
- const componentElement = routeElement.props.element;
929
- // Don't update match for parameterized routes navigating to different path instances
930
- // This preserves the original match so that findViewItemByPath can correctly skip this view
931
- if (match && viewItem.routeData.match !== match && !shouldSkipForDifferentParam) {
932
- viewItem.routeData.match = match;
933
- }
934
- const routeMatch = shouldSkipForDifferentParam ? (_g = viewItem.routeData) === null || _g === void 0 ? void 0 : _g.match : match || ((_h = viewItem.routeData) === null || _h === void 0 ? void 0 : _h.match);
935
- return (React.createElement(UNSAFE_RouteContext.Consumer, { key: `view-context-${viewItem.id}` }, (parentContext) => {
936
- var _a, _b;
937
- const parentMatches = ((_a = parentContext === null || parentContext === void 0 ? void 0 : parentContext.matches) !== null && _a !== void 0 ? _a : []);
938
- // Accumulate params from parent matches, with fallback to other outlets
939
- let accumulatedParentParams = parentMatches.reduce((acc, m) => (Object.assign(Object.assign({}, acc), m.params)), {});
940
- if (parentMatches.length === 0 && Object.keys(accumulatedParentParams).length === 0) {
941
- accumulatedParentParams = getFallbackParamsFromViewItems(this.getAllViewItems(), viewItem.outletId, routeInfo.pathname);
942
- }
943
- const combinedParams = Object.assign(Object.assign({}, accumulatedParentParams), ((_b = routeMatch === null || routeMatch === void 0 ? void 0 : routeMatch.params) !== null && _b !== void 0 ? _b : {}));
944
- const parentPathnameBase = parentMatches.length > 0 ? parentMatches[parentMatches.length - 1].pathnameBase : '/';
945
- const absolutePathnameBase = computeAbsolutePathnameBase(routeElement, routeMatch, parentPathnameBase, routeInfo.pathname);
946
- const contextMatches = buildContextMatches(parentMatches, combinedParams, routeMatch, routeInfo.pathname, absolutePathnameBase, viewItem, routeElement, componentElement);
947
- const routeContextValue = parentContext
948
- ? Object.assign(Object.assign({}, parentContext), { matches: contextMatches }) : { outlet: null, matches: contextMatches, isDataRoute: false };
949
- return (React.createElement(ViewLifeCycleManager, { key: `view-${viewItem.id}`, mount: viewItem.mount, removeView: () => this.remove(viewItem) },
950
- React.createElement(UNSAFE_RouteContext.Provider, { value: routeContextValue }, componentElement)));
951
- }));
59
+ if (reactElement.type === IonRoute) {
60
+ viewItem.ionRoute = true;
61
+ viewItem.disableIonPageManagement = reactElement.props.disableIonPageManagement;
62
+ }
63
+ viewItem.routeData = {
64
+ match: matchPath({
65
+ pathname: routeInfo.pathname,
66
+ componentProps: reactElement.props,
67
+ }),
68
+ childProps: reactElement.props,
952
69
  };
953
- /**
954
- * Re-renders all active view items for the specified outlet.
955
- * Ensures React elements are updated with the latest match.
956
- *
957
- * 1. Iterates through children of IonRouterOutlet
958
- * 2. Updates each matching viewItem with the current child React element
959
- * (important for updating props or changes to elements)
960
- * 3. Returns a list of React components that will be rendered inside the outlet
961
- * Each view is wrapped in <ViewLifeCycleManager> to manage lifecycle and rendering
962
- */
963
- this.getChildrenToRender = (outletId, ionRouterOutlet, routeInfo, reRender, parentPathnameBase) => {
964
- const viewItems = this.getViewItemsForOutlet(outletId);
965
- // Seed the mount path from the parent route context if available.
966
- // This provides the outlet's mount path immediately on first render,
967
- // eliminating the need for heuristic-based discovery in computeParentPath.
968
- if (parentPathnameBase && !this.outletMountPaths.has(outletId)) {
969
- this.outletMountPaths.set(outletId, parentPathnameBase);
970
- }
971
- // Determine parentPath for outlets with relative or index routes.
972
- // This populates outletParentPaths for findViewItemByPath's matchView
973
- // and the catch-all deactivation logic in renderViewItem.
974
- let parentPath = undefined;
975
- try {
976
- const routeChildren = extractRouteChildren(ionRouterOutlet.props.children);
977
- const { hasRelativeRoutes, hasIndexRoute, hasWildcardRoute } = analyzeRouteChildren(routeChildren);
978
- if (hasRelativeRoutes || hasIndexRoute) {
979
- const result = computeParentPath({
980
- currentPathname: routeInfo.pathname,
981
- outletMountPath: this.outletMountPaths.get(outletId),
982
- routeChildren,
983
- hasRelativeRoutes,
984
- hasIndexRoute,
985
- hasWildcardRoute,
986
- });
987
- parentPath = result.parentPath;
988
- // Persist the mount path for subsequent calls, mirroring StackManager.outletMountPath.
989
- // Unlike outletParentPaths (cleared when parentPath is undefined), the mount path is
990
- // intentionally sticky — it anchors the outlet's scope and is only removed in clear().
991
- if (result.outletMountPath && !this.outletMountPaths.has(outletId)) {
992
- this.outletMountPaths.set(outletId, result.outletMountPath);
993
- }
994
- }
995
- }
996
- catch (e) {
997
- // Non-fatal: if we fail to compute parentPath, fall back to previous behavior
998
- }
999
- // Store the computed parentPath for use in findViewItemByPath.
1000
- // Clear stale entries when parentPath is undefined (e.g., navigated out of scope).
1001
- if (parentPath !== undefined) {
1002
- this.outletParentPaths.set(outletId, parentPath);
1003
- }
1004
- else if (this.outletParentPaths.has(outletId)) {
1005
- this.outletParentPaths.delete(outletId);
1006
- }
1007
- // Sync child elements with stored viewItems (e.g. to reflect new props)
1008
- React.Children.forEach(ionRouterOutlet.props.children, (child) => {
1009
- // Ensure the child is a valid React element since we
1010
- // might have whitespace strings or other non-element children
1011
- if (React.isValidElement(child)) {
1012
- // Find view item by exact path match to avoid wildcard routes overwriting specific routes
1013
- const childPath = child.props.path;
1014
- const viewItem = viewItems.find((v) => {
1015
- var _a, _b;
1016
- const viewItemPath = (_b = (_a = v.reactElement) === null || _a === void 0 ? void 0 : _a.props) === null || _b === void 0 ? void 0 : _b.path;
1017
- // Only update if paths match exactly (prevents wildcard routes from overwriting specific routes)
1018
- return viewItemPath === childPath;
1019
- });
1020
- if (viewItem) {
1021
- viewItem.reactElement = child;
1022
- }
1023
- }
1024
- });
1025
- // Filter out duplicate view items by ID (but keep all mounted items)
1026
- const uniqueViewItems = viewItems.filter((viewItem, index, array) => {
1027
- // Remove duplicates by ID (keep first occurrence)
1028
- const isFirstOccurrence = array.findIndex((v) => v.id === viewItem.id) === index;
1029
- return isFirstOccurrence;
1030
- });
1031
- // Filter out unmounted Navigate components to prevent them from being rendered
1032
- // and triggering unwanted redirects
1033
- const renderableViewItems = uniqueViewItems.filter((viewItem) => {
1034
- var _a, _b, _c, _d;
1035
- const elementComponent = (_b = (_a = viewItem.reactElement) === null || _a === void 0 ? void 0 : _a.props) === null || _b === void 0 ? void 0 : _b.element;
1036
- const isNavigateComponent = isNavigateElement(elementComponent);
1037
- // Exclude unmounted Navigate components from rendering
1038
- if (isNavigateComponent && !viewItem.mount) {
1039
- return false;
1040
- }
1041
- // Filter out views that are unmounted, have no ionPageElement, and don't match the current route.
1042
- // These are "stale" views from previous routes that should not be rendered.
1043
- // Views WITH ionPageElement are handled by the normal lifecycle events.
1044
- // Views that MATCH the current route should be kept (they might be transitioning).
1045
- if (!viewItem.mount && !viewItem.ionPageElement) {
1046
- // Check if this view's route path matches the current pathname
1047
- const viewRoutePath = (_d = (_c = viewItem.reactElement) === null || _c === void 0 ? void 0 : _c.props) === null || _d === void 0 ? void 0 : _d.path;
1048
- if (viewRoutePath) {
1049
- // First try exact match using matchComponent
1050
- const routeMatch = matchComponent$1(viewItem.reactElement, routeInfo.pathname, false, parentPath);
1051
- if (routeMatch) {
1052
- // View matches current route, keep it
1053
- return true;
1054
- }
1055
- // For parent routes (like /multiple-tabs or /routing), check if current pathname
1056
- // starts with this route's path. This handles views with IonSplitPane/IonTabs
1057
- // that don't have IonPage but should remain mounted while navigating within their children.
1058
- const normalizedViewPath = normalizePathnameForComparison(viewRoutePath.replace(/\/?\*$/, '')); // Remove trailing wildcard
1059
- const normalizedCurrentPath = normalizePathnameForComparison(routeInfo.pathname);
1060
- // Check if current pathname is within this view's route hierarchy
1061
- const isWithinRouteHierarchy = normalizedCurrentPath === normalizedViewPath || normalizedCurrentPath.startsWith(normalizedViewPath + '/');
1062
- if (!isWithinRouteHierarchy) {
1063
- // View is outside current route hierarchy, remove it
1064
- setTimeout(() => {
1065
- this.remove(viewItem);
1066
- reRender();
1067
- }, 0);
1068
- return false;
1069
- }
1070
- }
1071
- }
1072
- return true;
70
+ return viewItem;
71
+ }
72
+ getChildrenToRender(outletId, ionRouterOutlet, routeInfo) {
73
+ const viewItems = this.getViewItemsForOutlet(outletId);
74
+ // Sync latest routes with viewItems
75
+ React.Children.forEach(ionRouterOutlet.props.children, (child) => {
76
+ const viewItem = viewItems.find((v) => {
77
+ return matchComponent$1(child, v.routeData.childProps.path || v.routeData.childProps.from);
1073
78
  });
1074
- const renderedItems = renderableViewItems.map((viewItem) => this.renderViewItem(viewItem, routeInfo, parentPath, reRender));
1075
- return renderedItems;
1076
- };
1077
- /**
1078
- * Finds a view item matching the current route, optionally updating its match state.
1079
- */
1080
- this.findViewItemByRouteInfo = (routeInfo, outletId, updateMatch) => {
1081
- const { viewItem, match } = this.findViewItemByPath(routeInfo.pathname, outletId);
1082
- const shouldUpdateMatch = updateMatch === undefined || updateMatch === true;
1083
- if (shouldUpdateMatch && viewItem && match) {
1084
- viewItem.routeData.match = match;
79
+ if (viewItem) {
80
+ viewItem.reactElement = child;
1085
81
  }
1086
- return viewItem;
1087
- };
1088
- /**
1089
- * Finds the view item that was previously active before a route change.
1090
- */
1091
- this.findLeavingViewItemByRouteInfo = (routeInfo, outletId, mustBeIonRoute = true) => {
1092
- // If the lastPathname is not set, we cannot find a leaving view item
1093
- if (!routeInfo.lastPathname) {
1094
- return undefined;
1095
- }
1096
- const { viewItem } = this.findViewItemByPath(routeInfo.lastPathname, outletId, mustBeIonRoute);
1097
- return viewItem;
1098
- };
1099
- /**
1100
- * Finds a view item by pathname only, used in simpler queries.
1101
- */
1102
- this.findViewItemByPathname = (pathname, outletId) => {
1103
- const { viewItem } = this.findViewItemByPath(pathname, outletId);
1104
- return viewItem;
1105
- };
1106
- /**
1107
- * Clean up old, unmounted view items to prevent memory leaks
1108
- */
1109
- this.cleanupStaleViewItems = (outletId) => {
1110
- const viewItems = this.getViewItemsForOutlet(outletId);
1111
- // Keep only the most recent mounted views and a few unmounted ones for history
1112
- const maxUnmountedItems = 3;
1113
- const unmountedItems = viewItems.filter((v) => !v.mount);
1114
- if (unmountedItems.length > maxUnmountedItems) {
1115
- // Remove oldest unmounted items
1116
- const itemsToRemove = unmountedItems.slice(0, unmountedItems.length - maxUnmountedItems);
1117
- itemsToRemove.forEach((item) => {
1118
- this.remove(item);
1119
- });
82
+ });
83
+ const children = viewItems.map((viewItem) => {
84
+ let clonedChild;
85
+ if (viewItem.ionRoute && !viewItem.disableIonPageManagement) {
86
+ clonedChild = (React.createElement(ViewLifeCycleManager, { key: `view-${viewItem.id}`, mount: viewItem.mount, removeView: () => this.remove(viewItem) }, React.cloneElement(viewItem.reactElement, {
87
+ computedMatch: viewItem.routeData.match,
88
+ })));
1120
89
  }
1121
- };
1122
- /**
1123
- * Override add to prevent duplicate view items with the same ID in the same outlet
1124
- * But allow multiple view items for the same route path (for navigation history)
1125
- */
1126
- this.add = (viewItem) => {
1127
- const existingViewItem = this.getViewItemsForOutlet(viewItem.outletId).find((v) => v.id === viewItem.id);
1128
- if (existingViewItem) {
1129
- return;
90
+ else {
91
+ const match = matchComponent$1(viewItem.reactElement, routeInfo.pathname);
92
+ clonedChild = (React.createElement(ViewLifeCycleManager, { key: `view-${viewItem.id}`, mount: viewItem.mount, removeView: () => this.remove(viewItem) }, React.cloneElement(viewItem.reactElement, {
93
+ computedMatch: viewItem.routeData.match,
94
+ })));
95
+ if (!match && viewItem.routeData.match) {
96
+ viewItem.routeData.match = undefined;
97
+ viewItem.mount = false;
98
+ }
1130
99
  }
1131
- super.add(viewItem);
1132
- this.cleanupStaleViewItems(viewItem.outletId);
1133
- };
1134
- /**
1135
- * Override clear to also clean up the stored parent path for the outlet.
1136
- */
1137
- this.clear = (outletId) => {
1138
- this.outletParentPaths.delete(outletId);
1139
- this.outletMountPaths.delete(outletId);
1140
- return super.clear(outletId);
1141
- };
1142
- /**
1143
- * Override remove
1144
- */
1145
- this.remove = (viewItem) => {
1146
- super.remove(viewItem);
1147
- };
100
+ return clonedChild;
101
+ });
102
+ return children;
103
+ }
104
+ findViewItemByRouteInfo(routeInfo, outletId, updateMatch) {
105
+ const { viewItem, match } = this.findViewItemByPath(routeInfo.pathname, outletId);
106
+ const shouldUpdateMatch = updateMatch === undefined || updateMatch === true;
107
+ if (shouldUpdateMatch && viewItem && match) {
108
+ viewItem.routeData.match = match;
109
+ }
110
+ return viewItem;
111
+ }
112
+ findLeavingViewItemByRouteInfo(routeInfo, outletId, mustBeIonRoute = true) {
113
+ const { viewItem } = this.findViewItemByPath(routeInfo.lastPathname, outletId, mustBeIonRoute);
114
+ return viewItem;
115
+ }
116
+ findViewItemByPathname(pathname, outletId) {
117
+ const { viewItem } = this.findViewItemByPath(pathname, outletId);
118
+ return viewItem;
1148
119
  }
1149
120
  /**
1150
- * Core function that matches a given pathname against all view items.
1151
- * Returns both the matched view item and match metadata.
121
+ * Returns the matching view item and the match result for a given pathname.
1152
122
  */
1153
- findViewItemByPath(pathname, outletId, mustBeIonRoute, allowDefaultMatch = true) {
123
+ findViewItemByPath(pathname, outletId, mustBeIonRoute) {
1154
124
  let viewItem;
1155
- let match = null;
125
+ let match;
1156
126
  let viewStack;
1157
- // Capture stored parent paths for use in nested matchView/matchDefaultRoute functions
1158
- const storedParentPaths = this.outletParentPaths;
1159
127
  if (outletId) {
1160
- viewStack = sortViewsBySpecificity(this.getViewItemsForOutlet(outletId));
128
+ viewStack = this.getViewItemsForOutlet(outletId);
1161
129
  viewStack.some(matchView);
1162
- if (!viewItem && allowDefaultMatch)
130
+ if (!viewItem) {
1163
131
  viewStack.some(matchDefaultRoute);
132
+ }
1164
133
  }
1165
134
  else {
1166
- const viewItems = sortViewsBySpecificity(this.getAllViewItems());
135
+ const viewItems = this.getAllViewItems();
1167
136
  viewItems.some(matchView);
1168
- if (!viewItem && allowDefaultMatch)
137
+ if (!viewItem) {
1169
138
  viewItems.some(matchDefaultRoute);
139
+ }
1170
140
  }
1171
- // If we still have not found a view item for this outlet, try to find a matching
1172
- // view item across all outlets and adopt it into the current outlet. This helps
1173
- // recover when an outlet remounts and receives a new id, leaving views associated
1174
- // with the previous outlet id.
1175
- // Do not adopt across outlets; if we didn't find a view for this outlet,
1176
- // defer to route matching to create a new one.
1177
141
  return { viewItem, match };
1178
- /**
1179
- * Matches a route path with dynamic parameters (e.g. /tabs/:id)
1180
- */
1181
142
  function matchView(v) {
1182
- var _a;
1183
- if (mustBeIonRoute && !v.ionRoute)
1184
- return false;
1185
- const viewItemPath = v.routeData.childProps.path || '';
1186
- // Skip unmounted catch-all wildcard views. After back navigation unmounts
1187
- // a wildcard view, it should not be reused for subsequent navigations.
1188
- // A fresh wildcard view will be created by createViewItem when needed.
1189
- if ((viewItemPath === '*' || viewItemPath === '/*') && !v.mount)
143
+ var _a, _b;
144
+ if (mustBeIonRoute && !v.ionRoute) {
1190
145
  return false;
1191
- const isIndexRoute = !!v.routeData.childProps.index;
1192
- const previousMatch = (_a = v.routeData) === null || _a === void 0 ? void 0 : _a.match;
1193
- const outletParentPath = storedParentPaths.get(v.outletId);
1194
- const result = v.reactElement ? matchComponent$1(v.reactElement, pathname, false, outletParentPath) : null;
1195
- if (!result) {
1196
- const indexMatch = resolveIndexRouteMatch(v, pathname, outletParentPath);
1197
- if (indexMatch) {
1198
- match = indexMatch;
1199
- viewItem = v;
1200
- return true;
1201
- }
1202
- // Empty path routes (path="") should match when the pathname matches the
1203
- // outlet's parent path exactly (no remaining segments). matchComponent doesn't
1204
- // handle this because it lacks parent path context. Without this check, a
1205
- // catch-all * view item (which matches any pathname) would be incorrectly
1206
- // returned instead of the empty path route on back navigation.
1207
- if (viewItemPath === '' && !isIndexRoute && outletParentPath) {
1208
- const normalizedParent = normalizePathnameForComparison(outletParentPath);
1209
- const normalizedPathname = normalizePathnameForComparison(pathname);
1210
- if (normalizedPathname === normalizedParent) {
1211
- match = createDefaultMatch(pathname, v.routeData.childProps);
1212
- viewItem = v;
1213
- return true;
1214
- }
1215
- }
1216
146
  }
1217
- if (result) {
1218
- const hasParams = result.params && Object.keys(result.params).length > 0;
1219
- const isSamePath = result.pathname === (previousMatch === null || previousMatch === void 0 ? void 0 : previousMatch.pathname);
1220
- const isWildcardRoute = viewItemPath.includes('*');
1221
- const isParameterRoute = viewItemPath.includes(':');
1222
- // Don't allow view items with undefined paths to match specific routes
1223
- // This prevents broken index route view items from interfering with navigation
1224
- if (!viewItemPath && !isIndexRoute && pathname !== '/' && pathname !== '') {
1225
- return false;
1226
- }
1227
- // For parameterized routes, check if we should reuse the view item.
1228
- // Wildcard routes (e.g., user/:userId/*) compare pathnameBase to allow
1229
- // child path changes while preserving the parent view.
1230
- if (isParameterRoute && !isSamePath) {
1231
- if (isWildcardRoute) {
1232
- const isSameBase = result.pathnameBase === (previousMatch === null || previousMatch === void 0 ? void 0 : previousMatch.pathnameBase);
1233
- if (isSameBase) {
1234
- match = result;
1235
- viewItem = v;
1236
- return true;
1237
- }
1238
- }
1239
- return false;
1240
- }
1241
- // For routes without params, or when navigating to the exact same path,
1242
- // or when there's no previous match, reuse the view item
1243
- if (!hasParams || isSamePath || !previousMatch) {
1244
- match = result;
147
+ match = matchPath({
148
+ pathname,
149
+ componentProps: v.routeData.childProps,
150
+ });
151
+ if (match) {
152
+ /**
153
+ * Even though we have a match from react-router, we do not know if the match
154
+ * is for this specific view item.
155
+ *
156
+ * To validate this, we need to check if the path and url match the view item's route data.
157
+ */
158
+ const hasParameter = match.path.includes(':');
159
+ if (!hasParameter || (hasParameter && match.url === ((_b = (_a = v.routeData) === null || _a === void 0 ? void 0 : _a.match) === null || _b === void 0 ? void 0 : _b.url))) {
1245
160
  viewItem = v;
1246
161
  return true;
1247
162
  }
1248
- // For pure wildcard routes (without : params), compare pathnameBase to allow
1249
- // child path changes while preserving the parent view. This handles container
1250
- // routes like /tabs/* where switching between /tabs/tab1 and /tabs/tab2
1251
- // should reuse the same ViewItem.
1252
- if (isWildcardRoute && !isParameterRoute) {
1253
- const isSameBase = result.pathnameBase === (previousMatch === null || previousMatch === void 0 ? void 0 : previousMatch.pathnameBase);
1254
- if (isSameBase) {
1255
- match = result;
1256
- viewItem = v;
1257
- return true;
1258
- }
1259
- }
1260
163
  }
1261
164
  return false;
1262
165
  }
1263
- /**
1264
- * Matches a view with no path prop (default fallback route) or index route.
1265
- */
1266
166
  function matchDefaultRoute(v) {
1267
- var _a, _b, _c;
1268
- const childProps = v.routeData.childProps;
1269
- const isDefaultRoute = childProps.path === undefined || childProps.path === '';
1270
- const isIndexRoute = !!childProps.index;
1271
- if (isIndexRoute) {
1272
- const outletParentPath = storedParentPaths.get(v.outletId);
1273
- const indexMatch = resolveIndexRouteMatch(v, pathname, outletParentPath);
1274
- if (indexMatch) {
1275
- match = indexMatch;
1276
- viewItem = v;
1277
- return true;
1278
- }
1279
- return false;
1280
- }
1281
- // For empty path routes, only match if we're at the same level as when the view was created.
1282
- // This prevents an empty path view item from being reused for different routes.
1283
- if (isDefaultRoute) {
1284
- const previousPathnameBase = ((_b = (_a = v.routeData) === null || _a === void 0 ? void 0 : _a.match) === null || _b === void 0 ? void 0 : _b.pathnameBase) || '';
1285
- const normalizedBase = normalizePathnameForComparison(previousPathnameBase);
1286
- const normalizedPathname = normalizePathnameForComparison(pathname);
1287
- if (normalizedPathname !== normalizedBase) {
1288
- return false;
1289
- }
167
+ // try to find a route that doesn't have a path or from prop, that will be our default route
168
+ if (!v.routeData.childProps.path && !v.routeData.childProps.from) {
1290
169
  match = {
170
+ path: pathname,
171
+ url: pathname,
172
+ isExact: true,
1291
173
  params: {},
1292
- pathname,
1293
- pathnameBase: pathname === '' ? '/' : pathname,
1294
- pattern: {
1295
- path: '',
1296
- caseSensitive: (_c = childProps.caseSensitive) !== null && _c !== void 0 ? _c : false,
1297
- end: true,
1298
- },
1299
174
  };
1300
175
  viewItem = v;
1301
176
  return true;
@@ -1304,40 +179,11 @@ class ReactRouterViewStack extends ViewStacks {
1304
179
  }
1305
180
  }
1306
181
  }
1307
- /**
1308
- * Utility to apply matchPath to a React element and return its match state.
1309
- */
1310
- function matchComponent$1(node, pathname, allowFallback = false, parentPath) {
1311
- var _a;
1312
- const routeProps = (_a = node === null || node === void 0 ? void 0 : node.props) !== null && _a !== void 0 ? _a : {};
1313
- const routePath = routeProps.path;
1314
- let pathnameToMatch;
1315
- if (parentPath && routePath && !routePath.startsWith('/')) {
1316
- // When parent path is known, compute exact relative pathname
1317
- // instead of using the tail-slice heuristic
1318
- const relative = pathname.startsWith(parentPath)
1319
- ? pathname.slice(parentPath.length).replace(/^\//, '')
1320
- : pathname;
1321
- pathnameToMatch = relative;
1322
- }
1323
- else {
1324
- pathnameToMatch = derivePathnameToMatch(pathname, routePath);
1325
- }
1326
- const match = matchPath({
1327
- pathname: pathnameToMatch,
1328
- componentProps: routeProps,
1329
- });
1330
- if (match || !allowFallback) {
1331
- return match;
1332
- }
1333
- const isIndexRoute = !!routeProps.index;
1334
- if (isIndexRoute) {
1335
- return createDefaultMatch(pathname, routeProps);
1336
- }
1337
- if (!routePath || routePath === '') {
1338
- return createDefaultMatch(pathname, routeProps);
1339
- }
1340
- return null;
182
+ function matchComponent$1(node, pathname) {
183
+ return matchPath({
184
+ pathname,
185
+ componentProps: node.props,
186
+ });
1341
187
  }
1342
188
 
1343
189
  function clonePageElement(leavingViewHtml) {
@@ -1362,37 +208,7 @@ function clonePageElement(leavingViewHtml) {
1362
208
  return undefined;
1363
209
  }
1364
210
 
1365
- /**
1366
- * `StackManager` is responsible for managing page transitions, keeping track
1367
- * of views (pages), and ensuring that navigation behaves like native apps —
1368
- * particularly with animations and swipe gestures.
1369
- */
1370
- /**
1371
- * Delay in milliseconds before unmounting a view after a transition completes.
1372
- * This ensures the page transition animation finishes before the view is removed.
1373
- */
1374
- const VIEW_UNMOUNT_DELAY_MS = 250;
1375
- /**
1376
- * Delay (ms) to wait for an IonPage to mount before proceeding with a
1377
- * page transition. Only container routes (nested outlets with no direct
1378
- * IonPage) actually hit this timeout; normal routes clear it early via
1379
- * registerIonPage, so a larger value here doesn't affect the happy path.
1380
- */
1381
- const ION_PAGE_WAIT_TIMEOUT_MS = 300;
1382
- const isViewVisible = (el) => !el.classList.contains('ion-page-invisible') && !el.classList.contains('ion-page-hidden') && el.style.visibility !== 'hidden';
1383
- const hideIonPageElement = (element) => {
1384
- if (element) {
1385
- element.classList.add('ion-page-hidden');
1386
- element.setAttribute('aria-hidden', 'true');
1387
- }
1388
- };
1389
- const showIonPageElement = (element) => {
1390
- if (element) {
1391
- element.style.removeProperty('visibility');
1392
- element.classList.remove('ion-page-hidden');
1393
- element.removeAttribute('aria-hidden');
1394
- }
1395
- };
211
+ const isViewVisible = (el) => !el.classList.contains('ion-page-invisible') && !el.classList.contains('ion-page-hidden');
1396
212
  class StackManager extends React.PureComponent {
1397
213
  constructor(props) {
1398
214
  super(props);
@@ -1401,613 +217,29 @@ class StackManager extends React.PureComponent {
1401
217
  isInOutlet: () => true,
1402
218
  };
1403
219
  this.pendingPageTransition = false;
1404
- this.waitingForIonPage = false;
1405
- /** Tracks whether the component is mounted to guard async transition paths. */
1406
- this._isMounted = false;
1407
- /** In-flight requestAnimationFrame IDs from transitionPage, cancelled on unmount. */
1408
- this.transitionRafIds = [];
1409
- /**
1410
- * Monotonically increasing counter incremented at the start of each transitionPage call.
1411
- * Used to detect when an async commit() resolves after a newer transition has already run,
1412
- * preventing the stale commit from hiding an element that the newer transition made visible.
1413
- */
1414
- this.transitionGeneration = 0;
1415
- this.outletMountPath = undefined;
1416
- /**
1417
- * Whether this outlet is at the root level (no parent route matches).
1418
- * Derived from UNSAFE_RouteContext in render() — empty matches means root.
1419
- */
1420
- this.isRootOutlet = true;
1421
220
  this.registerIonPage = this.registerIonPage.bind(this);
1422
221
  this.transitionPage = this.transitionPage.bind(this);
1423
222
  this.handlePageTransition = this.handlePageTransition.bind(this);
1424
- this.id = props.id || `routerOutlet-${generateId('routerOutlet')}`;
223
+ this.id = generateId('routerOutlet');
1425
224
  this.prevProps = undefined;
1426
225
  this.skipTransition = false;
1427
226
  }
1428
- /**
1429
- * Determines the parent path for nested routing in React Router 6.
1430
- *
1431
- * When the mount path is known (seeded from UNSAFE_RouteContext), returns
1432
- * it directly — no iterative discovery needed. The computeParentPath
1433
- * fallback only runs for root outlets where RouteContext doesn't provide
1434
- * a parent match.
1435
- */
1436
- getParentPath() {
1437
- const currentPathname = this.props.routeInfo.pathname;
1438
- // Prevent out-of-scope outlets from adopting unrelated routes.
1439
- // Uses segment-aware comparison: /tabs-secondary must NOT match /tabs scope.
1440
- if (this.outletMountPath && !isPathnameInScope(currentPathname, this.outletMountPath)) {
1441
- return undefined;
1442
- }
1443
- // Fast path: mount path is known from RouteContext. The parent path IS the
1444
- // mount path — no need to run the iterative computeParentPath algorithm.
1445
- if (this.outletMountPath && !this.isRootOutlet) {
1446
- return this.outletMountPath;
1447
- }
1448
- // Fallback: root outlet or mount path not yet seeded. Run the full
1449
- // computeParentPath algorithm to discover the parent depth.
1450
- if (this.ionRouterOutlet) {
1451
- const routeChildren = extractRouteChildren(this.ionRouterOutlet.props.children);
1452
- const { hasRelativeRoutes, hasIndexRoute, hasWildcardRoute } = analyzeRouteChildren(routeChildren);
1453
- if (!this.isRootOutlet || hasRelativeRoutes || hasIndexRoute) {
1454
- const result = computeParentPath({
1455
- currentPathname,
1456
- outletMountPath: this.outletMountPath,
1457
- routeChildren,
1458
- hasRelativeRoutes,
1459
- hasIndexRoute,
1460
- hasWildcardRoute,
1461
- });
1462
- if (result.outletMountPath && !this.outletMountPath) {
1463
- this.outletMountPath = result.outletMountPath;
1464
- }
1465
- return result.parentPath;
1466
- }
1467
- }
1468
- return this.outletMountPath;
1469
- }
1470
- /**
1471
- * Finds the entering and leaving view items, handling redirect cases.
1472
- */
1473
- findViewItems(routeInfo) {
1474
- const enteringViewItem = this.context.findViewItemByRouteInfo(routeInfo, this.id);
1475
- let leavingViewItem = this.context.findLeavingViewItemByRouteInfo(routeInfo, this.id);
1476
- // Try to find leaving view by previous pathname
1477
- if (!leavingViewItem && routeInfo.prevRouteLastPathname) {
1478
- leavingViewItem = this.context.findViewItemByPathname(routeInfo.prevRouteLastPathname, this.id);
1479
- }
1480
- // For redirects where entering === leaving, find the actual previous view
1481
- if (enteringViewItem &&
1482
- leavingViewItem &&
1483
- enteringViewItem === leavingViewItem &&
1484
- routeInfo.routeAction === 'replace' &&
1485
- routeInfo.prevRouteLastPathname) {
1486
- const actualLeavingView = this.context.findViewItemByPathname(routeInfo.prevRouteLastPathname, this.id);
1487
- if (actualLeavingView && actualLeavingView !== enteringViewItem) {
1488
- leavingViewItem = actualLeavingView;
1489
- }
1490
- }
1491
- // Handle redirect scenario with no leaving view
1492
- if (enteringViewItem &&
1493
- !leavingViewItem &&
1494
- routeInfo.routeAction === 'replace' &&
1495
- routeInfo.prevRouteLastPathname) {
1496
- const actualLeavingView = this.context.findViewItemByPathname(routeInfo.prevRouteLastPathname, this.id);
1497
- if (actualLeavingView && actualLeavingView !== enteringViewItem) {
1498
- leavingViewItem = actualLeavingView;
1499
- }
1500
- }
1501
- return { enteringViewItem, leavingViewItem };
1502
- }
1503
- shouldUnmountLeavingView(routeInfo, enteringViewItem, leavingViewItem) {
1504
- var _a, _b;
1505
- if (!leavingViewItem) {
1506
- return false;
1507
- }
1508
- if (routeInfo.routeAction === 'replace') {
1509
- const leavingRoutePath = (_b = (_a = leavingViewItem === null || leavingViewItem === void 0 ? void 0 : leavingViewItem.reactElement) === null || _a === void 0 ? void 0 : _a.props) === null || _b === void 0 ? void 0 : _b.path;
1510
- // Never unmount root path or views without a path - needed for back navigation
1511
- if (!leavingRoutePath || leavingRoutePath === '/' || leavingRoutePath === '') {
1512
- return false;
1513
- }
1514
- // Replace actions unmount the leaving view since it's being replaced in history.
1515
- return true;
1516
- }
1517
- // For non-replace actions, only unmount for back navigation
1518
- const isForwardPush = routeInfo.routeAction === 'push' && routeInfo.routeDirection === 'forward';
1519
- if (!isForwardPush && routeInfo.routeDirection !== 'none' && enteringViewItem !== leavingViewItem) {
1520
- return true;
1521
- }
1522
- return false;
1523
- }
1524
- /**
1525
- * Handles out-of-scope outlet. Returns true if transition should be aborted.
1526
- */
1527
- handleOutOfScopeOutlet(routeInfo) {
1528
- if (!this.outletMountPath || isPathnameInScope(routeInfo.pathname, this.outletMountPath)) {
1529
- return false;
1530
- }
1531
- if (this.outOfScopeUnmountTimeout) {
1532
- clearTimeout(this.outOfScopeUnmountTimeout);
1533
- this.outOfScopeUnmountTimeout = undefined;
1534
- }
1535
- // Fire lifecycle events on any visible view before unmounting.
1536
- // When navigating away from a tabbed section, the parent outlet fires
1537
- // ionViewDidLeave on the tabs container, but the active tab child page
1538
- // never receives its own lifecycle events because the core transition
1539
- // dispatches events with bubbles:false. This ensures tab child pages
1540
- // get ionViewWillLeave/ionViewDidLeave so useIonViewDidLeave fires.
1541
- const allViewsInOutlet = this.context.getViewItemsForOutlet(this.id);
1542
- allViewsInOutlet.forEach((viewItem) => {
1543
- if (viewItem.ionPageElement && isViewVisible(viewItem.ionPageElement)) {
1544
- viewItem.ionPageElement.dispatchEvent(new CustomEvent('ionViewWillLeave', { bubbles: false, cancelable: false }));
1545
- viewItem.ionPageElement.dispatchEvent(new CustomEvent('ionViewDidLeave', { bubbles: false, cancelable: false }));
1546
- }
1547
- });
1548
- // Remove view items from the stack but do NOT apply ion-page-hidden.
1549
- // ion-page-hidden sets display:none which immediately removes content
1550
- // from the layout, causing the parent outlet's leaving page to flash
1551
- // blank during its transition animation (issue #25477).
1552
- //
1553
- // Removing from the stack triggers React reconciliation via forceUpdate,
1554
- // which removes the DOM elements. React batches this re-render after all
1555
- // componentDidUpdate calls in the current cycle, so the parent outlet's
1556
- // commit() captures the current DOM state (with content visible) before
1557
- // React processes the removal. The compositor's cached layer is unaffected
1558
- // by subsequent DOM changes during the animation.
1559
- allViewsInOutlet.forEach((viewItem) => {
1560
- this.context.unMountViewItem(viewItem);
1561
- });
1562
- this.forceUpdate();
1563
- return true;
1564
- }
1565
- /**
1566
- * Handles root navigation by unmounting all non-entering views in this outlet.
1567
- * Fires ionViewWillLeave / ionViewDidLeave only on views that are currently visible.
1568
- * Views that are mounted but not visible (e.g., pages earlier in the back stack)
1569
- * are silently unmounted without lifecycle events, consistent with the behavior
1570
- * of out-of-scope outlet cleanup.
1571
- */
1572
- handleRootNavigation(enteringViewItem) {
1573
- const allViewsInOutlet = this.context.getViewItemsForOutlet(this.id);
1574
- allViewsInOutlet.forEach((viewItem) => {
1575
- if (viewItem === enteringViewItem) {
1576
- return;
1577
- }
1578
- if (viewItem.ionPageElement && isViewVisible(viewItem.ionPageElement)) {
1579
- viewItem.ionPageElement.dispatchEvent(new CustomEvent('ionViewWillLeave', { bubbles: false, cancelable: false }));
1580
- viewItem.ionPageElement.dispatchEvent(new CustomEvent('ionViewDidLeave', { bubbles: false, cancelable: false }));
1581
- }
1582
- this.context.unMountViewItem(viewItem);
1583
- });
1584
- }
1585
- /**
1586
- * Handles nested outlet with relative routes but no parent path. Returns true to abort.
1587
- */
1588
- handleOutOfContextNestedOutlet(parentPath, leavingViewItem) {
1589
- var _a;
1590
- if (this.isRootOutlet || parentPath !== undefined || !this.ionRouterOutlet) {
1591
- return false;
1592
- }
1593
- const routesChildren = (_a = getRoutesChildren(this.ionRouterOutlet.props.children)) !== null && _a !== void 0 ? _a : this.ionRouterOutlet.props.children;
1594
- const routeChildren = React.Children.toArray(routesChildren).filter((child) => React.isValidElement(child) && (child.type === Route || child.type === IonRoute));
1595
- const hasRelativeRoutes = routeChildren.some((route) => {
1596
- const path = route.props.path;
1597
- return path && !path.startsWith('/') && path !== '*';
1598
- });
1599
- if (hasRelativeRoutes) {
1600
- hideIonPageElement(leavingViewItem === null || leavingViewItem === void 0 ? void 0 : leavingViewItem.ionPageElement);
1601
- if (leavingViewItem) {
1602
- leavingViewItem.mount = false;
1603
- }
1604
- this.forceUpdate();
1605
- return true;
1606
- }
1607
- return false;
1608
- }
1609
- /**
1610
- * Handles nested outlet with no matching route. Returns true to abort.
1611
- */
1612
- handleNoMatchingRoute(enteringRoute, enteringViewItem, leavingViewItem) {
1613
- if (this.isRootOutlet || enteringRoute || enteringViewItem) {
1614
- return false;
1615
- }
1616
- hideIonPageElement(leavingViewItem === null || leavingViewItem === void 0 ? void 0 : leavingViewItem.ionPageElement);
1617
- if (leavingViewItem) {
1618
- leavingViewItem.mount = false;
1619
- }
1620
- this.forceUpdate();
1621
- return true;
1622
- }
1623
- /**
1624
- * Handles transition when entering view has ion-page element ready.
1625
- */
1626
- handleReadyEnteringView(routeInfo, enteringViewItem, leavingViewItem, shouldUnmountLeavingViewItem) {
1627
- var _a, _b;
1628
- const routePath = (_b = (_a = enteringViewItem.reactElement) === null || _a === void 0 ? void 0 : _a.props) === null || _b === void 0 ? void 0 : _b.path;
1629
- const isParameterizedRoute = routePath ? routePath.includes(':') : false;
1630
- const isWildcardContainerRoute = routePath ? routePath.endsWith('/*') : false;
1631
- // Handle same-view transitions (parameterized routes like /user/:id or container routes like /tabs/*)
1632
- // When entering === leaving, the view is already visible - skip transition to prevent flash
1633
- if (enteringViewItem === leavingViewItem) {
1634
- if (isParameterizedRoute || isWildcardContainerRoute) {
1635
- const updatedMatch = matchComponent(enteringViewItem.reactElement, routeInfo.pathname, true, this.outletMountPath);
1636
- if (updatedMatch) {
1637
- enteringViewItem.routeData.match = updatedMatch;
1638
- }
1639
- const enteringEl = enteringViewItem.ionPageElement;
1640
- if (enteringEl) {
1641
- showIonPageElement(enteringEl);
1642
- enteringEl.classList.remove('ion-page-invisible');
1643
- }
1644
- this.forceUpdate();
1645
- return;
1646
- }
1647
- }
1648
- // For wildcard container routes, check if we're navigating within the same container.
1649
- // If both the current pathname and the previous pathname match the same container route,
1650
- // skip the transition - the nested outlet will handle the actual page change.
1651
- // This handles cases where leavingViewItem lookup fails (e.g., no IonPage wrapper).
1652
- if (isWildcardContainerRoute && routeInfo.lastPathname) {
1653
- // routePath is guaranteed to exist since isWildcardContainerRoute checks routePath?.endsWith('/*')
1654
- const containerBase = routePath.replace(/\/\*$/, '');
1655
- const currentInContainer = routeInfo.pathname.startsWith(containerBase + '/') || routeInfo.pathname === containerBase;
1656
- const previousInContainer = routeInfo.lastPathname.startsWith(containerBase + '/') || routeInfo.lastPathname === containerBase;
1657
- if (currentInContainer && previousInContainer) {
1658
- const updatedMatch = matchComponent(enteringViewItem.reactElement, routeInfo.pathname, true, this.outletMountPath);
1659
- if (updatedMatch) {
1660
- enteringViewItem.routeData.match = updatedMatch;
1661
- }
1662
- this.forceUpdate();
1663
- return;
1664
- }
1665
- }
1666
- if (!leavingViewItem && this.props.routeInfo.prevRouteLastPathname) {
1667
- leavingViewItem = this.context.findViewItemByPathname(this.props.routeInfo.prevRouteLastPathname, this.id);
1668
- }
1669
- // Re-mount views that were previously unmounted (e.g., navigating back to home)
1670
- if (!enteringViewItem.mount) {
1671
- enteringViewItem.mount = true;
1672
- }
1673
- // Check visibility state BEFORE showing entering view
1674
- const enteringWasVisible = enteringViewItem.ionPageElement && isViewVisible(enteringViewItem.ionPageElement);
1675
- const leavingIsHidden = leavingViewItem !== undefined && leavingViewItem.ionPageElement && !isViewVisible(leavingViewItem.ionPageElement);
1676
- const currentTransition = {
1677
- enteringId: enteringViewItem.id,
1678
- leavingId: leavingViewItem === null || leavingViewItem === void 0 ? void 0 : leavingViewItem.id,
1679
- };
1680
- const isDuplicateTransition = leavingViewItem &&
1681
- this.lastTransition &&
1682
- this.lastTransition.leavingId &&
1683
- this.lastTransition.enteringId === currentTransition.enteringId &&
1684
- this.lastTransition.leavingId === currentTransition.leavingId;
1685
- // Skip if transition already performed (e.g., via swipe gesture)
1686
- if (enteringWasVisible && leavingIsHidden && isDuplicateTransition) {
1687
- if (this.skipTransition &&
1688
- shouldUnmountLeavingViewItem &&
1689
- leavingViewItem &&
1690
- enteringViewItem !== leavingViewItem) {
1691
- leavingViewItem.mount = false;
1692
- // Trigger ionViewDidLeave lifecycle for ViewLifeCycleManager cleanup
1693
- this.transitionPage(routeInfo, enteringViewItem, leavingViewItem, 'back');
1694
- }
1695
- this.skipTransition = false;
1696
- this.forceUpdate();
1697
- return;
1698
- }
1699
- showIonPageElement(enteringViewItem.ionPageElement);
1700
- // Handle duplicate transition or swipe gesture completion
1701
- if (isDuplicateTransition || this.skipTransition) {
1702
- if (this.skipTransition &&
1703
- shouldUnmountLeavingViewItem &&
1704
- leavingViewItem &&
1705
- enteringViewItem !== leavingViewItem) {
1706
- leavingViewItem.mount = false;
1707
- // Re-fire ionViewDidLeave since gesture completed before mount=false was set
1708
- this.transitionPage(routeInfo, enteringViewItem, leavingViewItem, 'back');
1709
- }
1710
- this.skipTransition = false;
1711
- this.forceUpdate();
1712
- return;
1713
- }
1714
- this.lastTransition = currentTransition;
1715
- const shouldSkipAnimation = this.applySkipAnimationIfNeeded(enteringViewItem, leavingViewItem);
1716
- this.transitionPage(routeInfo, enteringViewItem, leavingViewItem, undefined, false, shouldSkipAnimation);
1717
- if (shouldUnmountLeavingViewItem && leavingViewItem && enteringViewItem !== leavingViewItem) {
1718
- // For non-replace actions (back nav), set mount=false here to hide the view.
1719
- // For replace actions, handleLeavingViewUnmount sets mount=false only after
1720
- // its container-to-container guard passes, avoiding zombie state.
1721
- if (routeInfo.routeAction !== 'replace') {
1722
- leavingViewItem.mount = false;
1723
- }
1724
- this.handleLeavingViewUnmount(routeInfo, enteringViewItem, leavingViewItem);
1725
- }
1726
- // Clean up orphaned sibling views after replace actions (redirects)
1727
- this.cleanupOrphanedSiblingViews(routeInfo, enteringViewItem, leavingViewItem);
1728
- }
1729
- /**
1730
- * Handles leaving view unmount for replace actions.
1731
- */
1732
- handleLeavingViewUnmount(routeInfo, enteringViewItem, leavingViewItem) {
1733
- var _a, _b, _c, _d, _e, _f;
1734
- // Only replace actions unmount views; push/pop cache for navigation history
1735
- if (routeInfo.routeAction !== 'replace') {
1736
- return;
1737
- }
1738
- if (!leavingViewItem.ionPageElement) {
1739
- leavingViewItem.mount = false;
1740
- const viewToUnmount = leavingViewItem;
1741
- setTimeout(() => {
1742
- this.context.unMountViewItem(viewToUnmount);
1743
- this.forceUpdate();
1744
- }, VIEW_UNMOUNT_DELAY_MS);
1745
- return;
1746
- }
1747
- const enteringRoutePath = (_b = (_a = enteringViewItem.reactElement) === null || _a === void 0 ? void 0 : _a.props) === null || _b === void 0 ? void 0 : _b.path;
1748
- const leavingRoutePath = (_d = (_c = leavingViewItem.reactElement) === null || _c === void 0 ? void 0 : _c.props) === null || _d === void 0 ? void 0 : _d.path;
1749
- const isEnteringContainerRoute = enteringRoutePath && enteringRoutePath.endsWith('/*');
1750
- const isLeavingSpecificRoute = leavingRoutePath &&
1751
- leavingRoutePath !== '' &&
1752
- leavingRoutePath !== '*' &&
1753
- !leavingRoutePath.endsWith('/*') &&
1754
- !((_f = (_e = leavingViewItem.reactElement) === null || _e === void 0 ? void 0 : _e.props) === null || _f === void 0 ? void 0 : _f.index);
1755
- // Skip removal for container-to-container transitions (e.g., /tabs/* → /settings/*).
1756
- // These routes manage their own nested outlets; unmounting would disrupt child views.
1757
- if (isEnteringContainerRoute && !isLeavingSpecificRoute) {
1758
- return;
1759
- }
1760
- leavingViewItem.mount = false;
1761
- const viewToUnmount = leavingViewItem;
1762
- setTimeout(() => {
1763
- this.context.unMountViewItem(viewToUnmount);
1764
- this.forceUpdate();
1765
- }, VIEW_UNMOUNT_DELAY_MS);
1766
- }
1767
- /**
1768
- * Cleans up orphaned sibling views after replace actions or push-to-container navigations.
1769
- */
1770
- cleanupOrphanedSiblingViews(routeInfo, enteringViewItem, leavingViewItem) {
1771
- var _a, _b, _c, _d, _e, _f, _g;
1772
- const enteringRoutePath = (_b = (_a = enteringViewItem.reactElement) === null || _a === void 0 ? void 0 : _a.props) === null || _b === void 0 ? void 0 : _b.path;
1773
- if (!enteringRoutePath) {
1774
- return;
1775
- }
1776
- const leavingRoutePath = (_d = (_c = leavingViewItem === null || leavingViewItem === void 0 ? void 0 : leavingViewItem.reactElement) === null || _c === void 0 ? void 0 : _c.props) === null || _d === void 0 ? void 0 : _d.path;
1777
- const isContainerRoute = (path) => path === null || path === void 0 ? void 0 : path.endsWith('/*');
1778
- const isReplaceAction = routeInfo.routeAction === 'replace';
1779
- const isPushToContainer = routeInfo.routeAction === 'push' && routeInfo.routeDirection === 'none' && isContainerRoute(enteringRoutePath);
1780
- if (!isReplaceAction && !isPushToContainer) {
1781
- return;
1782
- }
1783
- // Skip cleanup for tab switches
1784
- const isSameView = enteringViewItem === leavingViewItem;
1785
- const isSameContainerRoute = isContainerRoute(enteringRoutePath) && leavingRoutePath === enteringRoutePath;
1786
- const isNavigatingWithinContainer = isPushToContainer &&
1787
- !leavingViewItem &&
1788
- ((_e = routeInfo.prevRouteLastPathname) === null || _e === void 0 ? void 0 : _e.startsWith(enteringRoutePath.replace(/\/\*$/, '')));
1789
- if (isSameView || isSameContainerRoute || isNavigatingWithinContainer) {
1790
- return;
1791
- }
1792
- const allViewsInOutlet = this.context.getViewItemsForOutlet(this.id);
1793
- const areSiblingRoutes = (path1, path2) => {
1794
- const path1IsRelative = !path1.startsWith('/');
1795
- const path2IsRelative = !path2.startsWith('/');
1796
- if (path1IsRelative && path2IsRelative) {
1797
- const path1Depth = path1.replace(/\/\*$/, '').split('/').filter(Boolean).length;
1798
- const path2Depth = path2.replace(/\/\*$/, '').split('/').filter(Boolean).length;
1799
- return path1Depth === path2Depth && path1Depth <= 1;
1800
- }
1801
- const getParent = (path) => {
1802
- const normalized = path.replace(/\/\*$/, '');
1803
- const segments = normalized.split('/').filter(Boolean);
1804
- // Strip trailing parameter segments (e.g., :id) so that
1805
- // sibling routes like /items/list/:id and /items/detail/:id
1806
- // resolve to the same parent (/items).
1807
- while (segments.length > 0 && segments[segments.length - 1].startsWith(':')) {
1808
- segments.pop();
1809
- }
1810
- segments.pop();
1811
- return segments.length > 0 ? '/' + segments.join('/') : '/';
1812
- };
1813
- const parent = getParent(path1);
1814
- // Exclude root-level routes from sibling detection to avoid unintended
1815
- // cleanup of unrelated top-level routes. Also covers single-depth param
1816
- // routes (e.g., /items/:id) which resolve to root after param stripping.
1817
- if (parent === '/') {
1818
- return false;
1819
- }
1820
- return parent === getParent(path2);
1821
- };
1822
- for (const viewItem of allViewsInOutlet) {
1823
- const viewRoutePath = (_g = (_f = viewItem.reactElement) === null || _f === void 0 ? void 0 : _f.props) === null || _g === void 0 ? void 0 : _g.path;
1824
- const shouldSkip = viewItem.id === enteringViewItem.id ||
1825
- (leavingViewItem && viewItem.id === leavingViewItem.id) ||
1826
- !viewItem.mount ||
1827
- !viewRoutePath ||
1828
- // Don't clean up container routes when entering a container route
1829
- // (e.g., /tabs/* and /settings/* coexist for tab switching)
1830
- (viewRoutePath.endsWith('/*') && enteringRoutePath.endsWith('/*'));
1831
- if (shouldSkip) {
1832
- continue;
1833
- }
1834
- const isOrphanedSpecificRoute = !viewRoutePath.endsWith('/*');
1835
- // Clean up sibling non-container routes that are no longer reachable.
1836
- let shouldCleanup = false;
1837
- if ((isReplaceAction || isPushToContainer) && isOrphanedSpecificRoute) {
1838
- shouldCleanup = areSiblingRoutes(enteringRoutePath, viewRoutePath);
1839
- }
1840
- if (shouldCleanup) {
1841
- hideIonPageElement(viewItem.ionPageElement);
1842
- viewItem.mount = false;
1843
- const viewToRemove = viewItem;
1844
- setTimeout(() => {
1845
- this.context.unMountViewItem(viewToRemove);
1846
- this.forceUpdate();
1847
- }, VIEW_UNMOUNT_DELAY_MS);
1848
- }
1849
- }
1850
- }
1851
- /**
1852
- * Determines whether to skip the transition animation and, if so, immediately
1853
- * hides the leaving view with inline `visibility:hidden`.
1854
- *
1855
- * Skips transitions only for outlets nested inside a parent IonPage's content
1856
- * area (i.e., an ion-content sits between the outlet and the .ion-page). These
1857
- * outlets render child pages inside a parent page's scrollable area, and the MD
1858
- * animation shows both entering and leaving pages simultaneously — causing text
1859
- * overlap and nested scrollbars. Standard page-level outlets (tabs, routing,
1860
- * swipe-to-go-back) animate normally even though they sit inside a framework-
1861
- * managed .ion-page wrapper from the parent outlet's view stack.
1862
- *
1863
- * Uses inline visibility:hidden rather than ion-page-hidden class because
1864
- * core's beforeTransition() removes ion-page-hidden via setPageHidden().
1865
- * Inline visibility:hidden survives that removal, keeping the page hidden
1866
- * until React unmounts it after ionViewDidLeave fires. Unlike display:none,
1867
- * visibility:hidden preserves element geometry so commit() animations
1868
- * can resolve normally.
1869
- */
1870
- applySkipAnimationIfNeeded(enteringViewItem, leavingViewItem) {
1871
- var _a, _b;
1872
- // Only skip for outlets genuinely nested inside a page's content area.
1873
- // Walk from the outlet up to the nearest .ion-page; if an ion-content
1874
- // sits in between, the outlet is inside scrollable page content and
1875
- // animating would cause overlapping pages with duplicate scrollbars.
1876
- let isInsidePageContent = false;
1877
- let el = (_b = (_a = this.routerOutletElement) === null || _a === void 0 ? void 0 : _a.parentElement) !== null && _b !== void 0 ? _b : null;
1878
- while (el) {
1879
- if (el.classList.contains('ion-page'))
1880
- break;
1881
- if (el.tagName === 'ION-CONTENT') {
1882
- isInsidePageContent = true;
1883
- break;
1884
- }
1885
- el = el.parentElement;
1886
- }
1887
- const shouldSkip = isInsidePageContent && !!leavingViewItem && enteringViewItem !== leavingViewItem;
1888
- if (shouldSkip && (leavingViewItem === null || leavingViewItem === void 0 ? void 0 : leavingViewItem.ionPageElement)) {
1889
- leavingViewItem.ionPageElement.style.setProperty('visibility', 'hidden');
1890
- leavingViewItem.ionPageElement.setAttribute('aria-hidden', 'true');
1891
- }
1892
- return shouldSkip;
1893
- }
1894
- /**
1895
- * Handles entering view with no ion-page element yet (waiting for render).
1896
- */
1897
- handleWaitingForIonPage(routeInfo, enteringViewItem, leavingViewItem, shouldUnmountLeavingViewItem) {
1898
- var _a, _b;
1899
- const enteringRouteElement = (_b = (_a = enteringViewItem.reactElement) === null || _a === void 0 ? void 0 : _a.props) === null || _b === void 0 ? void 0 : _b.element;
1900
- // Handle Navigate components (they never render an IonPage)
1901
- if (isNavigateElement(enteringRouteElement)) {
1902
- this.waitingForIonPage = false;
1903
- if (this.ionPageWaitTimeout) {
1904
- clearTimeout(this.ionPageWaitTimeout);
1905
- this.ionPageWaitTimeout = undefined;
1906
- }
1907
- this.pendingPageTransition = false;
1908
- // Hide ALL other visible views in this outlet for Navigate redirects.
1909
- // Same rationale as the timeout path: intermediate redirects can shift
1910
- // the leaving view reference, leaving the original page visible.
1911
- const allViewsInOutlet = this.context.getViewItemsForOutlet(this.id);
1912
- allViewsInOutlet.forEach((viewItem) => {
1913
- if (viewItem.id !== enteringViewItem.id && viewItem.ionPageElement) {
1914
- hideIonPageElement(viewItem.ionPageElement);
1915
- }
1916
- });
1917
- // Don't unmount if entering and leaving are the same view item
1918
- if (shouldUnmountLeavingViewItem && leavingViewItem && enteringViewItem !== leavingViewItem) {
1919
- if (routeInfo.routeAction !== 'replace') {
1920
- leavingViewItem.mount = false;
1921
- }
1922
- this.handleLeavingViewUnmount(routeInfo, enteringViewItem, leavingViewItem);
1923
- }
1924
- this.forceUpdate();
1925
- return;
1926
- }
1927
- // Do not hide the leaving view here - wait until the entering view is ready.
1928
- // Hiding the leaving view while the entering view is still mounting causes a flash
1929
- // where both views are hidden/invisible simultaneously.
1930
- // The leaving view will be hidden in transitionPage() after the entering view is visible.
1931
- this.waitingForIonPage = true;
1932
- if (this.ionPageWaitTimeout) {
1933
- clearTimeout(this.ionPageWaitTimeout);
1934
- }
1935
- this.ionPageWaitTimeout = setTimeout(() => {
1936
- var _a, _b;
1937
- this.ionPageWaitTimeout = undefined;
1938
- if (!this.waitingForIonPage) {
1939
- return;
1940
- }
1941
- this.waitingForIonPage = false;
1942
- const latestEnteringView = (_a = this.context.findViewItemByRouteInfo(routeInfo, this.id)) !== null && _a !== void 0 ? _a : enteringViewItem;
1943
- const latestLeavingView = (_b = this.context.findLeavingViewItemByRouteInfo(routeInfo, this.id)) !== null && _b !== void 0 ? _b : leavingViewItem;
1944
- if (latestEnteringView === null || latestEnteringView === void 0 ? void 0 : latestEnteringView.ionPageElement) {
1945
- const shouldSkipAnimation = this.applySkipAnimationIfNeeded(latestEnteringView, latestLeavingView !== null && latestLeavingView !== void 0 ? latestLeavingView : undefined);
1946
- this.transitionPage(routeInfo, latestEnteringView, latestLeavingView !== null && latestLeavingView !== void 0 ? latestLeavingView : undefined, undefined, false, shouldSkipAnimation);
1947
- if (shouldUnmountLeavingViewItem && latestLeavingView && latestEnteringView !== latestLeavingView) {
1948
- if (routeInfo.routeAction !== 'replace') {
1949
- latestLeavingView.mount = false;
1950
- }
1951
- this.handleLeavingViewUnmount(routeInfo, latestEnteringView, latestLeavingView);
1952
- }
1953
- this.forceUpdate();
1954
- }
1955
- else {
1956
- /**
1957
- * Timeout fired and entering view still has no ionPageElement.
1958
- * This happens for container routes that render nested outlets without a direct IonPage.
1959
- * Hide ALL other visible views in this outlet, not just the computed leaving view.
1960
- * This handles cases where intermediate redirects (e.g., Navigate in nested routes)
1961
- * change the leaving view reference, leaving the original page still visible.
1962
- */
1963
- const allViewsInOutlet = this.context.getViewItemsForOutlet(this.id);
1964
- allViewsInOutlet.forEach((viewItem) => {
1965
- if (viewItem.id !== latestEnteringView.id && viewItem.ionPageElement) {
1966
- hideIonPageElement(viewItem.ionPageElement);
1967
- }
1968
- });
1969
- this.forceUpdate();
1970
- // Safety net: after forceUpdate triggers a React render cycle, check if
1971
- // any pages in this outlet are stuck with ion-page-invisible. This can
1972
- // happen when view lookup fails (e.g., wildcard-to-index transitions
1973
- // where the view item gets corrupted). The forceUpdate above causes
1974
- // React to render the correct component, but ion-page-invisible may
1975
- // persist if no transition runs for that page.
1976
- setTimeout(() => {
1977
- if (!this._isMounted || !this.routerOutletElement)
1978
- return;
1979
- const stuckPages = this.routerOutletElement.querySelectorAll(':scope > .ion-page-invisible');
1980
- stuckPages.forEach((page) => {
1981
- page.classList.remove('ion-page-invisible');
1982
- });
1983
- }, ION_PAGE_WAIT_TIMEOUT_MS);
1984
- }
1985
- }, ION_PAGE_WAIT_TIMEOUT_MS);
1986
- this.forceUpdate();
1987
- }
1988
- /**
1989
- * Gets the route info to use for finding views during swipe-to-go-back gestures.
1990
- * This pattern is used in multiple places in setupRouterOutlet.
1991
- */
1992
- getSwipeBackRouteInfo() {
1993
- const { routeInfo } = this.props;
1994
- return this.prevProps && this.prevProps.routeInfo.pathname === routeInfo.pushedByRoute
1995
- ? this.prevProps.routeInfo
1996
- : { pathname: routeInfo.pushedByRoute || '' };
1997
- }
1998
227
  componentDidMount() {
1999
- this._isMounted = true;
228
+ if (this.clearOutletTimeout) {
229
+ /**
230
+ * The clearOutlet integration with React Router is a bit hacky.
231
+ * It uses a timeout to clear the outlet after a transition.
232
+ * In React v18, components are mounted and unmounted in development mode
233
+ * to check for side effects.
234
+ *
235
+ * This clearTimeout prevents the outlet from being cleared when the component is re-mounted,
236
+ * which should only happen in development mode and as a result of a hot reload.
237
+ */
238
+ clearTimeout(this.clearOutletTimeout);
239
+ }
2000
240
  if (this.routerOutletElement) {
2001
241
  this.setupRouterOutlet(this.routerOutletElement);
2002
- // Defer to a microtask to avoid calling forceUpdate() synchronously during
2003
- // React 19's reappearLayoutEffects phase, which re-runs componentDidMount
2004
- // without a preceding componentWillUnmount and causes "Maximum update depth exceeded".
2005
- const routeInfo = this.props.routeInfo;
2006
- queueMicrotask(() => {
2007
- if (this._isMounted && this.props.routeInfo.pathname === routeInfo.pathname) {
2008
- this.handlePageTransition(routeInfo);
2009
- }
2010
- });
242
+ this.handlePageTransition(this.props.routeInfo);
2011
243
  }
2012
244
  }
2013
245
  componentDidUpdate(prevProps) {
@@ -2023,192 +255,117 @@ class StackManager extends React.PureComponent {
2023
255
  }
2024
256
  }
2025
257
  componentWillUnmount() {
2026
- this._isMounted = false;
2027
- // Cancel any in-flight transition rAFs
2028
- for (const id of this.transitionRafIds) {
2029
- cancelAnimationFrame(id);
2030
- }
2031
- this.transitionRafIds = [];
2032
- // Disconnect any in-flight MutationObserver from waitForComponentsReady
2033
- if (this.transitionObserver) {
2034
- this.transitionObserver.disconnect();
2035
- this.transitionObserver = undefined;
2036
- }
2037
- if (this.ionPageWaitTimeout) {
2038
- clearTimeout(this.ionPageWaitTimeout);
2039
- this.ionPageWaitTimeout = undefined;
2040
- }
2041
- if (this.outOfScopeUnmountTimeout) {
2042
- clearTimeout(this.outOfScopeUnmountTimeout);
2043
- this.outOfScopeUnmountTimeout = undefined;
2044
- }
2045
- this.waitingForIonPage = false;
2046
- // Hide all views in this outlet before clearing.
2047
- // This is critical for nested outlets - when the parent component unmounts,
2048
- // the nested outlet's componentDidUpdate won't be called, so we must hide
2049
- // the ion-page elements here to prevent them from remaining visible on top
2050
- // of other content after navigation to a different route.
2051
- const allViewsInOutlet = this.context.getViewItemsForOutlet(this.id);
2052
- allViewsInOutlet.forEach((viewItem) => {
2053
- hideIonPageElement(viewItem.ionPageElement);
2054
- });
2055
- this.context.clearOutlet(this.id);
258
+ this.clearOutletTimeout = this.context.clearOutlet(this.id);
2056
259
  }
2057
- /**
2058
- * Sets the transition between pages within this router outlet.
2059
- * This function determines the entering and leaving views based on the
2060
- * provided route information and triggers the appropriate animation.
2061
- * It also handles scenarios like initial loads, back navigation, and
2062
- * navigation to the same view with different parameters.
2063
- *
2064
- * @param routeInfo It contains info about the current route,
2065
- * the previous route, and the action taken (e.g., push, replace).
2066
- *
2067
- * @returns A promise that resolves when the transition is complete.
2068
- * If no transition is needed or if the router outlet isn't ready,
2069
- * the Promise may resolve immediately.
2070
- */
2071
260
  async handlePageTransition(routeInfo) {
2072
- var _a;
2073
- // Wait for router outlet to mount
261
+ var _a, _b;
2074
262
  if (!this.routerOutletElement || !this.routerOutletElement.commit) {
263
+ /**
264
+ * The route outlet has not mounted yet. We need to wait for it to render
265
+ * before we can transition the page.
266
+ *
267
+ * Set a flag to indicate that we should transition the page after
268
+ * the component has updated.
269
+ */
2075
270
  this.pendingPageTransition = true;
2076
- return;
2077
- }
2078
- // Find entering and leaving view items
2079
- const viewItems = this.findViewItems(routeInfo);
2080
- let enteringViewItem = viewItems.enteringViewItem;
2081
- let leavingViewItem = viewItems.leavingViewItem;
2082
- let shouldUnmountLeavingViewItem = this.shouldUnmountLeavingView(routeInfo, enteringViewItem, leavingViewItem);
2083
- // Get parent path for nested outlets
2084
- const parentPath = this.getParentPath();
2085
- // Handle out-of-scope outlet (route outside mount path)
2086
- if (this.handleOutOfScopeOutlet(routeInfo)) {
2087
- return;
2088
- }
2089
- // Handle root navigation: unmount all non-entering views
2090
- if (routeInfo.routeDirection === 'root') {
2091
- this.handleRootNavigation(enteringViewItem);
2092
- leavingViewItem = undefined;
2093
- shouldUnmountLeavingViewItem = false;
2094
- }
2095
- // Clear any pending out-of-scope unmount timeout
2096
- if (this.outOfScopeUnmountTimeout) {
2097
- clearTimeout(this.outOfScopeUnmountTimeout);
2098
- this.outOfScopeUnmountTimeout = undefined;
2099
271
  }
2100
- // Handle nested outlet with relative routes but no valid parent path
2101
- if (this.handleOutOfContextNestedOutlet(parentPath, leavingViewItem)) {
2102
- return;
2103
- }
2104
- // Find the matching route element
2105
- const enteringRoute = findRouteByRouteInfo((_a = this.ionRouterOutlet) === null || _a === void 0 ? void 0 : _a.props.children, routeInfo, parentPath);
2106
- // Handle nested outlet with no matching route
2107
- if (this.handleNoMatchingRoute(enteringRoute, enteringViewItem, leavingViewItem)) {
2108
- return;
2109
- }
2110
- // Create or update the entering view item
2111
- if (enteringViewItem && enteringRoute) {
2112
- enteringViewItem.reactElement = enteringRoute;
2113
- }
2114
- else if (enteringRoute) {
2115
- enteringViewItem = this.context.createViewItem(this.id, enteringRoute, routeInfo);
2116
- this.context.addViewItem(enteringViewItem);
2117
- }
2118
- // Handle transition based on ion-page element availability
2119
- // Check if the ionPageElement is still in the document.
2120
- // If the view was previously unmounted (mount=false), the ViewLifeCycleManager
2121
- // removes the React component from the tree, which removes the IonPage from the DOM.
2122
- // The ionPageElement reference becomes stale and we need to wait for a new one.
2123
- const ionPageIsInDocument = (enteringViewItem === null || enteringViewItem === void 0 ? void 0 : enteringViewItem.ionPageElement) && document.body.contains(enteringViewItem.ionPageElement);
2124
- if (enteringViewItem && ionPageIsInDocument) {
2125
- // Clear waiting state
2126
- if (this.waitingForIonPage) {
2127
- this.waitingForIonPage = false;
272
+ else {
273
+ let enteringViewItem = this.context.findViewItemByRouteInfo(routeInfo, this.id);
274
+ let leavingViewItem = this.context.findLeavingViewItemByRouteInfo(routeInfo, this.id);
275
+ if (!leavingViewItem && routeInfo.prevRouteLastPathname) {
276
+ leavingViewItem = this.context.findViewItemByPathname(routeInfo.prevRouteLastPathname, this.id);
277
+ }
278
+ // Check if leavingViewItem should be unmounted
279
+ if (leavingViewItem) {
280
+ if (routeInfo.routeAction === 'replace') {
281
+ leavingViewItem.mount = false;
282
+ }
283
+ else if (!(routeInfo.routeAction === 'push' && routeInfo.routeDirection === 'forward')) {
284
+ if (routeInfo.routeDirection !== 'none' && enteringViewItem !== leavingViewItem) {
285
+ leavingViewItem.mount = false;
286
+ }
287
+ }
288
+ else if ((_a = routeInfo.routeOptions) === null || _a === void 0 ? void 0 : _a.unmount) {
289
+ leavingViewItem.mount = false;
290
+ }
2128
291
  }
2129
- if (this.ionPageWaitTimeout) {
2130
- clearTimeout(this.ionPageWaitTimeout);
2131
- this.ionPageWaitTimeout = undefined;
292
+ const enteringRoute = matchRoute((_b = this.ionRouterOutlet) === null || _b === void 0 ? void 0 : _b.props.children, routeInfo);
293
+ if (enteringViewItem) {
294
+ enteringViewItem.reactElement = enteringRoute;
2132
295
  }
2133
- this.handleReadyEnteringView(routeInfo, enteringViewItem, leavingViewItem, shouldUnmountLeavingViewItem);
2134
- }
2135
- else if (enteringViewItem && !ionPageIsInDocument) {
2136
- // Wait for ion-page to mount
2137
- // This handles both: no ionPageElement, or stale ionPageElement (not in document)
2138
- // Clear stale reference if the element is no longer in the document
2139
- if (enteringViewItem.ionPageElement && !document.body.contains(enteringViewItem.ionPageElement)) {
2140
- enteringViewItem.ionPageElement = undefined;
296
+ else if (enteringRoute) {
297
+ enteringViewItem = this.context.createViewItem(this.id, enteringRoute, routeInfo);
298
+ this.context.addViewItem(enteringViewItem);
2141
299
  }
2142
- // Ensure the view is marked as mounted so ViewLifeCycleManager renders the IonPage
2143
- if (!enteringViewItem.mount) {
2144
- enteringViewItem.mount = true;
300
+ if (enteringViewItem && enteringViewItem.ionPageElement) {
301
+ /**
302
+ * If the entering view item is the same as the leaving view item,
303
+ * then we don't need to transition.
304
+ */
305
+ if (enteringViewItem === leavingViewItem) {
306
+ /**
307
+ * If the entering view item is the same as the leaving view item,
308
+ * we are either transitioning using parameterized routes to the same view
309
+ * or a parent router outlet is re-rendering as a result of React props changing.
310
+ *
311
+ * If the route data does not match the current path, the parent router outlet
312
+ * is attempting to transition and we cancel the operation.
313
+ */
314
+ if (enteringViewItem.routeData.match.url !== routeInfo.pathname) {
315
+ return;
316
+ }
317
+ }
318
+ /**
319
+ * If there isn't a leaving view item, but the route info indicates
320
+ * that the user has routed from a previous path, then we need
321
+ * to find the leaving view item to transition between.
322
+ */
323
+ if (!leavingViewItem && this.props.routeInfo.prevRouteLastPathname) {
324
+ leavingViewItem = this.context.findViewItemByPathname(this.props.routeInfo.prevRouteLastPathname, this.id);
325
+ }
326
+ /**
327
+ * If the entering view is already visible and the leaving view is not, the transition does not need to occur.
328
+ */
329
+ if (isViewVisible(enteringViewItem.ionPageElement) &&
330
+ leavingViewItem !== undefined &&
331
+ !isViewVisible(leavingViewItem.ionPageElement)) {
332
+ return;
333
+ }
334
+ /**
335
+ * The view should only be transitioned in the following cases:
336
+ * 1. Performing a replace or pop action, such as a swipe to go back gesture
337
+ * to animation the leaving view off the screen.
338
+ *
339
+ * 2. Navigating between top-level router outlets, such as /page-1 to /page-2;
340
+ * or navigating within a nested outlet, such as /tabs/tab-1 to /tabs/tab-2.
341
+ *
342
+ * 3. The entering view is an ion-router-outlet containing a page
343
+ * matching the current route and that hasn't already transitioned in.
344
+ *
345
+ * This should only happen when navigating directly to a nested router outlet
346
+ * route or on an initial page load (i.e. refreshing). In cases when loading
347
+ * /tabs/tab-1, we need to transition the /tabs page element into the view.
348
+ */
349
+ this.transitionPage(routeInfo, enteringViewItem, leavingViewItem);
2145
350
  }
2146
- this.handleWaitingForIonPage(routeInfo, enteringViewItem, leavingViewItem, shouldUnmountLeavingViewItem);
2147
- return;
2148
- }
2149
- else if (!enteringViewItem && !enteringRoute) {
2150
- // No view or route found - likely leaving to another outlet
2151
- if (leavingViewItem) {
2152
- hideIonPageElement(leavingViewItem.ionPageElement);
2153
- if (shouldUnmountLeavingViewItem) {
2154
- leavingViewItem.mount = false;
351
+ else if (leavingViewItem && !enteringRoute && !enteringViewItem) {
352
+ // If we have a leavingView but no entering view/route, we are probably leaving to
353
+ // another outlet, so hide this leavingView. We do it in a timeout to give time for a
354
+ // transition to finish.
355
+ // setTimeout(() => {
356
+ if (leavingViewItem.ionPageElement) {
357
+ leavingViewItem.ionPageElement.classList.add('ion-page-hidden');
358
+ leavingViewItem.ionPageElement.setAttribute('aria-hidden', 'true');
2155
359
  }
360
+ // }, 250);
2156
361
  }
362
+ this.forceUpdate();
2157
363
  }
2158
- this.forceUpdate();
2159
364
  }
2160
- /**
2161
- * Registers an `<IonPage>` DOM element with the `StackManager`.
2162
- * This is called when `<IonPage>` has been mounted.
2163
- *
2164
- * @param page The element of the rendered `<IonPage>`.
2165
- * @param routeInfo The route information that associates with `<IonPage>`.
2166
- */
2167
365
  registerIonPage(page, routeInfo) {
2168
- /**
2169
- * DO NOT remove ion-page-invisible here.
2170
- *
2171
- * PageManager's ref callback adds ion-page-invisible synchronously to prevent flash.
2172
- * At this point, the <IonPage> div exists but its CHILDREN (header, toolbar, menu-button)
2173
- * have NOT rendered yet. If we remove ion-page-invisible now, the page becomes visible
2174
- * with empty/incomplete content, causing a flicker (especially for ion-menu-button which
2175
- * starts with menu-button-hidden class).
2176
- *
2177
- * Instead, let transitionPage handle visibility AFTER waiting for components to be ready.
2178
- * This ensures the page only becomes visible when its content is fully rendered.
2179
- */
2180
- this.waitingForIonPage = false;
2181
- if (this.ionPageWaitTimeout) {
2182
- clearTimeout(this.ionPageWaitTimeout);
2183
- this.ionPageWaitTimeout = undefined;
2184
- }
2185
- this.pendingPageTransition = false;
2186
366
  const foundView = this.context.findViewItemByRouteInfo(routeInfo, this.id);
2187
367
  if (foundView) {
2188
368
  const oldPageElement = foundView.ionPageElement;
2189
- /**
2190
- * FIX for issue #28878: Reject orphaned IonPage registrations.
2191
- *
2192
- * When a component conditionally renders different IonPages (e.g., list vs empty state)
2193
- * using React keys, and state changes simultaneously with navigation, the new IonPage
2194
- * tries to register for a route we're navigating away from. This creates a stale view.
2195
- *
2196
- * Only reject if both pageIds exist and differ, to allow nested outlet registrations.
2197
- */
2198
- if (this.shouldRejectOrphanedPage(page, oldPageElement, routeInfo)) {
2199
- this.hideAndRemoveOrphanedPage(page);
2200
- return;
2201
- }
2202
- /**
2203
- * Don't let a nested element (e.g., ion-router-outlet with ionPage prop)
2204
- * override an existing IonPage registration when the existing element is
2205
- * an ancestor of the new one. This ensures ionPageElement always points
2206
- * to the outermost IonPage, which is needed to properly hide the entire
2207
- * page during back navigation (not just the inner outlet).
2208
- */
2209
- if (oldPageElement && oldPageElement !== page && oldPageElement.isConnected && oldPageElement.contains(page)) {
2210
- return;
2211
- }
2212
369
  foundView.ionPageElement = page;
2213
370
  foundView.ionRoute = true;
2214
371
  /**
@@ -2222,32 +379,6 @@ class StackManager extends React.PureComponent {
2222
379
  }
2223
380
  this.handlePageTransition(routeInfo);
2224
381
  }
2225
- /**
2226
- * Checks if a new IonPage should be rejected (component re-rendered while navigating away).
2227
- */
2228
- shouldRejectOrphanedPage(newPage, oldPageElement, routeInfo) {
2229
- if (!oldPageElement || oldPageElement === newPage) {
2230
- return false;
2231
- }
2232
- const newPageId = newPage.getAttribute('data-pageid');
2233
- const oldPageId = oldPageElement.getAttribute('data-pageid');
2234
- if (!newPageId || !oldPageId || newPageId === oldPageId) {
2235
- return false;
2236
- }
2237
- return this.props.routeInfo.pathname !== routeInfo.pathname;
2238
- }
2239
- hideAndRemoveOrphanedPage(page) {
2240
- page.classList.add('ion-page-hidden');
2241
- page.setAttribute('aria-hidden', 'true');
2242
- setTimeout(() => {
2243
- if (page.parentElement) {
2244
- page.remove();
2245
- }
2246
- }, VIEW_UNMOUNT_DELAY_MS);
2247
- }
2248
- /**
2249
- * Configures swipe-to-go-back gesture for the router outlet.
2250
- */
2251
382
  async setupRouterOutlet(routerOutlet) {
2252
383
  const canStart = () => {
2253
384
  const config = getConfig();
@@ -2256,40 +387,40 @@ class StackManager extends React.PureComponent {
2256
387
  return false;
2257
388
  }
2258
389
  const { routeInfo } = this.props;
2259
- const swipeBackRouteInfo = this.getSwipeBackRouteInfo();
2260
- let enteringViewItem = this.context.findViewItemByRouteInfo(swipeBackRouteInfo, this.id, false);
2261
- if (!enteringViewItem) {
2262
- enteringViewItem = this.context.findViewItemByRouteInfo(swipeBackRouteInfo, undefined, false);
2263
- }
2264
- // View might have mount=false but ionPageElement still in DOM
2265
- const ionPageInDocument = Boolean((enteringViewItem === null || enteringViewItem === void 0 ? void 0 : enteringViewItem.ionPageElement) && document.body.contains(enteringViewItem.ionPageElement));
2266
- // For wildcard/parameterized routes, the pattern path (e.g. "/foo/*") will
2267
- // never equal the resolved pathname (e.g. "/foo/bar"), so the pattern check
2268
- // alone isn't sufficient. Also, verify the entering view's resolved pathname
2269
- // differs from the current pathname — if they match, the entering and leaving
2270
- // views are the same and the swipe gesture shouldn't start.
2271
- const canStartSwipe = !!enteringViewItem &&
2272
- (enteringViewItem.mount || ionPageInDocument) &&
2273
- enteringViewItem.routeData.match.pattern.path !== routeInfo.pathname &&
2274
- enteringViewItem.routeData.match.pathname !== routeInfo.pathname;
2275
- return canStartSwipe;
390
+ const propsToUse = this.prevProps && this.prevProps.routeInfo.pathname === routeInfo.pushedByRoute
391
+ ? this.prevProps.routeInfo
392
+ : { pathname: routeInfo.pushedByRoute || '' };
393
+ const enteringViewItem = this.context.findViewItemByRouteInfo(propsToUse, this.id, false);
394
+ return (!!enteringViewItem &&
395
+ /**
396
+ * The root url '/' is treated as
397
+ * the first view item (but is never mounted),
398
+ * so we do not want to swipe back to the
399
+ * root url.
400
+ */
401
+ enteringViewItem.mount &&
402
+ /**
403
+ * When on the first page (whatever view
404
+ * you land on after the root url) it
405
+ * is possible for findViewItemByRouteInfo to
406
+ * return the exact same view you are currently on.
407
+ * Make sure that we are not swiping back to the same
408
+ * instances of a view.
409
+ */
410
+ enteringViewItem.routeData.match.path !== routeInfo.pathname);
2276
411
  };
2277
412
  const onStart = async () => {
2278
413
  const { routeInfo } = this.props;
2279
- const swipeBackRouteInfo = this.getSwipeBackRouteInfo();
2280
- // First try to find the view in the current outlet, then search all outlets
2281
- let enteringViewItem = this.context.findViewItemByRouteInfo(swipeBackRouteInfo, this.id, false);
2282
- if (!enteringViewItem) {
2283
- enteringViewItem = this.context.findViewItemByRouteInfo(swipeBackRouteInfo, undefined, false);
2284
- }
414
+ const propsToUse = this.prevProps && this.prevProps.routeInfo.pathname === routeInfo.pushedByRoute
415
+ ? this.prevProps.routeInfo
416
+ : { pathname: routeInfo.pushedByRoute || '' };
417
+ const enteringViewItem = this.context.findViewItemByRouteInfo(propsToUse, this.id, false);
2285
418
  const leavingViewItem = this.context.findViewItemByRouteInfo(routeInfo, this.id, false);
2286
- // Ensure the entering view is mounted so React keeps rendering it during the gesture.
2287
- // This is important when the view was previously marked for unmount but its
2288
- // ionPageElement is still in the DOM.
2289
- if (enteringViewItem && !enteringViewItem.mount) {
2290
- enteringViewItem.mount = true;
2291
- }
2292
- // When the gesture starts, kick off a transition controlled via swipe gesture
419
+ /**
420
+ * When the gesture starts, kick off
421
+ * a transition that is controlled
422
+ * via a swipe gesture.
423
+ */
2293
424
  if (enteringViewItem && leavingViewItem) {
2294
425
  await this.transitionPage(routeInfo, enteringViewItem, leavingViewItem, 'back', true);
2295
426
  }
@@ -2297,23 +428,34 @@ class StackManager extends React.PureComponent {
2297
428
  };
2298
429
  const onEnd = (shouldContinue) => {
2299
430
  if (shouldContinue) {
2300
- // User finished the swipe gesture, so complete the back navigation
2301
431
  this.skipTransition = true;
2302
432
  this.context.goBack();
2303
433
  }
2304
434
  else {
2305
- // Swipe gesture was aborted - re-hide the page that was going to enter
435
+ /**
436
+ * In the event that the swipe
437
+ * gesture was aborted, we should
438
+ * re-hide the page that was going to enter.
439
+ */
2306
440
  const { routeInfo } = this.props;
2307
- const swipeBackRouteInfo = this.getSwipeBackRouteInfo();
2308
- // First try to find the view in the current outlet, then search all outlets
2309
- let enteringViewItem = this.context.findViewItemByRouteInfo(swipeBackRouteInfo, this.id, false);
2310
- if (!enteringViewItem) {
2311
- enteringViewItem = this.context.findViewItemByRouteInfo(swipeBackRouteInfo, undefined, false);
2312
- }
441
+ const propsToUse = this.prevProps && this.prevProps.routeInfo.pathname === routeInfo.pushedByRoute
442
+ ? this.prevProps.routeInfo
443
+ : { pathname: routeInfo.pushedByRoute || '' };
444
+ const enteringViewItem = this.context.findViewItemByRouteInfo(propsToUse, this.id, false);
2313
445
  const leavingViewItem = this.context.findViewItemByRouteInfo(routeInfo, this.id, false);
2314
- // Don't hide if entering and leaving are the same (parameterized route edge case)
446
+ /**
447
+ * Ionic React has a design defect where it
448
+ * a) Unmounts the leaving view item when using parameterized routes
449
+ * b) Considers the current view to be the entering view when using
450
+ * parameterized routes
451
+ *
452
+ * As a result, we should not hide the view item here
453
+ * as it will cause the current view to be hidden.
454
+ */
2315
455
  if (enteringViewItem !== leavingViewItem && (enteringViewItem === null || enteringViewItem === void 0 ? void 0 : enteringViewItem.ionPageElement) !== undefined) {
2316
- hideIonPageElement(enteringViewItem.ionPageElement);
456
+ const { ionPageElement } = enteringViewItem;
457
+ ionPageElement.setAttribute('aria-hidden', 'true');
458
+ ionPageElement.classList.add('ion-page-hidden');
2317
459
  }
2318
460
  }
2319
461
  };
@@ -2323,24 +465,7 @@ class StackManager extends React.PureComponent {
2323
465
  onEnd,
2324
466
  };
2325
467
  }
2326
- /**
2327
- * Animates the transition between the entering and leaving pages within the
2328
- * router outlet.
2329
- *
2330
- * @param routeInfo Info about the current route.
2331
- * @param enteringViewItem The view item that is entering.
2332
- * @param leavingViewItem The view item that is leaving.
2333
- * @param direction The direction of the transition.
2334
- * @param progressAnimation Indicates if the transition is part of a
2335
- * gesture controlled animation (e.g., swipe to go back).
2336
- * Defaults to `false`.
2337
- * @param skipAnimation When true, forces `duration: 0` so the page
2338
- * swap is instant (no visible animation). Used for ionPage outlets
2339
- * and back navigations that unmount the leaving view to prevent
2340
- * overlapping content during the transition. Defaults to `false`.
2341
- */
2342
- async transitionPage(routeInfo, enteringViewItem, leavingViewItem, direction, progressAnimation = false, skipAnimation = false) {
2343
- const myGeneration = ++this.transitionGeneration;
468
+ async transitionPage(routeInfo, enteringViewItem, leavingViewItem, direction, progressAnimation = false) {
2344
469
  const runCommit = async (enteringEl, leavingEl) => {
2345
470
  const skipTransition = this.skipTransition;
2346
471
  /**
@@ -2368,56 +493,24 @@ class StackManager extends React.PureComponent {
2368
493
  }
2369
494
  else {
2370
495
  enteringEl.classList.add('ion-page');
2371
- /**
2372
- * Only add ion-page-invisible if the element is not already visible.
2373
- * During tab switches, the container page (e.g., TabContext wrapper) is
2374
- * already visible and should remain so. Adding ion-page-invisible would
2375
- * cause a flash where the visible page briefly becomes invisible.
2376
- */
2377
- if (!isViewVisible(enteringEl)) {
2378
- enteringEl.classList.add('ion-page-invisible');
2379
- }
496
+ enteringEl.classList.add('ion-page-invisible');
2380
497
  }
2381
- const commitDuration = skipTransition || skipAnimation || directionToUse === undefined ? 0 : undefined;
2382
- // Race commit against a timeout to recover from hangs
2383
- const commitPromise = routerOutlet.commit(enteringEl, leavingEl, {
2384
- duration: commitDuration,
498
+ await routerOutlet.commit(enteringEl, leavingEl, {
499
+ duration: skipTransition || directionToUse === undefined ? 0 : undefined,
2385
500
  direction: directionToUse,
2386
501
  showGoBack: !!routeInfo.pushedByRoute,
2387
502
  progressAnimation,
2388
503
  animationBuilder: routeInfo.routeAnimation,
2389
504
  });
2390
- const timeoutMs = 5000;
2391
- const timeoutPromise = new Promise((resolve) => setTimeout(() => resolve('timeout'), timeoutMs));
2392
- const result = await Promise.race([commitPromise.then(() => 'done'), timeoutPromise]);
2393
- // Bail out if the component unmounted during the commit animation
2394
- if (!this._isMounted)
2395
- return;
2396
- if (result === 'timeout') {
2397
- // Force entering page visible even though commit hung
2398
- enteringEl.classList.remove('ion-page-invisible');
2399
- }
2400
- /**
2401
- * If a newer transitionPage call ran while this commit was in-flight (e.g., a tab
2402
- * switch fired during a forward animation), the core commit may have applied
2403
- * ion-page-hidden to leavingEl even though the newer transition already made it
2404
- * visible. Undo that stale hide so the newer transition's DOM state wins.
2405
- */
2406
- if (myGeneration !== this.transitionGeneration && leavingEl && leavingEl === this.transitionEnteringElement) {
2407
- showIonPageElement(leavingEl);
2408
- }
2409
- if (!progressAnimation) {
2410
- enteringEl.classList.remove('ion-page-invisible');
2411
- }
2412
505
  };
2413
506
  const routerOutlet = this.routerOutletElement;
2414
507
  const routeInfoFallbackDirection = routeInfo.routeDirection === 'none' || routeInfo.routeDirection === 'root' ? undefined : routeInfo.routeDirection;
2415
508
  const directionToUse = direction !== null && direction !== void 0 ? direction : routeInfoFallbackDirection;
2416
509
  if (enteringViewItem && enteringViewItem.ionPageElement && this.routerOutletElement) {
2417
- this.transitionEnteringElement = enteringViewItem.ionPageElement;
2418
510
  if (leavingViewItem && leavingViewItem.ionPageElement && enteringViewItem === leavingViewItem) {
2419
- // Clone page for same-view transitions (e.g., /user/1 /user/2)
2420
- const match = matchComponent(leavingViewItem.reactElement, routeInfo.pathname, undefined, this.outletMountPath);
511
+ // If a page is transitioning to another version of itself
512
+ // we clone it so we can have an animation to show
513
+ const match = matchComponent(leavingViewItem.reactElement, routeInfo.pathname, true);
2421
514
  if (match) {
2422
515
  const newLeavingElement = clonePageElement(leavingViewItem.ionPageElement.outerHTML);
2423
516
  if (newLeavingElement) {
@@ -2427,111 +520,14 @@ class StackManager extends React.PureComponent {
2427
520
  }
2428
521
  }
2429
522
  else {
2430
- // Route no longer matches (e.g., /user/1 → /settings)
2431
523
  await runCommit(enteringViewItem.ionPageElement, undefined);
2432
524
  }
2433
525
  }
2434
526
  else {
2435
- const leavingEl = leavingViewItem === null || leavingViewItem === void 0 ? void 0 : leavingViewItem.ionPageElement;
2436
- // For non-animated transitions, don't pass leaving element to commit() to avoid
2437
- // flicker caused by commit() briefly unhiding the leaving page
2438
- const isNonAnimatedTransition = directionToUse === undefined && !progressAnimation;
2439
- if (isNonAnimatedTransition && leavingEl) {
2440
- /**
2441
- * Skip commit() for non-animated transitions (like tab switches).
2442
- * commit() runs animation logic that can cause intermediate paints
2443
- * even with duration: 0. Instead, swap visibility synchronously.
2444
- *
2445
- * Synchronous DOM class changes are batched into a single browser
2446
- * paint, so there's no gap frame where neither page is visible and
2447
- * no overlap frame where both pages are visible.
2448
- */
2449
- const enteringEl = enteringViewItem.ionPageElement;
2450
- // Ensure entering element has proper base classes
2451
- enteringEl.classList.add('ion-page');
2452
- // Clear ALL hidden state from entering element. showIonPageElement
2453
- // removes visibility:hidden (from applySkipAnimationIfNeeded),
2454
- // ion-page-hidden, and aria-hidden in one call.
2455
- showIonPageElement(enteringEl);
2456
- // Handle can-go-back class since we're skipping commit() which normally sets this
2457
- if (routeInfo.pushedByRoute) {
2458
- enteringEl.classList.add('can-go-back');
2459
- }
2460
- else {
2461
- enteringEl.classList.remove('can-go-back');
2462
- }
2463
- /**
2464
- * Wait for components to be ready. Menu buttons start hidden (menu-button-hidden)
2465
- * and become visible after componentDidLoad. Wait for hydration and visibility.
2466
- */
2467
- const waitForComponentsReady = () => {
2468
- return new Promise((resolve) => {
2469
- const checkReady = () => {
2470
- const ionicComponents = enteringEl.querySelectorAll('ion-header, ion-toolbar, ion-buttons, ion-menu-button, ion-title, ion-content');
2471
- const allHydrated = Array.from(ionicComponents).every((el) => el.classList.contains('hydrated'));
2472
- const menuButtons = enteringEl.querySelectorAll('ion-menu-button');
2473
- const menuButtonsReady = Array.from(menuButtons).every((el) => !el.classList.contains('menu-button-hidden'));
2474
- return allHydrated && menuButtonsReady;
2475
- };
2476
- if (checkReady()) {
2477
- resolve();
2478
- return;
2479
- }
2480
- let resolved = false;
2481
- const observer = new MutationObserver(() => {
2482
- if (!resolved && checkReady()) {
2483
- resolved = true;
2484
- observer.disconnect();
2485
- if (this.transitionObserver === observer) {
2486
- this.transitionObserver = undefined;
2487
- }
2488
- resolve();
2489
- }
2490
- });
2491
- // Disconnect any previous observer before tracking the new one
2492
- if (this.transitionObserver) {
2493
- this.transitionObserver.disconnect();
2494
- }
2495
- this.transitionObserver = observer;
2496
- observer.observe(enteringEl, {
2497
- subtree: true,
2498
- attributes: true,
2499
- attributeFilter: ['class'],
2500
- });
2501
- setTimeout(() => {
2502
- if (!resolved) {
2503
- resolved = true;
2504
- observer.disconnect();
2505
- if (this.transitionObserver === observer) {
2506
- this.transitionObserver = undefined;
2507
- }
2508
- resolve();
2509
- }
2510
- }, 100);
2511
- });
2512
- };
2513
- await waitForComponentsReady();
2514
- // Bail out if the component unmounted during waitForComponentsReady
2515
- if (!this._isMounted)
2516
- return;
2517
- // Swap visibility synchronously - show entering, hide leaving
2518
- // Skip hiding if a newer transition already made leavingEl the entering view
2519
- enteringEl.classList.remove('ion-page-invisible');
2520
- if (myGeneration === this.transitionGeneration || leavingEl !== this.transitionEnteringElement) {
2521
- leavingEl.classList.add('ion-page-hidden');
2522
- leavingEl.setAttribute('aria-hidden', 'true');
2523
- }
2524
- }
2525
- else {
2526
- await runCommit(enteringViewItem.ionPageElement, leavingEl);
2527
- if (leavingEl && !progressAnimation) {
2528
- // Skip hiding if a newer transition already made leavingEl the entering view
2529
- // runCommit's generation check has already restored its visibility in that case
2530
- if (myGeneration === this.transitionGeneration || leavingEl !== this.transitionEnteringElement) {
2531
- leavingEl.classList.add('ion-page-hidden');
2532
- leavingEl.setAttribute('aria-hidden', 'true');
2533
- }
2534
- }
527
+ await runCommit(enteringViewItem.ionPageElement, leavingViewItem === null || leavingViewItem === void 0 ? void 0 : leavingViewItem.ionPageElement);
528
+ if (leavingViewItem && leavingViewItem.ionPageElement && !progressAnimation) {
529
+ leavingViewItem.ionPageElement.classList.add('ion-page-hidden');
530
+ leavingViewItem.ionPageElement.setAttribute('aria-hidden', 'true');
2535
531
  }
2536
532
  }
2537
533
  }
@@ -2539,474 +535,193 @@ class StackManager extends React.PureComponent {
2539
535
  render() {
2540
536
  const { children } = this.props;
2541
537
  const ionRouterOutlet = React.Children.only(children);
2542
- // Store reference for use in getParentPath() and handlePageTransition()
2543
538
  this.ionRouterOutlet = ionRouterOutlet;
2544
- return (React.createElement(UNSAFE_RouteContext.Consumer, null, (parentContext) => {
2545
- // Derive the outlet's mount path from React Router's matched route context.
2546
- // This eliminates the need for heuristic-based mount path discovery in
2547
- // computeParentPath, since React Router already knows the matched base path.
2548
- const parentMatches = parentContext === null || parentContext === void 0 ? void 0 : parentContext.matches;
2549
- const parentPathnameBase = parentMatches && parentMatches.length > 0
2550
- ? parentMatches[parentMatches.length - 1].pathnameBase
2551
- : undefined;
2552
- // Derive isRootOutlet from RouteContext: empty matches means root.
2553
- this.isRootOutlet = !parentMatches || parentMatches.length === 0;
2554
- // Seed StackManager's mount path from the parent route context
2555
- if (parentPathnameBase && !this.outletMountPath) {
2556
- this.outletMountPath = parentPathnameBase;
2557
- }
2558
- const components = this.context.getChildrenToRender(this.id, this.ionRouterOutlet, this.props.routeInfo, () => {
2559
- // Callback triggers re-render when view items are modified during getChildrenToRender
2560
- this.forceUpdate();
2561
- }, parentPathnameBase);
2562
- return (React.createElement(StackContext.Provider, { value: this.stackContextValue }, React.cloneElement(ionRouterOutlet, {
2563
- ref: (node) => {
2564
- if (ionRouterOutlet.props.setRef) {
2565
- // Needed to handle external refs from devs.
2566
- ionRouterOutlet.props.setRef(node);
2567
- }
2568
- if (ionRouterOutlet.props.forwardedRef) {
2569
- // Needed to handle external refs from devs.
2570
- ionRouterOutlet.props.forwardedRef.current = node;
2571
- }
2572
- this.routerOutletElement = node;
2573
- const { ref } = ionRouterOutlet;
2574
- // Check for legacy refs.
2575
- if (typeof ref === 'function') {
2576
- ref(node);
2577
- }
2578
- },
2579
- }, components)));
2580
- }));
539
+ const components = this.context.getChildrenToRender(this.id, this.ionRouterOutlet, this.props.routeInfo, () => {
540
+ this.forceUpdate();
541
+ });
542
+ return (React.createElement(StackContext.Provider, { value: this.stackContextValue }, React.cloneElement(ionRouterOutlet, {
543
+ ref: (node) => {
544
+ if (ionRouterOutlet.props.setRef) {
545
+ ionRouterOutlet.props.setRef(node);
546
+ }
547
+ if (ionRouterOutlet.props.forwardedRef) {
548
+ ionRouterOutlet.props.forwardedRef.current = node;
549
+ }
550
+ this.routerOutletElement = node;
551
+ const { ref } = ionRouterOutlet;
552
+ if (typeof ref === 'function') {
553
+ ref(node);
554
+ }
555
+ },
556
+ }, components)));
2581
557
  }
2582
558
  static get contextType() {
2583
559
  return RouteManagerContext;
2584
560
  }
2585
561
  }
2586
- /**
2587
- * Converts React Route elements to RouteObject format for use with matchRoutes().
2588
- * Filters out pathless routes (which are handled by fallback logic separately).
2589
- *
2590
- * When a basename is provided, absolute route paths are relativized by stripping
2591
- * the basename prefix. This is necessary because matchRoutes() strips the basename
2592
- * from the LOCATION pathname but not from route paths — absolute paths must be
2593
- * made relative to the basename for matching to work correctly.
2594
- *
2595
- * @param routeChildren The flat array of Route/IonRoute elements from the outlet.
2596
- * @param basename The resolved parent path (without trailing slash or `/*`) used to relativize absolute paths.
2597
- */
2598
- function routeElementsToRouteObjects(routeChildren, basename) {
2599
- return routeChildren
2600
- .filter((child) => child.props.path != null || child.props.index)
2601
- .map((child) => {
2602
- const handle = { _element: child };
2603
- let path = child.props.path;
2604
- // Relativize absolute paths by stripping the basename prefix
2605
- if (path && path.startsWith('/') && basename) {
2606
- if (path === basename) {
2607
- path = '';
2608
- }
2609
- else if (path.startsWith(basename + '/')) {
2610
- path = path.slice(basename.length + 1);
2611
- }
2612
- }
2613
- if (child.props.index) {
2614
- return {
2615
- index: true,
2616
- handle,
2617
- caseSensitive: child.props.caseSensitive || undefined,
2618
- };
562
+ function matchRoute(node, routeInfo) {
563
+ let matchedNode;
564
+ React.Children.forEach(node, (child) => {
565
+ const match = matchPath({
566
+ pathname: routeInfo.pathname,
567
+ componentProps: child.props,
568
+ });
569
+ if (match) {
570
+ matchedNode = child;
2619
571
  }
2620
- return {
2621
- path,
2622
- handle,
2623
- caseSensitive: child.props.caseSensitive || undefined,
2624
- };
2625
572
  });
2626
- }
2627
- /**
2628
- * Finds the `<Route />` node matching the current route info.
2629
- * If no `<Route />` can be matched, a fallback node is returned.
2630
- * Routes are prioritized by specificity (most specific first).
2631
- *
2632
- * @param node The root node to search for `<Route />` nodes.
2633
- * @param routeInfo The route information to match against.
2634
- * @param parentPath The parent path that was matched by the parent outlet (for nested routing)
2635
- */
2636
- function findRouteByRouteInfo(node, routeInfo, parentPath) {
2637
- var _a, _b, _c;
2638
- let matchedNode;
2639
- let fallbackNode;
2640
- // `<Route />` nodes are rendered inside of a <Routes /> node
2641
- const routesChildren = (_a = getRoutesChildren(node)) !== null && _a !== void 0 ? _a : node;
2642
- // Collect all route children
2643
- const routeChildren = React.Children.toArray(routesChildren).filter((child) => React.isValidElement(child) && (child.type === Route || child.type === IonRoute));
2644
- // Delegate route matching to RR6's matchRoutes(), which handles specificity ranking internally.
2645
- const basename = parentPath ? stripTrailingSlash(parentPath.replace('/*', '')) : undefined;
2646
- const routeObjects = routeElementsToRouteObjects(routeChildren, basename);
2647
- const matches = matchRoutes(routeObjects, { pathname: routeInfo.pathname }, basename);
2648
- if (matches && matches.length > 0) {
2649
- const bestMatch = matches[matches.length - 1];
2650
- matchedNode = (_c = (_b = bestMatch.route.handle) === null || _b === void 0 ? void 0 : _b._element) !== null && _c !== void 0 ? _c : undefined;
573
+ if (matchedNode) {
574
+ return matchedNode;
2651
575
  }
2652
- // Fallback: try pathless routes, but only if pathname is within scope.
2653
- if (!matchedNode) {
2654
- let pathnameInScope = true;
2655
- if (parentPath) {
2656
- pathnameInScope = isPathnameInScope(routeInfo.pathname, parentPath);
2657
- }
2658
- else {
2659
- const absolutePathRoutes = routeChildren.filter((r) => r.props.path && r.props.path.startsWith('/'));
2660
- if (absolutePathRoutes.length > 0) {
2661
- const absolutePaths = absolutePathRoutes.map((r) => r.props.path);
2662
- const commonPrefix = computeCommonPrefix(absolutePaths);
2663
- if (commonPrefix && commonPrefix !== '/') {
2664
- pathnameInScope = routeInfo.pathname.startsWith(commonPrefix);
2665
- }
2666
- }
2667
- }
2668
- if (pathnameInScope) {
2669
- for (const child of routeChildren) {
2670
- if (!child.props.path) {
2671
- fallbackNode = child;
2672
- break;
2673
- }
2674
- }
576
+ // If we haven't found a node
577
+ // try to find one that doesn't have a path or from prop, that will be our not found route
578
+ React.Children.forEach(node, (child) => {
579
+ if (!(child.props.path || child.props.from)) {
580
+ matchedNode = child;
2675
581
  }
2676
- }
2677
- return matchedNode !== null && matchedNode !== void 0 ? matchedNode : fallbackNode;
582
+ });
583
+ return matchedNode;
2678
584
  }
2679
- function matchComponent(node, pathname, forceExact, parentPath) {
2680
- var _a;
2681
- const routePath = (_a = node === null || node === void 0 ? void 0 : node.props) === null || _a === void 0 ? void 0 : _a.path;
2682
- let pathnameToMatch;
2683
- if (parentPath && routePath && !routePath.startsWith('/')) {
2684
- // When parent path is known, compute exact relative pathname
2685
- const relative = pathname.startsWith(parentPath)
2686
- ? pathname.slice(parentPath.length).replace(/^\//, '')
2687
- : pathname;
2688
- pathnameToMatch = relative;
2689
- }
2690
- else {
2691
- pathnameToMatch = derivePathnameToMatch(pathname, routePath);
2692
- }
585
+ function matchComponent(node, pathname, forceExact) {
2693
586
  return matchPath({
2694
- pathname: pathnameToMatch,
2695
- componentProps: Object.assign(Object.assign({}, node.props), { end: forceExact }),
587
+ pathname,
588
+ componentProps: Object.assign(Object.assign({}, node.props), { exact: forceExact }),
2696
589
  });
2697
590
  }
2698
591
 
2699
- /**
2700
- * `IonRouter` is responsible for managing the application's navigation
2701
- * state, tracking the history of visited routes, and coordinating
2702
- * transitions between different views. It intercepts route changes from
2703
- * React Router and translates them into actions that Ionic can understand
2704
- * and animate.
2705
- */
2706
- const filterUndefinedParams = (params) => {
2707
- const result = {};
2708
- for (const key of Object.keys(params)) {
2709
- const value = params[key];
2710
- if (value !== undefined) {
2711
- result[key] = value;
2712
- }
2713
- }
2714
- return result;
2715
- };
2716
- /**
2717
- * Checks if a POP event is a multi-step back navigation (navigate(-n) where n > 1).
2718
- * Walks the pushedByRoute chain from prevInfo to verify the destination is an ancestor
2719
- * in the same navigation chain. This distinguishes multi-step back from tab-crossing
2720
- * back navigation where prevInfo.pathname also differs from the browser destination.
2721
- */
2722
- const checkIsMultiStepBack = (prevInfo, destinationPathname, history) => {
2723
- if (!prevInfo || prevInfo.pathname === destinationPathname)
2724
- return false;
2725
- const visited = new Set();
2726
- let walker = prevInfo;
2727
- while (walker === null || walker === void 0 ? void 0 : walker.pushedByRoute) {
2728
- if (visited.has(walker.id))
2729
- break; // cycle guard
2730
- visited.add(walker.id);
2731
- if (walker.pushedByRoute === destinationPathname)
2732
- return true;
2733
- walker = history.findLastLocation(walker);
2734
- }
2735
- return false;
2736
- };
2737
- const areParamsEqual = (a, b) => {
2738
- const paramsA = a || {};
2739
- const paramsB = b || {};
2740
- const keysA = Object.keys(paramsA);
2741
- const keysB = Object.keys(paramsB);
2742
- if (keysA.length !== keysB.length) {
2743
- return false;
592
+ class IonRouterInner extends React.PureComponent {
593
+ constructor(props) {
594
+ super(props);
595
+ this.exitViewFromOtherOutletHandlers = [];
596
+ this.locationHistory = new LocationHistory();
597
+ this.viewStack = new ReactRouterViewStack();
598
+ this.routeMangerContextState = {
599
+ canGoBack: () => this.locationHistory.canGoBack(),
600
+ clearOutlet: this.viewStack.clear,
601
+ findViewItemByPathname: this.viewStack.findViewItemByPathname,
602
+ getChildrenToRender: this.viewStack.getChildrenToRender,
603
+ goBack: () => this.handleNavigateBack(),
604
+ createViewItem: this.viewStack.createViewItem,
605
+ findViewItemByRouteInfo: this.viewStack.findViewItemByRouteInfo,
606
+ findLeavingViewItemByRouteInfo: this.viewStack.findLeavingViewItemByRouteInfo,
607
+ addViewItem: this.viewStack.add,
608
+ unMountViewItem: this.viewStack.remove,
609
+ };
610
+ const routeInfo = {
611
+ id: generateId('routeInfo'),
612
+ pathname: this.props.location.pathname,
613
+ search: this.props.location.search,
614
+ };
615
+ this.locationHistory.add(routeInfo);
616
+ this.handleChangeTab = this.handleChangeTab.bind(this);
617
+ this.handleResetTab = this.handleResetTab.bind(this);
618
+ this.handleNativeBack = this.handleNativeBack.bind(this);
619
+ this.handleNavigate = this.handleNavigate.bind(this);
620
+ this.handleNavigateBack = this.handleNavigateBack.bind(this);
621
+ this.props.registerHistoryListener(this.handleHistoryChange.bind(this));
622
+ this.handleSetCurrentTab = this.handleSetCurrentTab.bind(this);
623
+ this.state = {
624
+ routeInfo,
625
+ };
2744
626
  }
2745
- return keysA.every((key) => {
2746
- const valueA = paramsA[key];
2747
- const valueB = paramsB[key];
2748
- if (Array.isArray(valueA) && Array.isArray(valueB)) {
2749
- if (valueA.length !== valueB.length) {
2750
- return false;
2751
- }
2752
- return valueA.every((entry, idx) => entry === valueB[idx]);
2753
- }
2754
- return valueA === valueB;
2755
- });
2756
- };
2757
- const IonRouter = ({ children, registerHistoryListener }) => {
2758
- const location = useLocation();
2759
- const navigate = useNavigate();
2760
- const didMountRef = useRef(false);
2761
- const locationHistory = useRef(new LocationHistory());
2762
- const currentTab = useRef(undefined);
2763
- const viewStack = useRef(new ReactRouterViewStack());
2764
- const incomingRouteParams = useRef(null);
2765
- /**
2766
- * Tracks location keys that the user navigated away from via browser back.
2767
- * When a POP event's destination key matches the top of this stack, it's a
2768
- * browser forward navigation. Uses React Router's unique location.key
2769
- * instead of URLs to correctly handle duplicate URLs in history (e.g.,
2770
- * navigating to /details, then /settings, then /details via routerLink,
2771
- * then pressing back).
2772
- * Cleared on PUSH (new navigation invalidates forward history).
2773
- */
2774
- const forwardStack = useRef([]);
2775
- /**
2776
- * Tracks the current location key so we can push it onto the forward stack
2777
- * when navigating back. Updated after each history change.
2778
- */
2779
- const currentLocationKeyRef = useRef(location.key);
2780
- const [routeInfo, setRouteInfo] = useState({
2781
- id: generateId('routeInfo'),
2782
- pathname: location.pathname,
2783
- search: location.search,
2784
- params: {},
2785
- });
2786
- useEffect(() => {
2787
- if (didMountRef.current) {
627
+ handleChangeTab(tab, path, routeOptions) {
628
+ if (!path) {
2788
629
  return;
2789
630
  }
2790
- // Seed the history stack with the initial location and begin listening
2791
- // for future navigations once React has committed the mount. This avoids
2792
- // duplicate entries when React StrictMode runs an extra render pre-commit.
2793
- locationHistory.current.add(routeInfo);
2794
- // If IonTabBar already called handleSetCurrentTab during render (before this
2795
- // effect), the tab was stored in currentTab.current but the history entry was
2796
- // not yet seeded. Apply the pending tab to the seed entry now.
2797
- if (currentTab.current) {
2798
- const ri = Object.assign({}, locationHistory.current.current());
2799
- if (ri.tab !== currentTab.current) {
2800
- ri.tab = currentTab.current;
2801
- locationHistory.current.update(ri);
631
+ const routeInfo = this.locationHistory.getCurrentRouteInfoForTab(tab);
632
+ const [pathname, search] = path.split('?');
633
+ if (routeInfo) {
634
+ this.incomingRouteParams = Object.assign(Object.assign({}, routeInfo), { routeAction: 'push', routeDirection: 'none' });
635
+ if (routeInfo.pathname === pathname) {
636
+ this.incomingRouteParams.routeOptions = routeOptions;
637
+ this.props.history.push(routeInfo.pathname + (routeInfo.search || ''));
638
+ }
639
+ else {
640
+ this.incomingRouteParams.pathname = pathname;
641
+ this.incomingRouteParams.search = search ? '?' + search : undefined;
642
+ this.incomingRouteParams.routeOptions = routeOptions;
643
+ this.props.history.push(pathname + (search ? '?' + search : ''));
2802
644
  }
2803
645
  }
2804
- registerHistoryListener(handleHistoryChange);
2805
- didMountRef.current = true;
2806
- }, []);
2807
- // Sync route params extracted by React Router's path matching back into routeInfo.
2808
- // The view stack's match may contain params (e.g., :id) not present in the initial routeInfo.
2809
- useEffect(() => {
2810
- var _a;
2811
- const activeView = viewStack.current.findViewItemByRouteInfo(routeInfo, undefined, true);
2812
- const matchedParams = (_a = activeView === null || activeView === void 0 ? void 0 : activeView.routeData.match) === null || _a === void 0 ? void 0 : _a.params;
2813
- if (matchedParams) {
2814
- const paramsCopy = filterUndefinedParams(Object.assign({}, matchedParams));
2815
- if (areParamsEqual(routeInfo.params, paramsCopy)) {
2816
- return;
646
+ else {
647
+ this.handleNavigate(pathname, 'push', 'none', undefined, routeOptions, tab);
648
+ }
649
+ }
650
+ handleHistoryChange(location, action) {
651
+ var _a, _b, _c;
652
+ let leavingLocationInfo;
653
+ if (this.incomingRouteParams) {
654
+ if (this.incomingRouteParams.routeAction === 'replace') {
655
+ leavingLocationInfo = this.locationHistory.previous();
656
+ }
657
+ else {
658
+ leavingLocationInfo = this.locationHistory.current();
2817
659
  }
2818
- const updatedRouteInfo = Object.assign(Object.assign({}, routeInfo), { params: paramsCopy });
2819
- locationHistory.current.update(updatedRouteInfo);
2820
- setRouteInfo(updatedRouteInfo);
2821
660
  }
2822
- }, [routeInfo]);
2823
- /**
2824
- * Triggered whenever the history changes, either through user navigation
2825
- * or programmatic changes. It transforms the raw browser history changes
2826
- * into `RouteInfo` objects, which are needed Ionic's animations and
2827
- * navigation patterns.
2828
- *
2829
- * @param location The current location object from the history.
2830
- * @param action The action that triggered the history change.
2831
- */
2832
- const handleHistoryChange = (location, action) => {
2833
- var _a, _b, _c, _d;
2834
- /**
2835
- * The leaving location is always the current route, for both programmatic
2836
- * and external navigations. Using `previous()` for replace actions was
2837
- * incorrect: it caused the equality check below to skip navigation when
2838
- * the replace destination matched the entry two slots back in history.
2839
- */
2840
- const leavingLocationInfo = locationHistory.current.current();
661
+ else {
662
+ leavingLocationInfo = this.locationHistory.current();
663
+ }
2841
664
  const leavingUrl = leavingLocationInfo.pathname + leavingLocationInfo.search;
2842
- if (leavingUrl !== location.pathname + location.search) {
2843
- if (!incomingRouteParams.current) {
2844
- // Use history-based tab detection instead of URL-pattern heuristics,
2845
- // so tab routes work with any URL structure (not just paths containing "/tabs").
2846
- // Fall back to currentTab.current only when the destination is within the
2847
- // current tab's path hierarchy (prevents non-tab routes from inheriting a tab).
2848
- let tabToUse = locationHistory.current.findTabForPathname(location.pathname);
2849
- if (!tabToUse && currentTab.current) {
2850
- const tabFirstRoute = locationHistory.current.getFirstRouteInfoForTab(currentTab.current);
2851
- const tabRootPath = tabFirstRoute === null || tabFirstRoute === void 0 ? void 0 : tabFirstRoute.pathname;
2852
- if (tabRootPath && (location.pathname === tabRootPath || location.pathname.startsWith(tabRootPath + '/'))) {
2853
- tabToUse = currentTab.current;
2854
- }
2855
- }
2856
- /**
2857
- * A `REPLACE` action can be triggered by React Router's
2858
- * `<Navigate />` component.
2859
- */
665
+ if (leavingUrl !== location.pathname) {
666
+ if (!this.incomingRouteParams) {
2860
667
  if (action === 'REPLACE') {
2861
- incomingRouteParams.current = {
668
+ this.incomingRouteParams = {
2862
669
  routeAction: 'replace',
2863
670
  routeDirection: 'none',
2864
- tab: tabToUse,
671
+ tab: this.currentTab,
2865
672
  };
2866
673
  }
2867
- /**
2868
- * A `POP` action can be triggered by the browser's back/forward
2869
- * button. Both fire as POP events, so we use a forward stack to
2870
- * distinguish them: when going back, we push the leaving pathname
2871
- * onto the stack. When the next POP's destination matches the top
2872
- * of the stack, it's a forward navigation.
2873
- */
2874
674
  if (action === 'POP') {
2875
- const currentRoute = locationHistory.current.current();
2876
- const isForwardNavigation = forwardStack.current.length > 0 &&
2877
- forwardStack.current[forwardStack.current.length - 1] === location.key;
2878
- if (isForwardNavigation) {
2879
- forwardStack.current.pop();
2880
- incomingRouteParams.current = {
2881
- routeAction: 'push',
2882
- routeDirection: 'forward',
2883
- tab: tabToUse,
2884
- };
2885
- }
2886
- else if (currentRoute && currentRoute.pushedByRoute) {
2887
- // Back navigation. Record current location key for potential forward
2888
- forwardStack.current.push(currentLocationKeyRef.current);
2889
- const prevInfo = locationHistory.current.findLastLocation(currentRoute);
2890
- const isMultiStepBack = checkIsMultiStepBack(prevInfo, location.pathname, locationHistory.current);
2891
- if (isMultiStepBack) {
2892
- const destinationInfo = locationHistory.current.findLastLocationByPathname(location.pathname);
2893
- incomingRouteParams.current = Object.assign(Object.assign({}, (destinationInfo || {})), { routeAction: 'pop', routeDirection: 'back' });
2894
- }
2895
- else if (prevInfo && prevInfo.pathname !== location.pathname && currentRoute.tab) {
2896
- // Browser POP destination differs from within-tab back target.
2897
- // Sync URL via replace, like handleNavigateBack's non-linear path (#25141).
2898
- incomingRouteParams.current = Object.assign(Object.assign({}, prevInfo), { routeAction: 'pop', routeDirection: 'back' });
2899
- forwardStack.current = [];
2900
- handleNavigate(prevInfo.pathname + (prevInfo.search || ''), 'pop', 'back', undefined, undefined, prevInfo.tab);
2901
- return;
2902
- }
2903
- else {
2904
- incomingRouteParams.current = Object.assign(Object.assign({}, prevInfo), { routeAction: 'pop', routeDirection: 'back' });
2905
- }
675
+ const currentRoute = this.locationHistory.current();
676
+ if (currentRoute && currentRoute.pushedByRoute) {
677
+ const prevInfo = this.locationHistory.findLastLocation(currentRoute);
678
+ this.incomingRouteParams = Object.assign(Object.assign({}, prevInfo), { routeAction: 'pop', routeDirection: 'back' });
2906
679
  }
2907
680
  else {
2908
- // It's a non-linear history path like a direct link.
2909
- // Still push the current location key so browser forward is detectable.
2910
- forwardStack.current.push(currentLocationKeyRef.current);
2911
- incomingRouteParams.current = {
681
+ this.incomingRouteParams = {
2912
682
  routeAction: 'pop',
2913
683
  routeDirection: 'none',
2914
- tab: tabToUse,
684
+ tab: this.currentTab,
2915
685
  };
2916
686
  }
2917
687
  }
2918
- if (!incomingRouteParams.current) {
2919
- const state = location.state;
2920
- incomingRouteParams.current = {
688
+ if (!this.incomingRouteParams) {
689
+ this.incomingRouteParams = {
2921
690
  routeAction: 'push',
2922
- routeDirection: (state === null || state === void 0 ? void 0 : state.direction) || 'forward',
2923
- routeOptions: state === null || state === void 0 ? void 0 : state.routerOptions,
2924
- tab: tabToUse,
691
+ routeDirection: ((_a = location.state) === null || _a === void 0 ? void 0 : _a.direction) || 'forward',
692
+ routeOptions: (_b = location.state) === null || _b === void 0 ? void 0 : _b.routerOptions,
693
+ tab: this.currentTab,
2925
694
  };
2926
695
  }
2927
696
  }
2928
- // New navigation (PUSH) invalidates browser forward history,
2929
- // so clear our forward stack to stay in sync.
2930
- if (action === 'PUSH') {
2931
- forwardStack.current = [];
2932
- }
2933
697
  let routeInfo;
2934
- // If we're navigating away from tabs to a non-tab route, clear the current tab
2935
- if (!locationHistory.current.findTabForPathname(location.pathname) && currentTab.current) {
2936
- currentTab.current = undefined;
2937
- }
2938
- /**
2939
- * An existing id indicates that it's re-activating an existing route.
2940
- * e.g., tab switching or navigating back to a previous route
2941
- */
2942
- if ((_a = incomingRouteParams.current) === null || _a === void 0 ? void 0 : _a.id) {
2943
- routeInfo = Object.assign(Object.assign({}, incomingRouteParams.current), { lastPathname: leavingLocationInfo.pathname });
2944
- locationHistory.current.add(routeInfo);
2945
- /**
2946
- * A new route is being created since it's not re-activating
2947
- * an existing route.
2948
- */
698
+ if ((_c = this.incomingRouteParams) === null || _c === void 0 ? void 0 : _c.id) {
699
+ routeInfo = Object.assign(Object.assign({}, this.incomingRouteParams), { lastPathname: leavingLocationInfo.pathname });
700
+ this.locationHistory.add(routeInfo);
2949
701
  }
2950
702
  else {
2951
- const isPushed = ((_b = incomingRouteParams.current) === null || _b === void 0 ? void 0 : _b.routeAction) === 'push' &&
2952
- incomingRouteParams.current.routeDirection === 'forward';
2953
- routeInfo = Object.assign(Object.assign({ id: generateId('routeInfo') }, incomingRouteParams.current), { lastPathname: leavingLocationInfo.pathname, pathname: location.pathname, search: location.search, params: ((_c = incomingRouteParams.current) === null || _c === void 0 ? void 0 : _c.params)
2954
- ? filterUndefinedParams(incomingRouteParams.current.params)
2955
- : {}, prevRouteLastPathname: leavingLocationInfo.lastPathname });
703
+ const isPushed = this.incomingRouteParams.routeAction === 'push' && this.incomingRouteParams.routeDirection === 'forward';
704
+ routeInfo = Object.assign(Object.assign({ id: generateId('routeInfo') }, this.incomingRouteParams), { lastPathname: leavingLocationInfo.pathname, pathname: location.pathname, search: location.search, params: this.props.match.params, prevRouteLastPathname: leavingLocationInfo.lastPathname });
2956
705
  if (isPushed) {
2957
- // Only inherit tab from leaving route if we don't already have one.
2958
- // This preserves tab context for same-tab navigation while allowing cross-tab navigation.
2959
- routeInfo.tab = routeInfo.tab || leavingLocationInfo.tab;
2960
- routeInfo.pushedByRoute = leavingLocationInfo.pathname;
2961
- }
2962
- else if (routeInfo.routeAction === 'push' &&
2963
- routeInfo.routeDirection === 'none' &&
2964
- routeInfo.tab === leavingLocationInfo.tab) {
2965
- // Push with routerDirection="none" within the same tab (or non-tab) context.
2966
- // Still needs pushedByRoute so the back button can navigate back correctly.
2967
- // Cross-tab navigations with direction "none" are handled by the tab-switching
2968
- // block below which has different pushedByRoute semantics.
2969
- routeInfo.tab = routeInfo.tab || leavingLocationInfo.tab;
706
+ routeInfo.tab = leavingLocationInfo.tab;
2970
707
  routeInfo.pushedByRoute = leavingLocationInfo.pathname;
2971
708
  }
2972
709
  else if (routeInfo.routeAction === 'pop') {
2973
- // Triggered by a browser back button or handleNavigateBack.
2974
- // Find the route that pushed this one.
2975
- const r = locationHistory.current.findLastLocation(routeInfo);
710
+ const r = this.locationHistory.findLastLocation(routeInfo);
2976
711
  routeInfo.pushedByRoute = r === null || r === void 0 ? void 0 : r.pushedByRoute;
2977
- // Navigating to a new tab.
2978
712
  }
2979
713
  else if (routeInfo.routeAction === 'push' && routeInfo.tab !== leavingLocationInfo.tab) {
2980
- /**
2981
- * If we are switching tabs grab the last route info for the
2982
- * tab and use its `pushedByRoute`.
2983
- */
2984
- const lastRoute = locationHistory.current.getCurrentRouteInfoForTab(routeInfo.tab);
2985
- /**
2986
- * Tab bar switches (direction 'none') should not create cross-tab back
2987
- * navigation. Only inherit pushedByRoute from the tab's own history.
2988
- */
2989
- if (routeInfo.routeDirection === 'none') {
2990
- routeInfo.pushedByRoute = lastRoute === null || lastRoute === void 0 ? void 0 : lastRoute.pushedByRoute;
2991
- }
2992
- else {
2993
- routeInfo.pushedByRoute = (_d = lastRoute === null || lastRoute === void 0 ? void 0 : lastRoute.pushedByRoute) !== null && _d !== void 0 ? _d : leavingLocationInfo.pathname;
2994
- }
2995
- // Triggered by `navigate()` with replace or a `<Navigate />` component, etc.
714
+ // If we are switching tabs grab the last route info for the tab and use its pushedByRoute
715
+ const lastRoute = this.locationHistory.getCurrentRouteInfoForTab(routeInfo.tab);
716
+ routeInfo.pushedByRoute = lastRoute === null || lastRoute === void 0 ? void 0 : lastRoute.pushedByRoute;
2996
717
  }
2997
718
  else if (routeInfo.routeAction === 'replace') {
719
+ // Make sure to set the lastPathname, etc.. to the current route so the page transitions out
720
+ const currentRouteInfo = this.locationHistory.current();
2998
721
  /**
2999
- * Make sure to set the `lastPathname`, etc.. to the current route
3000
- * so the page transitions out.
3001
- */
3002
- const currentRouteInfo = locationHistory.current.current();
3003
- /**
3004
- * Special handling for `replace` to ensure correct `pushedByRoute`
3005
- * and `lastPathname`.
3006
- *
3007
- * If going from `/home` to `/child`, then replacing from
3008
- * `/child` to `/home`, we don't want the route info to
3009
- * say that `/home` was pushed by `/home` which is not correct.
722
+ * If going from /home to /child, then replacing from
723
+ * /child to /home, we don't want the route info to
724
+ * say that /home was pushed by /home which is not correct.
3010
725
  */
3011
726
  const currentPushedBy = currentRouteInfo === null || currentRouteInfo === void 0 ? void 0 : currentRouteInfo.pushedByRoute;
3012
727
  const pushedByRoute = currentPushedBy !== undefined && currentPushedBy !== routeInfo.pathname
@@ -3024,121 +739,46 @@ const IonRouter = ({ children, registerHistoryListener }) => {
3024
739
  routeInfo.routeDirection = routeInfo.routeDirection || (currentRouteInfo === null || currentRouteInfo === void 0 ? void 0 : currentRouteInfo.routeDirection);
3025
740
  routeInfo.routeAnimation = routeInfo.routeAnimation || (currentRouteInfo === null || currentRouteInfo === void 0 ? void 0 : currentRouteInfo.routeAnimation);
3026
741
  }
3027
- locationHistory.current.add(routeInfo);
742
+ this.locationHistory.add(routeInfo);
3028
743
  }
3029
- setRouteInfo(routeInfo);
3030
- }
3031
- // Update the current location key after processing the history change.
3032
- // This ensures the forward stack records the correct key when navigating back.
3033
- currentLocationKeyRef.current = location.key;
3034
- incomingRouteParams.current = null;
3035
- };
3036
- /**
3037
- * Resets the specified tab to its initial, root route.
3038
- *
3039
- * @param tab The tab to reset.
3040
- * @param originalHref The original href for the tab.
3041
- * @param originalRouteOptions The original route options for the tab.
3042
- */
3043
- const handleResetTab = (tab, originalHref, originalRouteOptions) => {
3044
- const routeInfo = locationHistory.current.getFirstRouteInfoForTab(tab);
3045
- if (routeInfo) {
3046
- const [pathname, search] = originalHref.split('?');
3047
- const newRouteInfo = Object.assign({}, routeInfo);
3048
- newRouteInfo.pathname = pathname;
3049
- newRouteInfo.search = search ? '?' + search : '';
3050
- newRouteInfo.routeOptions = originalRouteOptions;
3051
- incomingRouteParams.current = Object.assign(Object.assign({}, newRouteInfo), { routeAction: 'pop', routeDirection: 'back' });
3052
- navigate(newRouteInfo.pathname + (newRouteInfo.search || ''));
744
+ this.setState({
745
+ routeInfo,
746
+ });
3053
747
  }
3054
- };
748
+ this.incomingRouteParams = undefined;
749
+ }
3055
750
  /**
3056
- * Handles tab changes.
3057
- *
3058
- * @param tab The tab to switch to.
3059
- * @param path The new path for the tab.
3060
- * @param routeOptions Additional route options.
751
+ * history@4.x uses goBack(), history@5.x uses back()
752
+ * TODO: If support for React Router <=5 is dropped
753
+ * this logic is no longer needed. We can just
754
+ * assume back() is available.
3061
755
  */
3062
- const handleChangeTab = (tab, path, routeOptions) => {
3063
- if (!path) {
3064
- return;
3065
- }
3066
- const routeInfo = locationHistory.current.getCurrentRouteInfoForTab(tab);
3067
- const [pathname, search] = path.split('?');
3068
- // User has navigated to the current tab before.
3069
- if (routeInfo) {
3070
- const routeParams = Object.assign(Object.assign({}, routeInfo), { routeAction: 'push', routeDirection: 'none' });
3071
- /**
3072
- * User is navigating to the same tab.
3073
- * e.g., `/tabs/home` → `/tabs/home`
3074
- */
3075
- if (routeInfo.pathname === pathname) {
3076
- const newSearch = search ? '?' + search : routeInfo.search;
3077
- incomingRouteParams.current = Object.assign(Object.assign({}, routeParams), { search: newSearch || '', routeOptions });
3078
- navigate(routeInfo.pathname + (newSearch || ''));
3079
- /**
3080
- * User is navigating to a different tab.
3081
- * e.g., `/tabs/home` → `/tabs/settings`
3082
- */
3083
- }
3084
- else {
3085
- incomingRouteParams.current = Object.assign(Object.assign({}, routeParams), { pathname, search: search ? '?' + search : '', routeOptions });
3086
- navigate(pathname + (search ? '?' + search : ''));
3087
- }
3088
- // User has not navigated to this tab before.
756
+ handleNativeBack() {
757
+ const history = this.props.history;
758
+ const goBack = history.goBack || history.back;
759
+ goBack();
760
+ }
761
+ handleNavigate(path, routeAction, routeDirection, routeAnimation, routeOptions, tab) {
762
+ this.incomingRouteParams = Object.assign(this.incomingRouteParams || {}, {
763
+ routeAction,
764
+ routeDirection,
765
+ routeOptions,
766
+ routeAnimation,
767
+ tab,
768
+ });
769
+ if (routeAction === 'push') {
770
+ this.props.history.push(path);
3089
771
  }
3090
772
  else {
3091
- const fullPath = pathname + (search ? '?' + search : '');
3092
- handleNavigate(fullPath, 'push', 'none', undefined, routeOptions, tab);
3093
- }
3094
- };
3095
- /**
3096
- * Set the current active tab in `locationHistory`.
3097
- * This is crucial for maintaining tab history since each tab has
3098
- * its own navigation stack.
3099
- *
3100
- * @param tab The tab to set as active.
3101
- */
3102
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
3103
- const handleSetCurrentTab = (tab, _routeInfo) => {
3104
- currentTab.current = tab;
3105
- const current = locationHistory.current.current();
3106
- if (!current) {
3107
- // locationHistory not yet seeded (e.g., called during initial render
3108
- // before mount effect). The mount effect will seed the correct entry.
3109
- return;
773
+ this.props.history.replace(path);
3110
774
  }
3111
- const ri = Object.assign({}, current);
3112
- if (ri.tab !== tab) {
3113
- ri.tab = tab;
3114
- locationHistory.current.update(ri);
3115
- }
3116
- };
3117
- /**
3118
- * Handles the native back button press.
3119
- * It's usually called when a user presses the platform-native back action.
3120
- */
3121
- const handleNativeBack = () => {
3122
- navigate(-1);
3123
- };
3124
- /**
3125
- * Used to manage the back navigation within the Ionic React's routing
3126
- * system. It's deeply integrated with Ionic's view lifecycle, animations,
3127
- * and its custom history tracking (`locationHistory`) to provide a
3128
- * native-like transition and maintain correct application state.
3129
- *
3130
- * @param defaultHref The fallback URL to navigate to if there's no
3131
- * previous entry in the `locationHistory` stack.
3132
- * @param routeAnimation A custom animation builder to override the
3133
- * default "back" animation.
3134
- */
3135
- const handleNavigateBack = (defaultHref, routeAnimation) => {
775
+ }
776
+ handleNavigateBack(defaultHref = '/', routeAnimation) {
3136
777
  const config = getConfig();
3137
- defaultHref = defaultHref !== null && defaultHref !== void 0 ? defaultHref : (config && config.get('backButtonDefaultHref'));
3138
- const routeInfo = locationHistory.current.current();
3139
- // It's a linear navigation.
778
+ defaultHref = defaultHref ? defaultHref : config && config.get('backButtonDefaultHref');
779
+ const routeInfo = this.locationHistory.current();
3140
780
  if (routeInfo && routeInfo.pushedByRoute) {
3141
- const prevInfo = locationHistory.current.findLastLocation(routeInfo);
781
+ const prevInfo = this.locationHistory.findLastLocation(routeInfo);
3142
782
  if (prevInfo) {
3143
783
  /**
3144
784
  * This needs to be passed to handleNavigate
@@ -3146,262 +786,160 @@ const IonRouter = ({ children, registerHistoryListener }) => {
3146
786
  * will be overridden.
3147
787
  */
3148
788
  const incomingAnimation = routeAnimation || routeInfo.routeAnimation;
3149
- incomingRouteParams.current = Object.assign(Object.assign({}, prevInfo), { routeAction: 'pop', routeDirection: 'back', routeAnimation: incomingAnimation });
3150
- /**
3151
- * Check if it's a simple linear back navigation (not tabbed).
3152
- * e.g., `/home` `/settings` back to `/home`
3153
- */
3154
- const condition1 = routeInfo.lastPathname === routeInfo.pushedByRoute;
3155
- const condition2 = prevInfo.pathname === routeInfo.pushedByRoute && !routeInfo.tab && !prevInfo.tab;
3156
- if (condition1 || condition2) {
3157
- // Record the current location key so browser forward is detectable
3158
- forwardStack.current.push(currentLocationKeyRef.current);
3159
- navigate(-1);
3160
- }
3161
- else {
789
+ this.incomingRouteParams = Object.assign(Object.assign({}, prevInfo), { routeAction: 'pop', routeDirection: 'back', routeAnimation: incomingAnimation });
790
+ if (routeInfo.lastPathname === routeInfo.pushedByRoute ||
791
+ /**
792
+ * We need to exclude tab switches/tab
793
+ * context changes here because tabbed
794
+ * navigation is not linear, but router.back()
795
+ * will go back in a linear fashion.
796
+ */
797
+ (prevInfo.pathname === routeInfo.pushedByRoute && routeInfo.tab === '' && prevInfo.tab === '')) {
3162
798
  /**
3163
- * It's a non-linear back navigation.
3164
- * e.g., direct link or tab switch or nested navigation with redirects
3165
- * Clear forward stack since the REPLACE-based navigate resets history
3166
- * position, making any prior forward entries unreachable.
799
+ * history@4.x uses goBack(), history@5.x uses back()
800
+ * TODO: If support for React Router <=5 is dropped
801
+ * this logic is no longer needed. We can just
802
+ * assume back() is available.
3167
803
  */
3168
- forwardStack.current = [];
3169
- handleNavigate(prevInfo.pathname + (prevInfo.search || ''), 'pop', 'back', incomingAnimation);
804
+ const history = this.props.history;
805
+ const goBack = history.goBack || history.back;
806
+ goBack();
807
+ }
808
+ else {
809
+ this.handleNavigate(prevInfo.pathname + (prevInfo.search || ''), 'pop', 'back', incomingAnimation);
3170
810
  }
3171
- /**
3172
- * `pushedByRoute` exists, but no corresponding previous entry in
3173
- * the history stack.
3174
- */
3175
811
  }
3176
- else if (defaultHref) {
3177
- handleNavigate(defaultHref, 'pop', 'back', routeAnimation);
812
+ else {
813
+ this.handleNavigate(defaultHref, 'pop', 'back', routeAnimation);
3178
814
  }
3179
- /**
3180
- * No `pushedByRoute` (e.g., initial page load or tab root).
3181
- * Navigate to defaultHref so the back button works on direct
3182
- * deep-link loads (e.g., loading /tab1/child directly).
3183
- * Only navigate when defaultHref is explicitly set. The core
3184
- * back-button component hides itself when no defaultHref is
3185
- * provided, so a click here means the user set one intentionally.
3186
- */
3187
815
  }
3188
- else if (defaultHref) {
3189
- handleNavigate(defaultHref, 'pop', 'back', routeAnimation);
816
+ else {
817
+ this.handleNavigate(defaultHref, 'pop', 'back', routeAnimation);
3190
818
  }
3191
- };
3192
- /**
3193
- * Used to programmatically navigate through the app.
3194
- *
3195
- * @param path The path to navigate to.
3196
- * @param routeAction The action to take (push, replace, etc.).
3197
- * @param routeDirection The direction of the navigation (forward,
3198
- * back, etc.).
3199
- * @param routeAnimation The animation to use for the transition.
3200
- * @param routeOptions Additional options for the route.
3201
- * @param tab The tab to navigate to, if applicable.
3202
- */
3203
- const handleNavigate = (path, routeAction, routeDirection, routeAnimation, routeOptions, tab) => {
3204
- var _a;
3205
- const normalizedRouteDirection = routeAction === 'push' && routeDirection === undefined ? 'forward' : routeDirection;
3206
- // When navigating from tabs context, we need to determine if the destination
3207
- // is also within tabs. If not, we should clear the tab context.
3208
- let navigationTab = tab;
3209
- // If no explicit tab is provided and we're in a tab context,
3210
- // check if the destination path is outside of the current tab context.
3211
- // Uses history-based tab detection instead of URL pattern matching,
3212
- // so it works with any tab URL structure.
3213
- if (!tab && currentTab.current && path) {
3214
- // Check if destination was previously visited in a tab context
3215
- const destinationTab = locationHistory.current.findTabForPathname(path);
3216
- if (destinationTab) {
3217
- // Previously visited as a tab route - use the known tab
3218
- navigationTab = destinationTab;
3219
- }
3220
- else {
3221
- // New destination - check if it's a child of the current tab's root path
3222
- const tabFirstRoute = locationHistory.current.getFirstRouteInfoForTab(currentTab.current);
3223
- if (tabFirstRoute) {
3224
- const tabRootPath = tabFirstRoute.pathname;
3225
- if (path === tabRootPath || path.startsWith(tabRootPath + '/')) {
3226
- // Still within the current tab's path hierarchy
3227
- navigationTab = currentTab.current;
3228
- }
3229
- else {
3230
- // Destination is outside the current tab context
3231
- currentTab.current = undefined;
3232
- navigationTab = undefined;
3233
- }
3234
- }
3235
- }
819
+ }
820
+ handleResetTab(tab, originalHref, originalRouteOptions) {
821
+ const routeInfo = this.locationHistory.getFirstRouteInfoForTab(tab);
822
+ if (routeInfo) {
823
+ const newRouteInfo = Object.assign({}, routeInfo);
824
+ newRouteInfo.pathname = originalHref;
825
+ newRouteInfo.routeOptions = originalRouteOptions;
826
+ this.incomingRouteParams = Object.assign(Object.assign({}, newRouteInfo), { routeAction: 'pop', routeDirection: 'back' });
827
+ this.props.history.push(newRouteInfo.pathname + (newRouteInfo.search || ''));
3236
828
  }
3237
- const baseParams = (_a = incomingRouteParams.current) !== null && _a !== void 0 ? _a : {};
3238
- incomingRouteParams.current = Object.assign(Object.assign({}, baseParams), { routeAction, routeDirection: normalizedRouteDirection, routeOptions,
3239
- routeAnimation, tab: navigationTab });
3240
- navigate(path, { replace: routeAction !== 'push' });
3241
- };
3242
- /**
3243
- * Navigates to a new root path, clearing Ionic's navigation history so that
3244
- * canGoBack() returns false after the transition. All previously mounted views
3245
- * are unmounted. Useful for post-login / post-logout root navigation.
3246
- *
3247
- * @param pathname The path to navigate to.
3248
- * @param routeAnimation An optional custom animation builder.
3249
- */
3250
- const handleNavigateRoot = (pathname, routeAnimation) => {
3251
- currentTab.current = undefined;
3252
- forwardStack.current = [];
3253
- incomingRouteParams.current = {
3254
- routeAction: 'replace',
3255
- routeDirection: 'root',
3256
- routeAnimation,
3257
- };
3258
- navigate(pathname, { replace: true });
3259
- };
3260
- const routeMangerContextValue = {
3261
- canGoBack: () => locationHistory.current.canGoBack(),
3262
- clearOutlet: viewStack.current.clear,
3263
- findViewItemByPathname: viewStack.current.findViewItemByPathname,
3264
- getChildrenToRender: viewStack.current.getChildrenToRender,
3265
- getViewItemsForOutlet: viewStack.current.getViewItemsForOutlet.bind(viewStack.current),
3266
- goBack: () => handleNavigateBack(),
3267
- createViewItem: viewStack.current.createViewItem,
3268
- findViewItemByRouteInfo: viewStack.current.findViewItemByRouteInfo,
3269
- findLeavingViewItemByRouteInfo: viewStack.current.findLeavingViewItemByRouteInfo,
3270
- addViewItem: viewStack.current.add,
3271
- unMountViewItem: viewStack.current.remove,
3272
- };
3273
- return (React.createElement(RouteManagerContext.Provider, { value: routeMangerContextValue },
3274
- React.createElement(NavManager, { ionRoute: IonRouteInner, stackManager: StackManager, routeInfo: routeInfo, onNativeBack: handleNativeBack, onNavigateBack: handleNavigateBack, onNavigate: handleNavigate, onNavigateRoot: handleNavigateRoot, onSetCurrentTab: handleSetCurrentTab, onChangeTab: handleChangeTab, onResetTab: handleResetTab, locationHistory: locationHistory.current }, children)));
3275
- };
829
+ }
830
+ handleSetCurrentTab(tab) {
831
+ this.currentTab = tab;
832
+ const ri = Object.assign({}, this.locationHistory.current());
833
+ if (ri.tab !== tab) {
834
+ ri.tab = tab;
835
+ this.locationHistory.update(ri);
836
+ }
837
+ }
838
+ render() {
839
+ return (React.createElement(RouteManagerContext.Provider, { value: this.routeMangerContextState },
840
+ React.createElement(NavManager, { ionRoute: IonRouteInner, ionRedirect: {}, stackManager: StackManager, routeInfo: this.state.routeInfo, onNativeBack: this.handleNativeBack, onNavigateBack: this.handleNavigateBack, onNavigate: this.handleNavigate, onSetCurrentTab: this.handleSetCurrentTab, onChangeTab: this.handleChangeTab, onResetTab: this.handleResetTab, locationHistory: this.locationHistory }, this.props.children)));
841
+ }
842
+ }
843
+ const IonRouter = withRouter(IonRouterInner);
3276
844
  IonRouter.displayName = 'IonRouter';
3277
845
 
3278
- /**
3279
- * `IonReactRouter` facilitates the integration of Ionic's specific
3280
- * navigation and UI management with the standard React Router mechanisms,
3281
- * allowing an inner Ionic-specific router (`IonRouter`) to react to
3282
- * navigation events.
3283
- */
3284
- /**
3285
- * This component acts as a bridge to ensure React Router hooks like
3286
- * `useLocation` and `useNavigationType` are called within the valid
3287
- * context of a `<BrowserRouter>`.
3288
- *
3289
- * It was split from `IonReactRouter` because these hooks must be
3290
- * descendants of a `<Router>` component, which `BrowserRouter` provides.
3291
- */
3292
- const RouterContent$2 = ({ children }) => {
3293
- const location = useLocation();
3294
- const navigationType = useNavigationType();
3295
- const historyListenHandler = useRef();
3296
- const registerHistoryListener = useCallback((cb) => {
3297
- historyListenHandler.current = cb;
3298
- }, []);
846
+ class IonReactRouter extends React.Component {
847
+ constructor(props) {
848
+ super(props);
849
+ const { history } = props, rest = __rest(props, ["history"]);
850
+ this.history = history || createBrowserHistory(rest);
851
+ this.history.listen(this.handleHistoryChange.bind(this));
852
+ this.registerHistoryListener = this.registerHistoryListener.bind(this);
853
+ }
3299
854
  /**
3300
- * Processes navigation changes within the application.
3301
- *
3302
- * Its purpose is to relay the current `location` and the associated
3303
- * `action` ('PUSH', 'POP', or 'REPLACE') to any registered listeners,
3304
- * primarily for `IonRouter` to manage Ionic-specific UI updates and
3305
- * navigation stack behavior.
3306
- *
3307
- * @param loc The current browser history location object.
3308
- * @param act The type of navigation action ('PUSH', 'POP', or
3309
- * 'REPLACE').
855
+ * history@4.x passes separate location and action
856
+ * params. history@5.x passes location and action
857
+ * together as a single object.
858
+ * TODO: If support for React Router <=5 is dropped
859
+ * this logic is no longer needed. We can just assume
860
+ * a single object with both location and action.
3310
861
  */
3311
- const handleHistoryChange = useCallback((loc, act) => {
3312
- if (historyListenHandler.current) {
3313
- historyListenHandler.current(loc, act);
862
+ handleHistoryChange(location, action) {
863
+ const locationValue = location.location || location;
864
+ const actionValue = location.action || action;
865
+ if (this.historyListenHandler) {
866
+ this.historyListenHandler(locationValue, actionValue);
3314
867
  }
3315
- }, []);
3316
- useEffect(() => {
3317
- handleHistoryChange(location, navigationType);
3318
- }, [location, navigationType, handleHistoryChange]);
3319
- return React.createElement(IonRouter, { registerHistoryListener: registerHistoryListener }, children);
3320
- };
3321
- const IonReactRouter = (_a) => {
3322
- var { children } = _a, browserRouterProps = __rest(_a, ["children"]);
3323
- return (React.createElement(BrowserRouter, Object.assign({}, browserRouterProps),
3324
- React.createElement(RouterContent$2, null, children)));
3325
- };
868
+ }
869
+ registerHistoryListener(cb) {
870
+ this.historyListenHandler = cb;
871
+ }
872
+ render() {
873
+ const _a = this.props, { children } = _a, props = __rest(_a, ["children"]);
874
+ return (React.createElement(Router, Object.assign({ history: this.history }, props),
875
+ React.createElement(IonRouter, { registerHistoryListener: this.registerHistoryListener }, children)));
876
+ }
877
+ }
3326
878
 
3327
- /**
3328
- * `IonReactMemoryRouter` provides a way to use `react-router` in
3329
- * environments where a traditional browser history (like `BrowserRouter`)
3330
- * isn't available or desirable.
3331
- */
3332
- const RouterContent$1 = ({ children }) => {
3333
- const location = useLocation$1();
3334
- const navigationType = useNavigationType$1();
3335
- const historyListenHandler = useRef();
3336
- const registerHistoryListener = useCallback((cb) => {
3337
- historyListenHandler.current = cb;
3338
- }, []);
879
+ class IonReactMemoryRouter extends React.Component {
880
+ constructor(props) {
881
+ super(props);
882
+ this.history = props.history;
883
+ this.history.listen(this.handleHistoryChange.bind(this));
884
+ this.registerHistoryListener = this.registerHistoryListener.bind(this);
885
+ }
3339
886
  /**
3340
- * Processes navigation changes within the application.
3341
- *
3342
- * Its purpose is to relay the current `location` and the associated
3343
- * `action` ('PUSH', 'POP', or 'REPLACE') to any registered listeners,
3344
- * primarily for `IonRouter` to manage Ionic-specific UI updates and
3345
- * navigation stack behavior.
3346
- *
3347
- * @param location The current browser history location object.
3348
- * @param action The type of navigation action ('PUSH', 'POP', or
3349
- * 'REPLACE').
887
+ * history@4.x passes separate location and action
888
+ * params. history@5.x passes location and action
889
+ * together as a single object.
890
+ * TODO: If support for React Router <=5 is dropped
891
+ * this logic is no longer needed. We can just assume
892
+ * a single object with both location and action.
3350
893
  */
3351
- const handleHistoryChange = useCallback((loc, act) => {
3352
- if (historyListenHandler.current) {
3353
- historyListenHandler.current(loc, act);
894
+ handleHistoryChange(location, action) {
895
+ const locationValue = location.location || location;
896
+ const actionValue = location.action || action;
897
+ if (this.historyListenHandler) {
898
+ this.historyListenHandler(locationValue, actionValue);
3354
899
  }
3355
- }, []);
3356
- useEffect(() => {
3357
- handleHistoryChange(location, navigationType);
3358
- }, [location, navigationType, handleHistoryChange]);
3359
- return React.createElement(IonRouter, { registerHistoryListener: registerHistoryListener }, children);
3360
- };
3361
- const IonReactMemoryRouter = (_a) => {
3362
- var { children } = _a, routerProps = __rest(_a, ["children"]);
3363
- return (React.createElement(MemoryRouter, Object.assign({}, routerProps),
3364
- React.createElement(RouterContent$1, null, children)));
3365
- };
900
+ }
901
+ registerHistoryListener(cb) {
902
+ this.historyListenHandler = cb;
903
+ }
904
+ render() {
905
+ const _a = this.props, { children } = _a, props = __rest(_a, ["children"]);
906
+ return (React.createElement(Router$1, Object.assign({}, props),
907
+ React.createElement(IonRouter, { registerHistoryListener: this.registerHistoryListener }, children)));
908
+ }
909
+ }
3366
910
 
3367
- /**
3368
- * `IonReactHashRouter` provides a way to use hash-based routing in Ionic
3369
- * React applications.
3370
- */
3371
- const RouterContent = ({ children }) => {
3372
- const location = useLocation();
3373
- const navigationType = useNavigationType();
3374
- const historyListenHandler = useRef();
3375
- const registerHistoryListener = useCallback((cb) => {
3376
- historyListenHandler.current = cb;
3377
- }, []);
911
+ class IonReactHashRouter extends React.Component {
912
+ constructor(props) {
913
+ super(props);
914
+ const { history } = props, rest = __rest(props, ["history"]);
915
+ this.history = history || createHashHistory(rest);
916
+ this.history.listen(this.handleHistoryChange.bind(this));
917
+ this.registerHistoryListener = this.registerHistoryListener.bind(this);
918
+ }
3378
919
  /**
3379
- * Processes navigation changes within the application.
3380
- *
3381
- * Its purpose is to relay the current `location` and the associated
3382
- * `action` ('PUSH', 'POP', or 'REPLACE') to any registered listeners,
3383
- * primarily for `IonRouter` to manage Ionic-specific UI updates and
3384
- * navigation stack behavior.
3385
- *
3386
- * @param location The current browser history location object.
3387
- * @param action The type of navigation action ('PUSH', 'POP', or
3388
- * 'REPLACE').
920
+ * history@4.x passes separate location and action
921
+ * params. history@5.x passes location and action
922
+ * together as a single object.
923
+ * TODO: If support for React Router <=5 is dropped
924
+ * this logic is no longer needed. We can just assume
925
+ * a single object with both location and action.
3389
926
  */
3390
- const handleHistoryChange = useCallback((loc, act) => {
3391
- if (historyListenHandler.current) {
3392
- historyListenHandler.current(loc, act);
927
+ handleHistoryChange(location, action) {
928
+ const locationValue = location.location || location;
929
+ const actionValue = location.action || action;
930
+ if (this.historyListenHandler) {
931
+ this.historyListenHandler(locationValue, actionValue);
3393
932
  }
3394
- }, []);
3395
- useEffect(() => {
3396
- handleHistoryChange(location, navigationType);
3397
- }, [location, navigationType, handleHistoryChange]);
3398
- return React.createElement(IonRouter, { registerHistoryListener: registerHistoryListener }, children);
3399
- };
3400
- const IonReactHashRouter = (_a) => {
3401
- var { children } = _a, routerProps = __rest(_a, ["children"]);
3402
- return (React.createElement(HashRouter, Object.assign({}, routerProps),
3403
- React.createElement(RouterContent, null, children)));
3404
- };
933
+ }
934
+ registerHistoryListener(cb) {
935
+ this.historyListenHandler = cb;
936
+ }
937
+ render() {
938
+ const _a = this.props, { children } = _a, props = __rest(_a, ["children"]);
939
+ return (React.createElement(Router, Object.assign({ history: this.history }, props),
940
+ React.createElement(IonRouter, { registerHistoryListener: this.registerHistoryListener }, children)));
941
+ }
942
+ }
3405
943
 
3406
944
  export { IonReactHashRouter, IonReactMemoryRouter, IonReactRouter };
3407
945
  //# sourceMappingURL=index.js.map