react-magma-dom 4.13.0-next.5 → 4.13.0-next.7
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/esm/index.js +232 -75
- package/dist/esm/index.js.map +1 -1
- package/dist/properties.json +138 -138
- package/dist/react-magma-dom.cjs.development.js +232 -75
- package/dist/react-magma-dom.cjs.development.js.map +1 -1
- package/dist/react-magma-dom.cjs.production.min.js +1 -1
- package/dist/react-magma-dom.cjs.production.min.js.map +1 -1
- package/package.json +1 -1
package/dist/esm/index.js
CHANGED
|
@@ -1568,14 +1568,15 @@ function generateId(id) {
|
|
|
1568
1568
|
return id ? id : v4();
|
|
1569
1569
|
}
|
|
1570
1570
|
function useGenerateId(newId) {
|
|
1571
|
-
var _React$useState = React__default.useState(
|
|
1571
|
+
var _React$useState = React__default.useState(function () {
|
|
1572
|
+
return generateId(newId);
|
|
1573
|
+
}),
|
|
1572
1574
|
id = _React$useState[0],
|
|
1573
1575
|
updateId = _React$useState[1];
|
|
1574
1576
|
React__default.useEffect(function () {
|
|
1575
|
-
|
|
1576
|
-
|
|
1577
|
-
|
|
1578
|
-
newId && updateId(generateId(newId));
|
|
1577
|
+
if (newId) {
|
|
1578
|
+
updateId(generateId(newId));
|
|
1579
|
+
}
|
|
1579
1580
|
}, [newId]);
|
|
1580
1581
|
return id;
|
|
1581
1582
|
}
|
|
@@ -11262,25 +11263,141 @@ var CalendarHeader = function CalendarHeader(props) {
|
|
|
11262
11263
|
})));
|
|
11263
11264
|
};
|
|
11264
11265
|
|
|
11266
|
+
var useDeviceDetect = function useDeviceDetect() {
|
|
11267
|
+
var userAgent = useMemo(function () {
|
|
11268
|
+
return typeof navigator !== 'undefined' ? navigator.userAgent.toLowerCase() : '';
|
|
11269
|
+
}, []);
|
|
11270
|
+
var isSafari = useMemo(function () {
|
|
11271
|
+
return /^((?!chrome|android).)*safari/i.test(userAgent);
|
|
11272
|
+
}, [userAgent]);
|
|
11273
|
+
var isChrome = useMemo(function () {
|
|
11274
|
+
return /chrome|crios/i.test(userAgent) && !/edge|edg/i.test(userAgent);
|
|
11275
|
+
}, [userAgent]);
|
|
11276
|
+
var isFirefox = useMemo(function () {
|
|
11277
|
+
return /firefox|fxios/i.test(userAgent);
|
|
11278
|
+
}, [userAgent]);
|
|
11279
|
+
var isEdge = useMemo(function () {
|
|
11280
|
+
return /edge|edg/i.test(userAgent);
|
|
11281
|
+
}, [userAgent]);
|
|
11282
|
+
var isMobile = useMemo(function () {
|
|
11283
|
+
return /mobi|android|touch|mini/i.test(userAgent);
|
|
11284
|
+
}, [userAgent]);
|
|
11285
|
+
var isWindows = useMemo(function () {
|
|
11286
|
+
return /windows nt/.test(userAgent);
|
|
11287
|
+
}, [userAgent]);
|
|
11288
|
+
var isMacOS = useMemo(function () {
|
|
11289
|
+
return /macintosh/.test(userAgent);
|
|
11290
|
+
}, [userAgent]);
|
|
11291
|
+
var isAndroid = useMemo(function () {
|
|
11292
|
+
return /android/.test(userAgent);
|
|
11293
|
+
}, [userAgent]);
|
|
11294
|
+
var isIOS = useMemo(function () {
|
|
11295
|
+
return /iphone|ipad|ipod/.test(userAgent);
|
|
11296
|
+
}, [userAgent]);
|
|
11297
|
+
var isLinux = useMemo(function () {
|
|
11298
|
+
return /linux/.test(userAgent) && !isAndroid;
|
|
11299
|
+
}, [userAgent, isAndroid]);
|
|
11300
|
+
return {
|
|
11301
|
+
isSafari: isSafari,
|
|
11302
|
+
isChrome: isChrome,
|
|
11303
|
+
isFirefox: isFirefox,
|
|
11304
|
+
isEdge: isEdge,
|
|
11305
|
+
isMobile: isMobile,
|
|
11306
|
+
isWindows: isWindows,
|
|
11307
|
+
isMacOS: isMacOS,
|
|
11308
|
+
isLinux: isLinux,
|
|
11309
|
+
isAndroid: isAndroid,
|
|
11310
|
+
isIOS: isIOS
|
|
11311
|
+
};
|
|
11312
|
+
};
|
|
11313
|
+
|
|
11314
|
+
/**
|
|
11315
|
+
* Returns a grouping key if the element belongs to a group
|
|
11316
|
+
* that shares a single tab stop (e.g. named radio groups).
|
|
11317
|
+
* Returns null if the element is independently tabbable.
|
|
11318
|
+
*
|
|
11319
|
+
* Extend this function to support new grouping patterns.
|
|
11320
|
+
*/
|
|
11321
|
+
function getTabStopGroupKey(el) {
|
|
11322
|
+
// Native radio buttons with the same name share a single tab stop.
|
|
11323
|
+
// Scope by form so identically-named groups in different forms stay separate.
|
|
11324
|
+
if (el instanceof HTMLInputElement && el.type === 'radio' && el.name) {
|
|
11325
|
+
var formScope = el.form ? "f" + (el.form.id || '') : 'noform';
|
|
11326
|
+
return "radio:" + formScope + ":" + el.name;
|
|
11327
|
+
}
|
|
11328
|
+
return null;
|
|
11329
|
+
}
|
|
11330
|
+
/**
|
|
11331
|
+
* Within a shared-tab-stop group, decides whether `candidate`
|
|
11332
|
+
* should replace `current` as the representative tab stop.
|
|
11333
|
+
* For radio buttons the checked element wins; otherwise the
|
|
11334
|
+
* first element in DOM order is kept (it was set when the
|
|
11335
|
+
* group was first encountered).
|
|
11336
|
+
*/
|
|
11337
|
+
function isPreferredTabStop(candidate, current) {
|
|
11338
|
+
if (candidate instanceof HTMLInputElement && candidate.type === 'radio' && current instanceof HTMLInputElement) {
|
|
11339
|
+
return candidate.checked && !current.checked;
|
|
11340
|
+
}
|
|
11341
|
+
return false;
|
|
11342
|
+
}
|
|
11343
|
+
/**
|
|
11344
|
+
* From a flat list of focusable elements, removes duplicates
|
|
11345
|
+
* that share a single tab stop, keeping only the representative
|
|
11346
|
+
* element for each group (the checked radio, or the first one
|
|
11347
|
+
* in DOM order when nothing is checked, etc.).
|
|
11348
|
+
*/
|
|
11349
|
+
function deduplicateTabStops(allFocusable) {
|
|
11350
|
+
var groupRepresentatives = new Map();
|
|
11351
|
+
for (var _iterator = _createForOfIteratorHelperLoose(allFocusable), _step; !(_step = _iterator()).done;) {
|
|
11352
|
+
var el = _step.value;
|
|
11353
|
+
var key = getTabStopGroupKey(el);
|
|
11354
|
+
if (key) {
|
|
11355
|
+
var current = groupRepresentatives.get(key);
|
|
11356
|
+
if (!current || isPreferredTabStop(el, current)) {
|
|
11357
|
+
groupRepresentatives.set(key, el);
|
|
11358
|
+
}
|
|
11359
|
+
}
|
|
11360
|
+
}
|
|
11361
|
+
return allFocusable.filter(function (el) {
|
|
11362
|
+
var key = getTabStopGroupKey(el);
|
|
11363
|
+
if (key) {
|
|
11364
|
+
return groupRepresentatives.get(key) === el;
|
|
11365
|
+
}
|
|
11366
|
+
return true;
|
|
11367
|
+
});
|
|
11368
|
+
}
|
|
11369
|
+
/**
|
|
11370
|
+
* Module-level registry of currently active focus-lock root elements.
|
|
11371
|
+
* Used in Safari to prevent an outer lock from handling Tab for elements
|
|
11372
|
+
* that belong to an inner (descendant) lock.
|
|
11373
|
+
*/
|
|
11374
|
+
var activeFocusLockRoots = /*#__PURE__*/new Set();
|
|
11265
11375
|
function useFocusLock(active, header, body) {
|
|
11266
11376
|
var rootNode = useRef(null);
|
|
11267
11377
|
var focusableItems = useRef([]);
|
|
11378
|
+
var _useDeviceDetect = useDeviceDetect(),
|
|
11379
|
+
isSafari = _useDeviceDetect.isSafari;
|
|
11268
11380
|
// The filter is necessary for the proper functioning of focus in drawer-navigation or similar cases
|
|
11269
11381
|
var updateFocusableItems = function updateFocusableItems() {
|
|
11270
11382
|
var _rootNode$current;
|
|
11271
|
-
|
|
11383
|
+
var allFocusable = Array.from(((_rootNode$current = rootNode.current) == null ? void 0 : _rootNode$current.querySelectorAll('button, [href], input, select, textarea, [tabindex]:not([tabindex="-1"]), video')) || []).filter(function (element) {
|
|
11272
11384
|
var style = window.getComputedStyle(element);
|
|
11273
|
-
return element instanceof HTMLElement && style.display !== 'none' && style.visibility !== 'hidden' && !element.hasAttribute('disabled');
|
|
11385
|
+
return element instanceof HTMLElement && style.display !== 'none' && style.visibility !== 'hidden' && !element.hasAttribute('disabled') && element.tabIndex >= 0;
|
|
11274
11386
|
});
|
|
11387
|
+
focusableItems.current = deduplicateTabStops(allFocusable);
|
|
11275
11388
|
};
|
|
11276
11389
|
useEffect(function () {
|
|
11277
11390
|
if (active) {
|
|
11391
|
+
var root = rootNode.current;
|
|
11392
|
+
if (root) {
|
|
11393
|
+
activeFocusLockRoots.add(root);
|
|
11394
|
+
}
|
|
11278
11395
|
updateFocusableItems();
|
|
11279
11396
|
var observer = new MutationObserver(function () {
|
|
11280
11397
|
updateFocusableItems();
|
|
11281
11398
|
});
|
|
11282
|
-
if (
|
|
11283
|
-
observer.observe(
|
|
11399
|
+
if (root) {
|
|
11400
|
+
observer.observe(root, {
|
|
11284
11401
|
childList: true,
|
|
11285
11402
|
subtree: true
|
|
11286
11403
|
});
|
|
@@ -11295,6 +11412,9 @@ function useFocusLock(active, header, body) {
|
|
|
11295
11412
|
body.current.firstChild.focus();
|
|
11296
11413
|
}
|
|
11297
11414
|
return function () {
|
|
11415
|
+
if (root) {
|
|
11416
|
+
activeFocusLockRoots["delete"](root);
|
|
11417
|
+
}
|
|
11298
11418
|
observer.disconnect();
|
|
11299
11419
|
};
|
|
11300
11420
|
}
|
|
@@ -11304,12 +11424,44 @@ function useFocusLock(active, header, body) {
|
|
|
11304
11424
|
if (!focusableItems.current) return;
|
|
11305
11425
|
var key = event.key,
|
|
11306
11426
|
shiftKey = event.shiftKey;
|
|
11307
|
-
var _focusableItems$curre = focusableItems.current,
|
|
11308
|
-
length = _focusableItems$curre.length,
|
|
11309
|
-
firstItem = _focusableItems$curre[0],
|
|
11310
|
-
lastItem = _focusableItems$curre[length - 1];
|
|
11311
11427
|
if (active && key === 'Tab') {
|
|
11312
|
-
|
|
11428
|
+
updateFocusableItems();
|
|
11429
|
+
var length = focusableItems.current.length;
|
|
11430
|
+
var firstItem = focusableItems.current[0];
|
|
11431
|
+
var lastItem = focusableItems.current[length - 1];
|
|
11432
|
+
var activeElement = document.activeElement;
|
|
11433
|
+
var eventTarget = event.target;
|
|
11434
|
+
var isEventInsideCurrentLock = !!rootNode.current && !!eventTarget && rootNode.current.contains(eventTarget);
|
|
11435
|
+
var isActiveElementTracked = !!activeElement && (activeElement === (header == null ? void 0 : header.current) || focusableItems.current.includes(activeElement));
|
|
11436
|
+
var isActiveElementInsideCurrentLock = !!rootNode.current && !!activeElement && rootNode.current.contains(activeElement);
|
|
11437
|
+
/**
|
|
11438
|
+
* Safari + VoiceOver can place screen reader focus on non-interactive content.
|
|
11439
|
+
* In that case, on next Tab the DOM focus may no longer be on one of the tracked
|
|
11440
|
+
* interactive elements, and the default browser tabbing can escape the lock.
|
|
11441
|
+
*
|
|
11442
|
+
* We only handle this as a fallback:
|
|
11443
|
+
* - lock is active
|
|
11444
|
+
* - Tab was pressed
|
|
11445
|
+
* - event still belongs to the current lock
|
|
11446
|
+
* - but DOM focus is no longer on a tracked interactive element
|
|
11447
|
+
*
|
|
11448
|
+
* This keeps the default logic intact and avoids breaking nested focus locks.
|
|
11449
|
+
*/
|
|
11450
|
+
// Check whether the active element sits inside a nested active lock.
|
|
11451
|
+
// If so, that lock should handle Tab — not this (outer) one.
|
|
11452
|
+
var isInsideNestedLock = !!activeElement && Array.from(activeFocusLockRoots).some(function (lockRoot) {
|
|
11453
|
+
return lockRoot !== rootNode.current && rootNode.current.contains(lockRoot) && lockRoot.contains(activeElement);
|
|
11454
|
+
});
|
|
11455
|
+
if (length > 0 && (isEventInsideCurrentLock || isSafari) && !isActiveElementTracked && !isInsideNestedLock && (isActiveElementInsideCurrentLock || activeElement === document.body)) {
|
|
11456
|
+
event.preventDefault();
|
|
11457
|
+
if (shiftKey) {
|
|
11458
|
+
lastItem.focus();
|
|
11459
|
+
} else {
|
|
11460
|
+
firstItem.focus();
|
|
11461
|
+
}
|
|
11462
|
+
return;
|
|
11463
|
+
}
|
|
11464
|
+
// If no focusable items
|
|
11313
11465
|
if (length === 0) {
|
|
11314
11466
|
event.preventDefault();
|
|
11315
11467
|
return;
|
|
@@ -11322,6 +11474,56 @@ function useFocusLock(active, header, body) {
|
|
|
11322
11474
|
}
|
|
11323
11475
|
return;
|
|
11324
11476
|
}
|
|
11477
|
+
// If focused on header then focus on first/last item
|
|
11478
|
+
if (document.activeElement === (header == null ? void 0 : header.current)) {
|
|
11479
|
+
event.preventDefault();
|
|
11480
|
+
(shiftKey ? lastItem : firstItem).focus();
|
|
11481
|
+
return;
|
|
11482
|
+
}
|
|
11483
|
+
/**
|
|
11484
|
+
* Safari does not keep Tab navigation inside React portals,
|
|
11485
|
+
* so we manually move focus to the next/prev item for every Tab.
|
|
11486
|
+
* Other browsers handle intermediate navigation natively and
|
|
11487
|
+
* only need the boundary guards below.
|
|
11488
|
+
*/
|
|
11489
|
+
if (isSafari) {
|
|
11490
|
+
// In Safari we manually manage every Tab, so we must exclude:
|
|
11491
|
+
// 1. Elements hidden by an ancestor (e.g. closed DatePicker calendar)
|
|
11492
|
+
// 2. Elements inside a nested active focus lock (they have their own handler)
|
|
11493
|
+
var nestedLockRoots = Array.from(activeFocusLockRoots).filter(function (lockRoot) {
|
|
11494
|
+
return lockRoot !== rootNode.current && rootNode.current.contains(lockRoot);
|
|
11495
|
+
});
|
|
11496
|
+
var visibleItems = focusableItems.current.filter(function (el) {
|
|
11497
|
+
if (typeof el.checkVisibility === 'function' && !el.checkVisibility()) {
|
|
11498
|
+
return false;
|
|
11499
|
+
}
|
|
11500
|
+
// Skip elements managed by a nested lock
|
|
11501
|
+
if (nestedLockRoots.length > 0 && nestedLockRoots.some(function (lockRoot) {
|
|
11502
|
+
return lockRoot.contains(el);
|
|
11503
|
+
})) {
|
|
11504
|
+
return false;
|
|
11505
|
+
}
|
|
11506
|
+
return true;
|
|
11507
|
+
});
|
|
11508
|
+
var visibleLength = visibleItems.length;
|
|
11509
|
+
if (visibleLength === 0) {
|
|
11510
|
+
event.preventDefault();
|
|
11511
|
+
return;
|
|
11512
|
+
}
|
|
11513
|
+
var idx = visibleItems.indexOf(activeElement);
|
|
11514
|
+
// Active element is not in this lock's own elements
|
|
11515
|
+
// (it must be inside a nested lock) — let that lock handle it.
|
|
11516
|
+
if (idx === -1) {
|
|
11517
|
+
return;
|
|
11518
|
+
}
|
|
11519
|
+
event.preventDefault();
|
|
11520
|
+
if (shiftKey) {
|
|
11521
|
+
visibleItems[idx <= 0 ? visibleLength - 1 : idx - 1].focus();
|
|
11522
|
+
} else {
|
|
11523
|
+
visibleItems[idx >= visibleLength - 1 ? 0 : idx + 1].focus();
|
|
11524
|
+
}
|
|
11525
|
+
return;
|
|
11526
|
+
}
|
|
11325
11527
|
// If focused on last item then focus on first item when tab is pressed
|
|
11326
11528
|
if (!shiftKey && document.activeElement === lastItem) {
|
|
11327
11529
|
event.preventDefault();
|
|
@@ -11329,7 +11531,7 @@ function useFocusLock(active, header, body) {
|
|
|
11329
11531
|
return;
|
|
11330
11532
|
}
|
|
11331
11533
|
// If focused on first item then focus on last item when shift + tab is pressed
|
|
11332
|
-
if (shiftKey &&
|
|
11534
|
+
if (shiftKey && document.activeElement === firstItem) {
|
|
11333
11535
|
event.preventDefault();
|
|
11334
11536
|
lastItem.focus();
|
|
11335
11537
|
return;
|
|
@@ -14907,54 +15109,6 @@ var DropdownMenuItem = /*#__PURE__*/forwardRef(function (props, forwardedRef) {
|
|
|
14907
15109
|
});
|
|
14908
15110
|
DropdownMenuItem.displayName = 'DropdownMenuItem';
|
|
14909
15111
|
|
|
14910
|
-
var useDeviceDetect = function useDeviceDetect() {
|
|
14911
|
-
var userAgent = useMemo(function () {
|
|
14912
|
-
return navigator.userAgent.toLowerCase();
|
|
14913
|
-
}, []);
|
|
14914
|
-
var isSafari = useMemo(function () {
|
|
14915
|
-
return /^((?!chrome|android).)*safari/i.test(userAgent);
|
|
14916
|
-
}, [userAgent]);
|
|
14917
|
-
var isChrome = useMemo(function () {
|
|
14918
|
-
return /chrome|crios/i.test(userAgent) && !/edge|edg/i.test(userAgent);
|
|
14919
|
-
}, [userAgent]);
|
|
14920
|
-
var isFirefox = useMemo(function () {
|
|
14921
|
-
return /firefox|fxios/i.test(userAgent);
|
|
14922
|
-
}, [userAgent]);
|
|
14923
|
-
var isEdge = useMemo(function () {
|
|
14924
|
-
return /edge|edg/i.test(userAgent);
|
|
14925
|
-
}, [userAgent]);
|
|
14926
|
-
var isMobile = useMemo(function () {
|
|
14927
|
-
return /mobi|android|touch|mini/i.test(userAgent);
|
|
14928
|
-
}, [userAgent]);
|
|
14929
|
-
var isWindows = useMemo(function () {
|
|
14930
|
-
return /windows nt/.test(userAgent);
|
|
14931
|
-
}, [userAgent]);
|
|
14932
|
-
var isMacOS = useMemo(function () {
|
|
14933
|
-
return /macintosh/.test(userAgent);
|
|
14934
|
-
}, [userAgent]);
|
|
14935
|
-
var isAndroid = useMemo(function () {
|
|
14936
|
-
return /android/.test(userAgent);
|
|
14937
|
-
}, [userAgent]);
|
|
14938
|
-
var isIOS = useMemo(function () {
|
|
14939
|
-
return /iphone|ipad|ipod/.test(userAgent);
|
|
14940
|
-
}, [userAgent]);
|
|
14941
|
-
var isLinux = useMemo(function () {
|
|
14942
|
-
return /linux/.test(userAgent) && !isAndroid;
|
|
14943
|
-
}, [userAgent, isAndroid]);
|
|
14944
|
-
return {
|
|
14945
|
-
isSafari: isSafari,
|
|
14946
|
-
isChrome: isChrome,
|
|
14947
|
-
isFirefox: isFirefox,
|
|
14948
|
-
isEdge: isEdge,
|
|
14949
|
-
isMobile: isMobile,
|
|
14950
|
-
isWindows: isWindows,
|
|
14951
|
-
isMacOS: isMacOS,
|
|
14952
|
-
isLinux: isLinux,
|
|
14953
|
-
isAndroid: isAndroid,
|
|
14954
|
-
isIOS: isIOS
|
|
14955
|
-
};
|
|
14956
|
-
};
|
|
14957
|
-
|
|
14958
15112
|
var _excluded$T = ["children", "icon", "to"];
|
|
14959
15113
|
var StyledItem$3 = /*#__PURE__*/_styled("a", {
|
|
14960
15114
|
target: "e1b21r9q0",
|
|
@@ -18727,13 +18881,13 @@ var ModalContainer = /*#__PURE__*/_styled(Transition, {
|
|
|
18727
18881
|
return props.theme.spaceScale.spacing03;
|
|
18728
18882
|
}, ";right:0;top:0;z-index:", function (props) {
|
|
18729
18883
|
return props.modalCount >= 2 ? '999' : '998';
|
|
18730
|
-
}, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Modal.tsx"],"names":[],"mappings":"AAuB0C","file":"Modal.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css, Global } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport ReactDOM from 'react-dom';\r\nimport { CloseIcon } from 'react-magma-icons';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { I18nContext } from '../../i18n';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { magma } from '../../theme/magma';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { omit, useGenerateId, usePrevious } from '../../utils';\r\nimport { ButtonColor, ButtonVariant } from '../Button';\r\nimport { Heading } from '../Heading';\r\nimport { IconButton } from '../IconButton';\r\nimport { Transition } from '../Transition';\r\nimport { TypographyVisualStyle } from '../Typography';\r\nexport var ModalSize;\r\n(function (ModalSize) {\r\n    ModalSize[\"large\"] = \"large\";\r\n    ModalSize[\"medium\"] = \"medium\";\r\n    ModalSize[\"small\"] = \"small\";\r\n})(ModalSize || (ModalSize = {}));\r\nconst ModalContainer = styled(Transition) `\n  position: fixed;\n  bottom: 0;\n  left: 0;\n  overflow-y: auto;\n  padding: ${props => props.theme.spaceScale.spacing03};\n  right: 0;\n  top: 0;\n  z-index: ${props => (props.modalCount >= 2 ? '999' : '998')};\n`;\r\nconst ModalBackdrop = styled(Transition) `\n  backdrop-filter: blur(3px);\n  background: ${props => transparentize(0.4, props.theme.colors.neutral900)};\n  bottom: 0;\n  left: 0;\n  right: 0;\n  top: 0;\n  z-index: 997;\n  position: fixed;\n`;\r\nconst ModalContent = styled.div `\n  background: ${props => props.isInverse\r\n    ? props.theme.colors.primary600\r\n    : props.theme.colors.neutral100};\n  border: ${props => {\r\n    if (!props.showBackgroundOverlay && props.isInverse) {\r\n        return `1px solid ${transparentize(0.5, props.theme.colors.tertiary)}`;\r\n    }\r\n    return 'none';\r\n}};\n  border-radius: ${props => props.theme.borderRadius};\n  box-shadow: ${props => {\r\n    const amount = props.isInverse ? 0.82 : 0.6;\r\n    return `0 2px 6px ${transparentize(amount, props.theme.colors.neutral900)}`;\r\n}};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0 auto;\n  position: relative;\n  z-index: 1000;\n\n  max-width: ${props => {\r\n    switch (props.size) {\r\n        case 'large':\r\n            return props.theme.modal.width.large;\r\n        case 'small':\r\n            return props.theme.modal.width.small;\r\n        default:\r\n            return props.theme.modal.width.medium;\r\n    }\r\n}};\n\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    margin: ${props => props.theme.spaceScale.spacing08} auto;\n  }\n`;\r\nconst ModalHeader = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05}\n    ${props => props.theme.spaceScale.spacing05} 0\n    ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06}\n      ${props => props.theme.spaceScale.spacing06} 0\n      ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst ModalWrapper = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst H2 = styled(Heading) `\n  font-size: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.fontSize};\n  line-height: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.lineHeight};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0;\n  padding-right: ${props => props.theme.spaceScale.spacing10};\n  font-weight: 600;\n`;\r\nconst CloseBtn = styled.span `\n  position: absolute;\n  top: 0;\n  right: 0;\n  margin: ${props => props.theme.spaceScale.spacing02};\n`;\r\nexport const Modal = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, children, closeAriaLabel, closeButtonSize, portalContainer, containerStyle, containerTransition, isBackgroundClickDisabled, isEscKeyDownDisabled, header, isCloseButtonHidden, isOpen, unmountOnExit = true, testId, isModalClosingControlledManually, headerRef, onClose, hasDrawerAnimation, showBackgroundOverlay = true, headerLevel = 2, ...rest } = props;\r\n    const lastFocus = React.useRef();\r\n    const headingRef = React.useRef();\r\n    const bodyRef = React.useRef();\r\n    const id = useGenerateId(props.id);\r\n    const headingId = `${id}_heading`;\r\n    const contentId = `${id}_content`;\r\n    const [isModalOpen, setIsModalOpen] = React.useState(isOpen);\r\n    const [currentTarget, setCurrentTarget] = React.useState(null);\r\n    const [modalCount, setModalCount] = React.useState(0);\r\n    const focusTrapElement = useFocusLock(isModalOpen, headingRef, bodyRef);\r\n    const prevOpen = usePrevious(isOpen);\r\n    React.useEffect(() => {\r\n        if (isModalClosingControlledManually &&\r\n            prevOpen &&\r\n            !isOpen &&\r\n            isModalOpen) {\r\n            setIsModalOpen(false);\r\n        }\r\n        else if (!prevOpen && isOpen) {\r\n            setIsModalOpen(true);\r\n            if (headerRef && typeof headerRef === 'function') {\r\n                headerRef(headingRef);\r\n            }\r\n        }\r\n        else if (prevOpen && !isOpen && isModalOpen) {\r\n            handleClose();\r\n        }\r\n    }, [isOpen]);\r\n    React.useEffect(() => {\r\n        if (isModalOpen) {\r\n            lastFocus.current = document.activeElement;\r\n            const count = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            setModalCount(count);\r\n            if (!isEscKeyDownDisabled) {\r\n                document.body.addEventListener('keydown', handleEscapeKeyDown, false);\r\n            }\r\n        }\r\n        return () => {\r\n            document.body.removeEventListener('keydown', handleEscapeKeyDown, false);\r\n        };\r\n    }, [isModalOpen]);\r\n    function handleModalClick(event) {\r\n        if (!document.getElementById(contentId).contains(event.target) &&\r\n            event.target === currentTarget) {\r\n            handleClose(event);\r\n        }\r\n    }\r\n    function handleModalOnMouseDown(event) {\r\n        setCurrentTarget(event.target);\r\n    }\r\n    function handleEscapeKeyDown(event) {\r\n        if (event.key === 'Escape') {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            props.onEscKeyDown &&\r\n                typeof props.onEscKeyDown === 'function' &&\r\n                props.onEscKeyDown(event);\r\n            //Supports nested modals\r\n            const modalsInDom = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            if (modalCount <= 1 && modalsInDom !== 1) {\r\n                if (document.getElementById(id).contains(event.target)) {\r\n                    handleClose(event);\r\n                }\r\n                else {\r\n                    headingRef.current.focus();\r\n                }\r\n            }\r\n            else {\r\n                handleClose(event);\r\n            }\r\n        }\r\n    }\r\n    function handleClose(event) {\r\n        event?.stopPropagation();\r\n        setTimeout(() => {\r\n            if (!props.isModalClosingControlledManually) {\r\n                setIsModalOpen(false);\r\n            }\r\n            lastFocus.current?.focus();\r\n            onClose && typeof onClose === 'function' && onClose();\r\n        }, 0);\r\n    }\r\n    const isInverse = useIsInverse(props.isInverse);\r\n    const other = omit(['onEscKeyDown'], rest);\r\n    const theme = React.useContext(ThemeContext);\r\n    const i18n = React.useContext(I18nContext);\r\n    const CloseIconButton = (React.createElement(CloseIcon, { size: magma.iconSizes[closeButtonSize]\r\n            ? magma.iconSizes[closeButtonSize]\r\n            : magma.iconSizes.medium }));\r\n    return isModalOpen\r\n        ? ReactDOM.createPortal(React.createElement(\"div\", { ref: focusTrapElement },\r\n            React.createElement(Global, { styles: css `\n                html {\n                  overflow: ${isOpen ? 'hidden' : 'auto'};\n                }\n              ` }),\r\n            React.createElement(ModalContainer, Object.assign({ \"aria-labelledby\": header ? headingId : null, \"aria-label\": !header ? ariaLabel : null, \"aria-modal\": true, \"data-testid\": testId, id: id, modalCount: modalCount, onClick: isBackgroundClickDisabled ? null : handleModalClick, onMouseDown: isBackgroundClickDisabled ? null : handleModalOnMouseDown, role: \"dialog\", style: containerStyle, theme: theme, isOpen: isModalOpen }, containerTransition, { unmountOnExit: unmountOnExit, hasDrawerAnimation: hasDrawerAnimation }),\r\n                React.createElement(ModalContent, Object.assign({}, other, { \"data-testid\": \"modal-content\", id: contentId, ref: ref, showBackgroundOverlay: showBackgroundOverlay, theme: theme }),\r\n                    header && (React.createElement(ModalHeader, { theme: theme }, header && (React.createElement(H2, { id: headingId, isInverse: isInverse, level: headerLevel, ref: headingRef, visualStyle: TypographyVisualStyle.headingSmall, tabIndex: -1, theme: theme }, header)))),\r\n                    !isCloseButtonHidden && (React.createElement(CloseBtn, { theme: theme },\r\n                        React.createElement(IconButton, { \"aria-label\": closeAriaLabel\r\n                                ? closeAriaLabel\r\n                                : i18n.modal.closeAriaLabel, color: ButtonColor.primary, icon: CloseIconButton, isInverse: isInverse, onClick: handleClose, testId: \"modal-closebtn\", variant: ButtonVariant.link }))),\r\n                    React.createElement(ModalWrapper, { ref: bodyRef, theme: theme }, children))),\r\n            showBackgroundOverlay && (React.createElement(ModalBackdrop, { \"data-testid\": \"modal-backdrop\", onMouseDown: isBackgroundClickDisabled ? undefined : handleClose, fade: hasDrawerAnimation, isOpen: isModalOpen, style: modalCount >= 2 && { zIndex: '998' }, unmountOnExit: true, theme: theme }))), portalContainer ?? document.getElementsByTagName('body')[0])\r\n        : null;\r\n});\r\n//# sourceMappingURL=Modal.js.map"]} */"));
|
|
18884
|
+
}, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Modal.tsx"],"names":[],"mappings":"AAuB0C","file":"Modal.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css, Global } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport ReactDOM from 'react-dom';\r\nimport { CloseIcon } from 'react-magma-icons';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { I18nContext } from '../../i18n';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { magma } from '../../theme/magma';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { omit, useGenerateId, usePrevious } from '../../utils';\r\nimport { ButtonColor, ButtonVariant } from '../Button';\r\nimport { Heading } from '../Heading';\r\nimport { IconButton } from '../IconButton';\r\nimport { Transition } from '../Transition';\r\nimport { TypographyVisualStyle } from '../Typography';\r\nexport var ModalSize;\r\n(function (ModalSize) {\r\n    ModalSize[\"large\"] = \"large\";\r\n    ModalSize[\"medium\"] = \"medium\";\r\n    ModalSize[\"small\"] = \"small\";\r\n})(ModalSize || (ModalSize = {}));\r\nconst ModalContainer = styled(Transition) `\n  position: fixed;\n  bottom: 0;\n  left: 0;\n  overflow-y: auto;\n  padding: ${props => props.theme.spaceScale.spacing03};\n  right: 0;\n  top: 0;\n  z-index: ${props => (props.modalCount >= 2 ? '999' : '998')};\n`;\r\nconst ModalBackdrop = styled(Transition) `\n  backdrop-filter: blur(3px);\n  background: ${props => transparentize(0.4, props.theme.colors.neutral900)};\n  bottom: 0;\n  left: 0;\n  right: 0;\n  top: 0;\n  z-index: 997;\n  position: fixed;\n`;\r\nconst ModalContent = styled.div `\n  background: ${props => props.isInverse\r\n    ? props.theme.colors.primary600\r\n    : props.theme.colors.neutral100};\n  border: ${props => {\r\n    if (!props.showBackgroundOverlay && props.isInverse) {\r\n        return `1px solid ${transparentize(0.5, props.theme.colors.tertiary)}`;\r\n    }\r\n    return 'none';\r\n}};\n  border-radius: ${props => props.theme.borderRadius};\n  box-shadow: ${props => {\r\n    const amount = props.isInverse ? 0.82 : 0.6;\r\n    return `0 2px 6px ${transparentize(amount, props.theme.colors.neutral900)}`;\r\n}};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0 auto;\n  position: relative;\n  z-index: 1000;\n\n  max-width: ${props => {\r\n    switch (props.size) {\r\n        case 'large':\r\n            return props.theme.modal.width.large;\r\n        case 'small':\r\n            return props.theme.modal.width.small;\r\n        default:\r\n            return props.theme.modal.width.medium;\r\n    }\r\n}};\n\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    margin: ${props => props.theme.spaceScale.spacing08} auto;\n  }\n`;\r\nconst ModalHeader = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05}\n    ${props => props.theme.spaceScale.spacing05} 0\n    ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06}\n      ${props => props.theme.spaceScale.spacing06} 0\n      ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst ModalWrapper = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst H2 = styled(Heading) `\n  font-size: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.fontSize};\n  line-height: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.lineHeight};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0;\n  padding-right: ${props => props.theme.spaceScale.spacing10};\n  font-weight: 600;\n`;\r\nconst CloseBtn = styled.span `\n  position: absolute;\n  top: 0;\n  right: 0;\n  margin: ${props => props.theme.spaceScale.spacing02};\n`;\r\nexport const Modal = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, children, closeAriaLabel, closeButtonSize, portalContainer, containerStyle, containerTransition, isBackgroundClickDisabled, isEscKeyDownDisabled, header, isCloseButtonHidden, isOpen, unmountOnExit = true, testId, isModalClosingControlledManually, headerRef, onClose, hasDrawerAnimation, showBackgroundOverlay = true, headerLevel = 2, ...rest } = props;\r\n    const lastFocus = React.useRef();\r\n    const headingRef = React.useRef();\r\n    const bodyRef = React.useRef();\r\n    const id = useGenerateId(props.id);\r\n    const headingId = `${id}_heading`;\r\n    const contentId = `${id}_content`;\r\n    const [isModalOpen, setIsModalOpen] = React.useState(isOpen);\r\n    const [currentTarget, setCurrentTarget] = React.useState(null);\r\n    const [modalCount, setModalCount] = React.useState(0);\r\n    const focusTrapElement = useFocusLock(isModalOpen, headingRef, bodyRef);\r\n    const prevOpen = usePrevious(isOpen);\r\n    React.useEffect(() => {\r\n        if (isModalClosingControlledManually &&\r\n            prevOpen &&\r\n            !isOpen &&\r\n            isModalOpen) {\r\n            setIsModalOpen(false);\r\n        }\r\n        else if (!prevOpen && isOpen) {\r\n            setIsModalOpen(true);\r\n            if (headerRef && typeof headerRef === 'function') {\r\n                headerRef(headingRef);\r\n            }\r\n        }\r\n        else if (prevOpen && !isOpen && isModalOpen) {\r\n            handleClose();\r\n        }\r\n    }, [isOpen]);\r\n    React.useEffect(() => {\r\n        if (isModalOpen) {\r\n            lastFocus.current = document.activeElement;\r\n            const count = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            setModalCount(count);\r\n            if (!isEscKeyDownDisabled) {\r\n                document.body.addEventListener('keydown', handleEscapeKeyDown, false);\r\n            }\r\n        }\r\n        return () => {\r\n            document.body.removeEventListener('keydown', handleEscapeKeyDown, false);\r\n        };\r\n    }, [isModalOpen, isEscKeyDownDisabled]);\r\n    function handleModalClick(event) {\r\n        const contentEl = document.getElementById(contentId);\r\n        if (contentEl &&\r\n            !contentEl.contains(event.target) &&\r\n            event.target === currentTarget) {\r\n            handleClose(event);\r\n        }\r\n    }\r\n    function handleModalOnMouseDown(event) {\r\n        setCurrentTarget(event.target);\r\n    }\r\n    function handleEscapeKeyDown(event) {\r\n        if (event.key === 'Escape') {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            props.onEscKeyDown &&\r\n                typeof props.onEscKeyDown === 'function' &&\r\n                props.onEscKeyDown(event);\r\n            //Supports nested modals\r\n            const modalsInDom = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            if (modalCount <= 1 && modalsInDom !== 1) {\r\n                const modalEl = document.getElementById(id);\r\n                if (modalEl && modalEl.contains(event.target)) {\r\n                    handleClose(event);\r\n                }\r\n                else {\r\n                    headingRef.current?.focus();\r\n                }\r\n            }\r\n            else {\r\n                handleClose(event);\r\n            }\r\n        }\r\n    }\r\n    function handleClose(event) {\r\n        event?.stopPropagation();\r\n        setTimeout(() => {\r\n            if (!props.isModalClosingControlledManually) {\r\n                setIsModalOpen(false);\r\n            }\r\n            lastFocus.current?.focus();\r\n            onClose && typeof onClose === 'function' && onClose();\r\n        }, 0);\r\n    }\r\n    const isInverse = useIsInverse(props.isInverse);\r\n    const other = omit(['onEscKeyDown'], rest);\r\n    const theme = React.useContext(ThemeContext);\r\n    const i18n = React.useContext(I18nContext);\r\n    const CloseIconButton = (React.createElement(CloseIcon, { size: magma.iconSizes[closeButtonSize]\r\n            ? magma.iconSizes[closeButtonSize]\r\n            : magma.iconSizes.medium }));\r\n    return isModalOpen\r\n        ? ReactDOM.createPortal(React.createElement(\"div\", { ref: focusTrapElement },\r\n            React.createElement(Global, { styles: css `\n                html {\n                  overflow: ${isOpen ? 'hidden' : 'auto'};\n                }\n              ` }),\r\n            React.createElement(ModalContainer, Object.assign({ \"aria-labelledby\": header ? headingId : null, \"aria-label\": !header ? ariaLabel : null, \"aria-modal\": true, \"data-testid\": testId, id: id, modalCount: modalCount, onClick: isBackgroundClickDisabled ? null : handleModalClick, onMouseDown: isBackgroundClickDisabled ? null : handleModalOnMouseDown, role: \"dialog\", style: containerStyle, theme: theme, isOpen: isModalOpen }, containerTransition, { unmountOnExit: unmountOnExit, hasDrawerAnimation: hasDrawerAnimation }),\r\n                React.createElement(ModalContent, Object.assign({}, other, { \"data-testid\": \"modal-content\", id: contentId, ref: ref, showBackgroundOverlay: showBackgroundOverlay, theme: theme }),\r\n                    header && (React.createElement(ModalHeader, { theme: theme }, header && (React.createElement(H2, { id: headingId, isInverse: isInverse, level: headerLevel, ref: headingRef, visualStyle: TypographyVisualStyle.headingSmall, tabIndex: -1, theme: theme }, header)))),\r\n                    !isCloseButtonHidden && (React.createElement(CloseBtn, { theme: theme },\r\n                        React.createElement(IconButton, { \"aria-label\": closeAriaLabel\r\n                                ? closeAriaLabel\r\n                                : i18n.modal.closeAriaLabel, color: ButtonColor.primary, icon: CloseIconButton, isInverse: isInverse, onClick: handleClose, testId: \"modal-closebtn\", variant: ButtonVariant.link }))),\r\n                    React.createElement(ModalWrapper, { ref: bodyRef, theme: theme }, children))),\r\n            showBackgroundOverlay && (React.createElement(ModalBackdrop, { \"data-testid\": \"modal-backdrop\", onMouseDown: isBackgroundClickDisabled ? undefined : handleClose, fade: hasDrawerAnimation, isOpen: isModalOpen, style: modalCount >= 2 && { zIndex: '998' }, unmountOnExit: true, theme: theme }))), portalContainer ?? document.getElementsByTagName('body')[0])\r\n        : null;\r\n});\r\n//# sourceMappingURL=Modal.js.map"]} */"));
|
|
18731
18885
|
var ModalBackdrop = /*#__PURE__*/_styled(Transition, {
|
|
18732
18886
|
target: "e1pxemtw5",
|
|
18733
18887
|
label: "ModalBackdrop"
|
|
18734
18888
|
})("backdrop-filter:blur(3px);background:", function (props) {
|
|
18735
18889
|
return curriedTransparentize(0.4, props.theme.colors.neutral900);
|
|
18736
|
-
}, ";bottom:0;left:0;right:0;top:0;z-index:997;position:fixed;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Modal.tsx"],"names":[],"mappings":"AAiCyC","file":"Modal.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css, Global } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport ReactDOM from 'react-dom';\r\nimport { CloseIcon } from 'react-magma-icons';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { I18nContext } from '../../i18n';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { magma } from '../../theme/magma';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { omit, useGenerateId, usePrevious } from '../../utils';\r\nimport { ButtonColor, ButtonVariant } from '../Button';\r\nimport { Heading } from '../Heading';\r\nimport { IconButton } from '../IconButton';\r\nimport { Transition } from '../Transition';\r\nimport { TypographyVisualStyle } from '../Typography';\r\nexport var ModalSize;\r\n(function (ModalSize) {\r\n    ModalSize[\"large\"] = \"large\";\r\n    ModalSize[\"medium\"] = \"medium\";\r\n    ModalSize[\"small\"] = \"small\";\r\n})(ModalSize || (ModalSize = {}));\r\nconst ModalContainer = styled(Transition) `\n  position: fixed;\n  bottom: 0;\n  left: 0;\n  overflow-y: auto;\n  padding: ${props => props.theme.spaceScale.spacing03};\n  right: 0;\n  top: 0;\n  z-index: ${props => (props.modalCount >= 2 ? '999' : '998')};\n`;\r\nconst ModalBackdrop = styled(Transition) `\n  backdrop-filter: blur(3px);\n  background: ${props => transparentize(0.4, props.theme.colors.neutral900)};\n  bottom: 0;\n  left: 0;\n  right: 0;\n  top: 0;\n  z-index: 997;\n  position: fixed;\n`;\r\nconst ModalContent = styled.div `\n  background: ${props => props.isInverse\r\n    ? props.theme.colors.primary600\r\n    : props.theme.colors.neutral100};\n  border: ${props => {\r\n    if (!props.showBackgroundOverlay && props.isInverse) {\r\n        return `1px solid ${transparentize(0.5, props.theme.colors.tertiary)}`;\r\n    }\r\n    return 'none';\r\n}};\n  border-radius: ${props => props.theme.borderRadius};\n  box-shadow: ${props => {\r\n    const amount = props.isInverse ? 0.82 : 0.6;\r\n    return `0 2px 6px ${transparentize(amount, props.theme.colors.neutral900)}`;\r\n}};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0 auto;\n  position: relative;\n  z-index: 1000;\n\n  max-width: ${props => {\r\n    switch (props.size) {\r\n        case 'large':\r\n            return props.theme.modal.width.large;\r\n        case 'small':\r\n            return props.theme.modal.width.small;\r\n        default:\r\n            return props.theme.modal.width.medium;\r\n    }\r\n}};\n\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    margin: ${props => props.theme.spaceScale.spacing08} auto;\n  }\n`;\r\nconst ModalHeader = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05}\n    ${props => props.theme.spaceScale.spacing05} 0\n    ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06}\n      ${props => props.theme.spaceScale.spacing06} 0\n      ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst ModalWrapper = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst H2 = styled(Heading) `\n  font-size: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.fontSize};\n  line-height: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.lineHeight};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0;\n  padding-right: ${props => props.theme.spaceScale.spacing10};\n  font-weight: 600;\n`;\r\nconst CloseBtn = styled.span `\n  position: absolute;\n  top: 0;\n  right: 0;\n  margin: ${props => props.theme.spaceScale.spacing02};\n`;\r\nexport const Modal = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, children, closeAriaLabel, closeButtonSize, portalContainer, containerStyle, containerTransition, isBackgroundClickDisabled, isEscKeyDownDisabled, header, isCloseButtonHidden, isOpen, unmountOnExit = true, testId, isModalClosingControlledManually, headerRef, onClose, hasDrawerAnimation, showBackgroundOverlay = true, headerLevel = 2, ...rest } = props;\r\n    const lastFocus = React.useRef();\r\n    const headingRef = React.useRef();\r\n    const bodyRef = React.useRef();\r\n    const id = useGenerateId(props.id);\r\n    const headingId = `${id}_heading`;\r\n    const contentId = `${id}_content`;\r\n    const [isModalOpen, setIsModalOpen] = React.useState(isOpen);\r\n    const [currentTarget, setCurrentTarget] = React.useState(null);\r\n    const [modalCount, setModalCount] = React.useState(0);\r\n    const focusTrapElement = useFocusLock(isModalOpen, headingRef, bodyRef);\r\n    const prevOpen = usePrevious(isOpen);\r\n    React.useEffect(() => {\r\n        if (isModalClosingControlledManually &&\r\n            prevOpen &&\r\n            !isOpen &&\r\n            isModalOpen) {\r\n            setIsModalOpen(false);\r\n        }\r\n        else if (!prevOpen && isOpen) {\r\n            setIsModalOpen(true);\r\n            if (headerRef && typeof headerRef === 'function') {\r\n                headerRef(headingRef);\r\n            }\r\n        }\r\n        else if (prevOpen && !isOpen && isModalOpen) {\r\n            handleClose();\r\n        }\r\n    }, [isOpen]);\r\n    React.useEffect(() => {\r\n        if (isModalOpen) {\r\n            lastFocus.current = document.activeElement;\r\n            const count = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            setModalCount(count);\r\n            if (!isEscKeyDownDisabled) {\r\n                document.body.addEventListener('keydown', handleEscapeKeyDown, false);\r\n            }\r\n        }\r\n        return () => {\r\n            document.body.removeEventListener('keydown', handleEscapeKeyDown, false);\r\n        };\r\n    }, [isModalOpen]);\r\n    function handleModalClick(event) {\r\n        if (!document.getElementById(contentId).contains(event.target) &&\r\n            event.target === currentTarget) {\r\n            handleClose(event);\r\n        }\r\n    }\r\n    function handleModalOnMouseDown(event) {\r\n        setCurrentTarget(event.target);\r\n    }\r\n    function handleEscapeKeyDown(event) {\r\n        if (event.key === 'Escape') {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            props.onEscKeyDown &&\r\n                typeof props.onEscKeyDown === 'function' &&\r\n                props.onEscKeyDown(event);\r\n            //Supports nested modals\r\n            const modalsInDom = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            if (modalCount <= 1 && modalsInDom !== 1) {\r\n                if (document.getElementById(id).contains(event.target)) {\r\n                    handleClose(event);\r\n                }\r\n                else {\r\n                    headingRef.current.focus();\r\n                }\r\n            }\r\n            else {\r\n                handleClose(event);\r\n            }\r\n        }\r\n    }\r\n    function handleClose(event) {\r\n        event?.stopPropagation();\r\n        setTimeout(() => {\r\n            if (!props.isModalClosingControlledManually) {\r\n                setIsModalOpen(false);\r\n            }\r\n            lastFocus.current?.focus();\r\n            onClose && typeof onClose === 'function' && onClose();\r\n        }, 0);\r\n    }\r\n    const isInverse = useIsInverse(props.isInverse);\r\n    const other = omit(['onEscKeyDown'], rest);\r\n    const theme = React.useContext(ThemeContext);\r\n    const i18n = React.useContext(I18nContext);\r\n    const CloseIconButton = (React.createElement(CloseIcon, { size: magma.iconSizes[closeButtonSize]\r\n            ? magma.iconSizes[closeButtonSize]\r\n            : magma.iconSizes.medium }));\r\n    return isModalOpen\r\n        ? ReactDOM.createPortal(React.createElement(\"div\", { ref: focusTrapElement },\r\n            React.createElement(Global, { styles: css `\n                html {\n                  overflow: ${isOpen ? 'hidden' : 'auto'};\n                }\n              ` }),\r\n            React.createElement(ModalContainer, Object.assign({ \"aria-labelledby\": header ? headingId : null, \"aria-label\": !header ? ariaLabel : null, \"aria-modal\": true, \"data-testid\": testId, id: id, modalCount: modalCount, onClick: isBackgroundClickDisabled ? null : handleModalClick, onMouseDown: isBackgroundClickDisabled ? null : handleModalOnMouseDown, role: \"dialog\", style: containerStyle, theme: theme, isOpen: isModalOpen }, containerTransition, { unmountOnExit: unmountOnExit, hasDrawerAnimation: hasDrawerAnimation }),\r\n                React.createElement(ModalContent, Object.assign({}, other, { \"data-testid\": \"modal-content\", id: contentId, ref: ref, showBackgroundOverlay: showBackgroundOverlay, theme: theme }),\r\n                    header && (React.createElement(ModalHeader, { theme: theme }, header && (React.createElement(H2, { id: headingId, isInverse: isInverse, level: headerLevel, ref: headingRef, visualStyle: TypographyVisualStyle.headingSmall, tabIndex: -1, theme: theme }, header)))),\r\n                    !isCloseButtonHidden && (React.createElement(CloseBtn, { theme: theme },\r\n                        React.createElement(IconButton, { \"aria-label\": closeAriaLabel\r\n                                ? closeAriaLabel\r\n                                : i18n.modal.closeAriaLabel, color: ButtonColor.primary, icon: CloseIconButton, isInverse: isInverse, onClick: handleClose, testId: \"modal-closebtn\", variant: ButtonVariant.link }))),\r\n                    React.createElement(ModalWrapper, { ref: bodyRef, theme: theme }, children))),\r\n            showBackgroundOverlay && (React.createElement(ModalBackdrop, { \"data-testid\": \"modal-backdrop\", onMouseDown: isBackgroundClickDisabled ? undefined : handleClose, fade: hasDrawerAnimation, isOpen: isModalOpen, style: modalCount >= 2 && { zIndex: '998' }, unmountOnExit: true, theme: theme }))), portalContainer ?? document.getElementsByTagName('body')[0])\r\n        : null;\r\n});\r\n//# sourceMappingURL=Modal.js.map"]} */"));
|
|
18890
|
+
}, ";bottom:0;left:0;right:0;top:0;z-index:997;position:fixed;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Modal.tsx"],"names":[],"mappings":"AAiCyC","file":"Modal.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css, Global } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport ReactDOM from 'react-dom';\r\nimport { CloseIcon } from 'react-magma-icons';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { I18nContext } from '../../i18n';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { magma } from '../../theme/magma';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { omit, useGenerateId, usePrevious } from '../../utils';\r\nimport { ButtonColor, ButtonVariant } from '../Button';\r\nimport { Heading } from '../Heading';\r\nimport { IconButton } from '../IconButton';\r\nimport { Transition } from '../Transition';\r\nimport { TypographyVisualStyle } from '../Typography';\r\nexport var ModalSize;\r\n(function (ModalSize) {\r\n    ModalSize[\"large\"] = \"large\";\r\n    ModalSize[\"medium\"] = \"medium\";\r\n    ModalSize[\"small\"] = \"small\";\r\n})(ModalSize || (ModalSize = {}));\r\nconst ModalContainer = styled(Transition) `\n  position: fixed;\n  bottom: 0;\n  left: 0;\n  overflow-y: auto;\n  padding: ${props => props.theme.spaceScale.spacing03};\n  right: 0;\n  top: 0;\n  z-index: ${props => (props.modalCount >= 2 ? '999' : '998')};\n`;\r\nconst ModalBackdrop = styled(Transition) `\n  backdrop-filter: blur(3px);\n  background: ${props => transparentize(0.4, props.theme.colors.neutral900)};\n  bottom: 0;\n  left: 0;\n  right: 0;\n  top: 0;\n  z-index: 997;\n  position: fixed;\n`;\r\nconst ModalContent = styled.div `\n  background: ${props => props.isInverse\r\n    ? props.theme.colors.primary600\r\n    : props.theme.colors.neutral100};\n  border: ${props => {\r\n    if (!props.showBackgroundOverlay && props.isInverse) {\r\n        return `1px solid ${transparentize(0.5, props.theme.colors.tertiary)}`;\r\n    }\r\n    return 'none';\r\n}};\n  border-radius: ${props => props.theme.borderRadius};\n  box-shadow: ${props => {\r\n    const amount = props.isInverse ? 0.82 : 0.6;\r\n    return `0 2px 6px ${transparentize(amount, props.theme.colors.neutral900)}`;\r\n}};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0 auto;\n  position: relative;\n  z-index: 1000;\n\n  max-width: ${props => {\r\n    switch (props.size) {\r\n        case 'large':\r\n            return props.theme.modal.width.large;\r\n        case 'small':\r\n            return props.theme.modal.width.small;\r\n        default:\r\n            return props.theme.modal.width.medium;\r\n    }\r\n}};\n\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    margin: ${props => props.theme.spaceScale.spacing08} auto;\n  }\n`;\r\nconst ModalHeader = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05}\n    ${props => props.theme.spaceScale.spacing05} 0\n    ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06}\n      ${props => props.theme.spaceScale.spacing06} 0\n      ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst ModalWrapper = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst H2 = styled(Heading) `\n  font-size: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.fontSize};\n  line-height: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.lineHeight};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0;\n  padding-right: ${props => props.theme.spaceScale.spacing10};\n  font-weight: 600;\n`;\r\nconst CloseBtn = styled.span `\n  position: absolute;\n  top: 0;\n  right: 0;\n  margin: ${props => props.theme.spaceScale.spacing02};\n`;\r\nexport const Modal = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, children, closeAriaLabel, closeButtonSize, portalContainer, containerStyle, containerTransition, isBackgroundClickDisabled, isEscKeyDownDisabled, header, isCloseButtonHidden, isOpen, unmountOnExit = true, testId, isModalClosingControlledManually, headerRef, onClose, hasDrawerAnimation, showBackgroundOverlay = true, headerLevel = 2, ...rest } = props;\r\n    const lastFocus = React.useRef();\r\n    const headingRef = React.useRef();\r\n    const bodyRef = React.useRef();\r\n    const id = useGenerateId(props.id);\r\n    const headingId = `${id}_heading`;\r\n    const contentId = `${id}_content`;\r\n    const [isModalOpen, setIsModalOpen] = React.useState(isOpen);\r\n    const [currentTarget, setCurrentTarget] = React.useState(null);\r\n    const [modalCount, setModalCount] = React.useState(0);\r\n    const focusTrapElement = useFocusLock(isModalOpen, headingRef, bodyRef);\r\n    const prevOpen = usePrevious(isOpen);\r\n    React.useEffect(() => {\r\n        if (isModalClosingControlledManually &&\r\n            prevOpen &&\r\n            !isOpen &&\r\n            isModalOpen) {\r\n            setIsModalOpen(false);\r\n        }\r\n        else if (!prevOpen && isOpen) {\r\n            setIsModalOpen(true);\r\n            if (headerRef && typeof headerRef === 'function') {\r\n                headerRef(headingRef);\r\n            }\r\n        }\r\n        else if (prevOpen && !isOpen && isModalOpen) {\r\n            handleClose();\r\n        }\r\n    }, [isOpen]);\r\n    React.useEffect(() => {\r\n        if (isModalOpen) {\r\n            lastFocus.current = document.activeElement;\r\n            const count = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            setModalCount(count);\r\n            if (!isEscKeyDownDisabled) {\r\n                document.body.addEventListener('keydown', handleEscapeKeyDown, false);\r\n            }\r\n        }\r\n        return () => {\r\n            document.body.removeEventListener('keydown', handleEscapeKeyDown, false);\r\n        };\r\n    }, [isModalOpen, isEscKeyDownDisabled]);\r\n    function handleModalClick(event) {\r\n        const contentEl = document.getElementById(contentId);\r\n        if (contentEl &&\r\n            !contentEl.contains(event.target) &&\r\n            event.target === currentTarget) {\r\n            handleClose(event);\r\n        }\r\n    }\r\n    function handleModalOnMouseDown(event) {\r\n        setCurrentTarget(event.target);\r\n    }\r\n    function handleEscapeKeyDown(event) {\r\n        if (event.key === 'Escape') {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            props.onEscKeyDown &&\r\n                typeof props.onEscKeyDown === 'function' &&\r\n                props.onEscKeyDown(event);\r\n            //Supports nested modals\r\n            const modalsInDom = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            if (modalCount <= 1 && modalsInDom !== 1) {\r\n                const modalEl = document.getElementById(id);\r\n                if (modalEl && modalEl.contains(event.target)) {\r\n                    handleClose(event);\r\n                }\r\n                else {\r\n                    headingRef.current?.focus();\r\n                }\r\n            }\r\n            else {\r\n                handleClose(event);\r\n            }\r\n        }\r\n    }\r\n    function handleClose(event) {\r\n        event?.stopPropagation();\r\n        setTimeout(() => {\r\n            if (!props.isModalClosingControlledManually) {\r\n                setIsModalOpen(false);\r\n            }\r\n            lastFocus.current?.focus();\r\n            onClose && typeof onClose === 'function' && onClose();\r\n        }, 0);\r\n    }\r\n    const isInverse = useIsInverse(props.isInverse);\r\n    const other = omit(['onEscKeyDown'], rest);\r\n    const theme = React.useContext(ThemeContext);\r\n    const i18n = React.useContext(I18nContext);\r\n    const CloseIconButton = (React.createElement(CloseIcon, { size: magma.iconSizes[closeButtonSize]\r\n            ? magma.iconSizes[closeButtonSize]\r\n            : magma.iconSizes.medium }));\r\n    return isModalOpen\r\n        ? ReactDOM.createPortal(React.createElement(\"div\", { ref: focusTrapElement },\r\n            React.createElement(Global, { styles: css `\n                html {\n                  overflow: ${isOpen ? 'hidden' : 'auto'};\n                }\n              ` }),\r\n            React.createElement(ModalContainer, Object.assign({ \"aria-labelledby\": header ? headingId : null, \"aria-label\": !header ? ariaLabel : null, \"aria-modal\": true, \"data-testid\": testId, id: id, modalCount: modalCount, onClick: isBackgroundClickDisabled ? null : handleModalClick, onMouseDown: isBackgroundClickDisabled ? null : handleModalOnMouseDown, role: \"dialog\", style: containerStyle, theme: theme, isOpen: isModalOpen }, containerTransition, { unmountOnExit: unmountOnExit, hasDrawerAnimation: hasDrawerAnimation }),\r\n                React.createElement(ModalContent, Object.assign({}, other, { \"data-testid\": \"modal-content\", id: contentId, ref: ref, showBackgroundOverlay: showBackgroundOverlay, theme: theme }),\r\n                    header && (React.createElement(ModalHeader, { theme: theme }, header && (React.createElement(H2, { id: headingId, isInverse: isInverse, level: headerLevel, ref: headingRef, visualStyle: TypographyVisualStyle.headingSmall, tabIndex: -1, theme: theme }, header)))),\r\n                    !isCloseButtonHidden && (React.createElement(CloseBtn, { theme: theme },\r\n                        React.createElement(IconButton, { \"aria-label\": closeAriaLabel\r\n                                ? closeAriaLabel\r\n                                : i18n.modal.closeAriaLabel, color: ButtonColor.primary, icon: CloseIconButton, isInverse: isInverse, onClick: handleClose, testId: \"modal-closebtn\", variant: ButtonVariant.link }))),\r\n                    React.createElement(ModalWrapper, { ref: bodyRef, theme: theme }, children))),\r\n            showBackgroundOverlay && (React.createElement(ModalBackdrop, { \"data-testid\": \"modal-backdrop\", onMouseDown: isBackgroundClickDisabled ? undefined : handleClose, fade: hasDrawerAnimation, isOpen: isModalOpen, style: modalCount >= 2 && { zIndex: '998' }, unmountOnExit: true, theme: theme }))), portalContainer ?? document.getElementsByTagName('body')[0])\r\n        : null;\r\n});\r\n//# sourceMappingURL=Modal.js.map"]} */"));
|
|
18737
18891
|
var ModalContent = /*#__PURE__*/_styled("div", {
|
|
18738
18892
|
target: "e1pxemtw4",
|
|
18739
18893
|
label: "ModalContent"
|
|
@@ -18764,7 +18918,7 @@ var ModalContent = /*#__PURE__*/_styled("div", {
|
|
|
18764
18918
|
return props.theme.breakpoints.small;
|
|
18765
18919
|
}, "px){margin:", function (props) {
|
|
18766
18920
|
return props.theme.spaceScale.spacing08;
|
|
18767
|
-
}, " auto;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Modal.tsx"],"names":[],"mappings":"AA2CgC","file":"Modal.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css, Global } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport ReactDOM from 'react-dom';\r\nimport { CloseIcon } from 'react-magma-icons';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { I18nContext } from '../../i18n';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { magma } from '../../theme/magma';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { omit, useGenerateId, usePrevious } from '../../utils';\r\nimport { ButtonColor, ButtonVariant } from '../Button';\r\nimport { Heading } from '../Heading';\r\nimport { IconButton } from '../IconButton';\r\nimport { Transition } from '../Transition';\r\nimport { TypographyVisualStyle } from '../Typography';\r\nexport var ModalSize;\r\n(function (ModalSize) {\r\n    ModalSize[\"large\"] = \"large\";\r\n    ModalSize[\"medium\"] = \"medium\";\r\n    ModalSize[\"small\"] = \"small\";\r\n})(ModalSize || (ModalSize = {}));\r\nconst ModalContainer = styled(Transition) `\n  position: fixed;\n  bottom: 0;\n  left: 0;\n  overflow-y: auto;\n  padding: ${props => props.theme.spaceScale.spacing03};\n  right: 0;\n  top: 0;\n  z-index: ${props => (props.modalCount >= 2 ? '999' : '998')};\n`;\r\nconst ModalBackdrop = styled(Transition) `\n  backdrop-filter: blur(3px);\n  background: ${props => transparentize(0.4, props.theme.colors.neutral900)};\n  bottom: 0;\n  left: 0;\n  right: 0;\n  top: 0;\n  z-index: 997;\n  position: fixed;\n`;\r\nconst ModalContent = styled.div `\n  background: ${props => props.isInverse\r\n    ? props.theme.colors.primary600\r\n    : props.theme.colors.neutral100};\n  border: ${props => {\r\n    if (!props.showBackgroundOverlay && props.isInverse) {\r\n        return `1px solid ${transparentize(0.5, props.theme.colors.tertiary)}`;\r\n    }\r\n    return 'none';\r\n}};\n  border-radius: ${props => props.theme.borderRadius};\n  box-shadow: ${props => {\r\n    const amount = props.isInverse ? 0.82 : 0.6;\r\n    return `0 2px 6px ${transparentize(amount, props.theme.colors.neutral900)}`;\r\n}};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0 auto;\n  position: relative;\n  z-index: 1000;\n\n  max-width: ${props => {\r\n    switch (props.size) {\r\n        case 'large':\r\n            return props.theme.modal.width.large;\r\n        case 'small':\r\n            return props.theme.modal.width.small;\r\n        default:\r\n            return props.theme.modal.width.medium;\r\n    }\r\n}};\n\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    margin: ${props => props.theme.spaceScale.spacing08} auto;\n  }\n`;\r\nconst ModalHeader = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05}\n    ${props => props.theme.spaceScale.spacing05} 0\n    ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06}\n      ${props => props.theme.spaceScale.spacing06} 0\n      ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst ModalWrapper = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst H2 = styled(Heading) `\n  font-size: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.fontSize};\n  line-height: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.lineHeight};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0;\n  padding-right: ${props => props.theme.spaceScale.spacing10};\n  font-weight: 600;\n`;\r\nconst CloseBtn = styled.span `\n  position: absolute;\n  top: 0;\n  right: 0;\n  margin: ${props => props.theme.spaceScale.spacing02};\n`;\r\nexport const Modal = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, children, closeAriaLabel, closeButtonSize, portalContainer, containerStyle, containerTransition, isBackgroundClickDisabled, isEscKeyDownDisabled, header, isCloseButtonHidden, isOpen, unmountOnExit = true, testId, isModalClosingControlledManually, headerRef, onClose, hasDrawerAnimation, showBackgroundOverlay = true, headerLevel = 2, ...rest } = props;\r\n    const lastFocus = React.useRef();\r\n    const headingRef = React.useRef();\r\n    const bodyRef = React.useRef();\r\n    const id = useGenerateId(props.id);\r\n    const headingId = `${id}_heading`;\r\n    const contentId = `${id}_content`;\r\n    const [isModalOpen, setIsModalOpen] = React.useState(isOpen);\r\n    const [currentTarget, setCurrentTarget] = React.useState(null);\r\n    const [modalCount, setModalCount] = React.useState(0);\r\n    const focusTrapElement = useFocusLock(isModalOpen, headingRef, bodyRef);\r\n    const prevOpen = usePrevious(isOpen);\r\n    React.useEffect(() => {\r\n        if (isModalClosingControlledManually &&\r\n            prevOpen &&\r\n            !isOpen &&\r\n            isModalOpen) {\r\n            setIsModalOpen(false);\r\n        }\r\n        else if (!prevOpen && isOpen) {\r\n            setIsModalOpen(true);\r\n            if (headerRef && typeof headerRef === 'function') {\r\n                headerRef(headingRef);\r\n            }\r\n        }\r\n        else if (prevOpen && !isOpen && isModalOpen) {\r\n            handleClose();\r\n        }\r\n    }, [isOpen]);\r\n    React.useEffect(() => {\r\n        if (isModalOpen) {\r\n            lastFocus.current = document.activeElement;\r\n            const count = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            setModalCount(count);\r\n            if (!isEscKeyDownDisabled) {\r\n                document.body.addEventListener('keydown', handleEscapeKeyDown, false);\r\n            }\r\n        }\r\n        return () => {\r\n            document.body.removeEventListener('keydown', handleEscapeKeyDown, false);\r\n        };\r\n    }, [isModalOpen]);\r\n    function handleModalClick(event) {\r\n        if (!document.getElementById(contentId).contains(event.target) &&\r\n            event.target === currentTarget) {\r\n            handleClose(event);\r\n        }\r\n    }\r\n    function handleModalOnMouseDown(event) {\r\n        setCurrentTarget(event.target);\r\n    }\r\n    function handleEscapeKeyDown(event) {\r\n        if (event.key === 'Escape') {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            props.onEscKeyDown &&\r\n                typeof props.onEscKeyDown === 'function' &&\r\n                props.onEscKeyDown(event);\r\n            //Supports nested modals\r\n            const modalsInDom = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            if (modalCount <= 1 && modalsInDom !== 1) {\r\n                if (document.getElementById(id).contains(event.target)) {\r\n                    handleClose(event);\r\n                }\r\n                else {\r\n                    headingRef.current.focus();\r\n                }\r\n            }\r\n            else {\r\n                handleClose(event);\r\n            }\r\n        }\r\n    }\r\n    function handleClose(event) {\r\n        event?.stopPropagation();\r\n        setTimeout(() => {\r\n            if (!props.isModalClosingControlledManually) {\r\n                setIsModalOpen(false);\r\n            }\r\n            lastFocus.current?.focus();\r\n            onClose && typeof onClose === 'function' && onClose();\r\n        }, 0);\r\n    }\r\n    const isInverse = useIsInverse(props.isInverse);\r\n    const other = omit(['onEscKeyDown'], rest);\r\n    const theme = React.useContext(ThemeContext);\r\n    const i18n = React.useContext(I18nContext);\r\n    const CloseIconButton = (React.createElement(CloseIcon, { size: magma.iconSizes[closeButtonSize]\r\n            ? magma.iconSizes[closeButtonSize]\r\n            : magma.iconSizes.medium }));\r\n    return isModalOpen\r\n        ? ReactDOM.createPortal(React.createElement(\"div\", { ref: focusTrapElement },\r\n            React.createElement(Global, { styles: css `\n                html {\n                  overflow: ${isOpen ? 'hidden' : 'auto'};\n                }\n              ` }),\r\n            React.createElement(ModalContainer, Object.assign({ \"aria-labelledby\": header ? headingId : null, \"aria-label\": !header ? ariaLabel : null, \"aria-modal\": true, \"data-testid\": testId, id: id, modalCount: modalCount, onClick: isBackgroundClickDisabled ? null : handleModalClick, onMouseDown: isBackgroundClickDisabled ? null : handleModalOnMouseDown, role: \"dialog\", style: containerStyle, theme: theme, isOpen: isModalOpen }, containerTransition, { unmountOnExit: unmountOnExit, hasDrawerAnimation: hasDrawerAnimation }),\r\n                React.createElement(ModalContent, Object.assign({}, other, { \"data-testid\": \"modal-content\", id: contentId, ref: ref, showBackgroundOverlay: showBackgroundOverlay, theme: theme }),\r\n                    header && (React.createElement(ModalHeader, { theme: theme }, header && (React.createElement(H2, { id: headingId, isInverse: isInverse, level: headerLevel, ref: headingRef, visualStyle: TypographyVisualStyle.headingSmall, tabIndex: -1, theme: theme }, header)))),\r\n                    !isCloseButtonHidden && (React.createElement(CloseBtn, { theme: theme },\r\n                        React.createElement(IconButton, { \"aria-label\": closeAriaLabel\r\n                                ? closeAriaLabel\r\n                                : i18n.modal.closeAriaLabel, color: ButtonColor.primary, icon: CloseIconButton, isInverse: isInverse, onClick: handleClose, testId: \"modal-closebtn\", variant: ButtonVariant.link }))),\r\n                    React.createElement(ModalWrapper, { ref: bodyRef, theme: theme }, children))),\r\n            showBackgroundOverlay && (React.createElement(ModalBackdrop, { \"data-testid\": \"modal-backdrop\", onMouseDown: isBackgroundClickDisabled ? undefined : handleClose, fade: hasDrawerAnimation, isOpen: isModalOpen, style: modalCount >= 2 && { zIndex: '998' }, unmountOnExit: true, theme: theme }))), portalContainer ?? document.getElementsByTagName('body')[0])\r\n        : null;\r\n});\r\n//# sourceMappingURL=Modal.js.map"]} */"));
|
|
18921
|
+
}, " auto;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Modal.tsx"],"names":[],"mappings":"AA2CgC","file":"Modal.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css, Global } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport ReactDOM from 'react-dom';\r\nimport { CloseIcon } from 'react-magma-icons';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { I18nContext } from '../../i18n';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { magma } from '../../theme/magma';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { omit, useGenerateId, usePrevious } from '../../utils';\r\nimport { ButtonColor, ButtonVariant } from '../Button';\r\nimport { Heading } from '../Heading';\r\nimport { IconButton } from '../IconButton';\r\nimport { Transition } from '../Transition';\r\nimport { TypographyVisualStyle } from '../Typography';\r\nexport var ModalSize;\r\n(function (ModalSize) {\r\n    ModalSize[\"large\"] = \"large\";\r\n    ModalSize[\"medium\"] = \"medium\";\r\n    ModalSize[\"small\"] = \"small\";\r\n})(ModalSize || (ModalSize = {}));\r\nconst ModalContainer = styled(Transition) `\n  position: fixed;\n  bottom: 0;\n  left: 0;\n  overflow-y: auto;\n  padding: ${props => props.theme.spaceScale.spacing03};\n  right: 0;\n  top: 0;\n  z-index: ${props => (props.modalCount >= 2 ? '999' : '998')};\n`;\r\nconst ModalBackdrop = styled(Transition) `\n  backdrop-filter: blur(3px);\n  background: ${props => transparentize(0.4, props.theme.colors.neutral900)};\n  bottom: 0;\n  left: 0;\n  right: 0;\n  top: 0;\n  z-index: 997;\n  position: fixed;\n`;\r\nconst ModalContent = styled.div `\n  background: ${props => props.isInverse\r\n    ? props.theme.colors.primary600\r\n    : props.theme.colors.neutral100};\n  border: ${props => {\r\n    if (!props.showBackgroundOverlay && props.isInverse) {\r\n        return `1px solid ${transparentize(0.5, props.theme.colors.tertiary)}`;\r\n    }\r\n    return 'none';\r\n}};\n  border-radius: ${props => props.theme.borderRadius};\n  box-shadow: ${props => {\r\n    const amount = props.isInverse ? 0.82 : 0.6;\r\n    return `0 2px 6px ${transparentize(amount, props.theme.colors.neutral900)}`;\r\n}};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0 auto;\n  position: relative;\n  z-index: 1000;\n\n  max-width: ${props => {\r\n    switch (props.size) {\r\n        case 'large':\r\n            return props.theme.modal.width.large;\r\n        case 'small':\r\n            return props.theme.modal.width.small;\r\n        default:\r\n            return props.theme.modal.width.medium;\r\n    }\r\n}};\n\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    margin: ${props => props.theme.spaceScale.spacing08} auto;\n  }\n`;\r\nconst ModalHeader = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05}\n    ${props => props.theme.spaceScale.spacing05} 0\n    ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06}\n      ${props => props.theme.spaceScale.spacing06} 0\n      ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst ModalWrapper = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst H2 = styled(Heading) `\n  font-size: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.fontSize};\n  line-height: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.lineHeight};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0;\n  padding-right: ${props => props.theme.spaceScale.spacing10};\n  font-weight: 600;\n`;\r\nconst CloseBtn = styled.span `\n  position: absolute;\n  top: 0;\n  right: 0;\n  margin: ${props => props.theme.spaceScale.spacing02};\n`;\r\nexport const Modal = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, children, closeAriaLabel, closeButtonSize, portalContainer, containerStyle, containerTransition, isBackgroundClickDisabled, isEscKeyDownDisabled, header, isCloseButtonHidden, isOpen, unmountOnExit = true, testId, isModalClosingControlledManually, headerRef, onClose, hasDrawerAnimation, showBackgroundOverlay = true, headerLevel = 2, ...rest } = props;\r\n    const lastFocus = React.useRef();\r\n    const headingRef = React.useRef();\r\n    const bodyRef = React.useRef();\r\n    const id = useGenerateId(props.id);\r\n    const headingId = `${id}_heading`;\r\n    const contentId = `${id}_content`;\r\n    const [isModalOpen, setIsModalOpen] = React.useState(isOpen);\r\n    const [currentTarget, setCurrentTarget] = React.useState(null);\r\n    const [modalCount, setModalCount] = React.useState(0);\r\n    const focusTrapElement = useFocusLock(isModalOpen, headingRef, bodyRef);\r\n    const prevOpen = usePrevious(isOpen);\r\n    React.useEffect(() => {\r\n        if (isModalClosingControlledManually &&\r\n            prevOpen &&\r\n            !isOpen &&\r\n            isModalOpen) {\r\n            setIsModalOpen(false);\r\n        }\r\n        else if (!prevOpen && isOpen) {\r\n            setIsModalOpen(true);\r\n            if (headerRef && typeof headerRef === 'function') {\r\n                headerRef(headingRef);\r\n            }\r\n        }\r\n        else if (prevOpen && !isOpen && isModalOpen) {\r\n            handleClose();\r\n        }\r\n    }, [isOpen]);\r\n    React.useEffect(() => {\r\n        if (isModalOpen) {\r\n            lastFocus.current = document.activeElement;\r\n            const count = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            setModalCount(count);\r\n            if (!isEscKeyDownDisabled) {\r\n                document.body.addEventListener('keydown', handleEscapeKeyDown, false);\r\n            }\r\n        }\r\n        return () => {\r\n            document.body.removeEventListener('keydown', handleEscapeKeyDown, false);\r\n        };\r\n    }, [isModalOpen, isEscKeyDownDisabled]);\r\n    function handleModalClick(event) {\r\n        const contentEl = document.getElementById(contentId);\r\n        if (contentEl &&\r\n            !contentEl.contains(event.target) &&\r\n            event.target === currentTarget) {\r\n            handleClose(event);\r\n        }\r\n    }\r\n    function handleModalOnMouseDown(event) {\r\n        setCurrentTarget(event.target);\r\n    }\r\n    function handleEscapeKeyDown(event) {\r\n        if (event.key === 'Escape') {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            props.onEscKeyDown &&\r\n                typeof props.onEscKeyDown === 'function' &&\r\n                props.onEscKeyDown(event);\r\n            //Supports nested modals\r\n            const modalsInDom = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            if (modalCount <= 1 && modalsInDom !== 1) {\r\n                const modalEl = document.getElementById(id);\r\n                if (modalEl && modalEl.contains(event.target)) {\r\n                    handleClose(event);\r\n                }\r\n                else {\r\n                    headingRef.current?.focus();\r\n                }\r\n            }\r\n            else {\r\n                handleClose(event);\r\n            }\r\n        }\r\n    }\r\n    function handleClose(event) {\r\n        event?.stopPropagation();\r\n        setTimeout(() => {\r\n            if (!props.isModalClosingControlledManually) {\r\n                setIsModalOpen(false);\r\n            }\r\n            lastFocus.current?.focus();\r\n            onClose && typeof onClose === 'function' && onClose();\r\n        }, 0);\r\n    }\r\n    const isInverse = useIsInverse(props.isInverse);\r\n    const other = omit(['onEscKeyDown'], rest);\r\n    const theme = React.useContext(ThemeContext);\r\n    const i18n = React.useContext(I18nContext);\r\n    const CloseIconButton = (React.createElement(CloseIcon, { size: magma.iconSizes[closeButtonSize]\r\n            ? magma.iconSizes[closeButtonSize]\r\n            : magma.iconSizes.medium }));\r\n    return isModalOpen\r\n        ? ReactDOM.createPortal(React.createElement(\"div\", { ref: focusTrapElement },\r\n            React.createElement(Global, { styles: css `\n                html {\n                  overflow: ${isOpen ? 'hidden' : 'auto'};\n                }\n              ` }),\r\n            React.createElement(ModalContainer, Object.assign({ \"aria-labelledby\": header ? headingId : null, \"aria-label\": !header ? ariaLabel : null, \"aria-modal\": true, \"data-testid\": testId, id: id, modalCount: modalCount, onClick: isBackgroundClickDisabled ? null : handleModalClick, onMouseDown: isBackgroundClickDisabled ? null : handleModalOnMouseDown, role: \"dialog\", style: containerStyle, theme: theme, isOpen: isModalOpen }, containerTransition, { unmountOnExit: unmountOnExit, hasDrawerAnimation: hasDrawerAnimation }),\r\n                React.createElement(ModalContent, Object.assign({}, other, { \"data-testid\": \"modal-content\", id: contentId, ref: ref, showBackgroundOverlay: showBackgroundOverlay, theme: theme }),\r\n                    header && (React.createElement(ModalHeader, { theme: theme }, header && (React.createElement(H2, { id: headingId, isInverse: isInverse, level: headerLevel, ref: headingRef, visualStyle: TypographyVisualStyle.headingSmall, tabIndex: -1, theme: theme }, header)))),\r\n                    !isCloseButtonHidden && (React.createElement(CloseBtn, { theme: theme },\r\n                        React.createElement(IconButton, { \"aria-label\": closeAriaLabel\r\n                                ? closeAriaLabel\r\n                                : i18n.modal.closeAriaLabel, color: ButtonColor.primary, icon: CloseIconButton, isInverse: isInverse, onClick: handleClose, testId: \"modal-closebtn\", variant: ButtonVariant.link }))),\r\n                    React.createElement(ModalWrapper, { ref: bodyRef, theme: theme }, children))),\r\n            showBackgroundOverlay && (React.createElement(ModalBackdrop, { \"data-testid\": \"modal-backdrop\", onMouseDown: isBackgroundClickDisabled ? undefined : handleClose, fade: hasDrawerAnimation, isOpen: isModalOpen, style: modalCount >= 2 && { zIndex: '998' }, unmountOnExit: true, theme: theme }))), portalContainer ?? document.getElementsByTagName('body')[0])\r\n        : null;\r\n});\r\n//# sourceMappingURL=Modal.js.map"]} */"));
|
|
18768
18922
|
var ModalHeader = /*#__PURE__*/_styled("div", {
|
|
18769
18923
|
target: "e1pxemtw3",
|
|
18770
18924
|
label: "ModalHeader"
|
|
@@ -18782,7 +18936,7 @@ var ModalHeader = /*#__PURE__*/_styled("div", {
|
|
|
18782
18936
|
return props.theme.spaceScale.spacing06;
|
|
18783
18937
|
}, " 0 ", function (props) {
|
|
18784
18938
|
return props.theme.spaceScale.spacing06;
|
|
18785
|
-
}, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Modal.tsx"],"names":[],"mappings":"AAgF+B","file":"Modal.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css, Global } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport ReactDOM from 'react-dom';\r\nimport { CloseIcon } from 'react-magma-icons';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { I18nContext } from '../../i18n';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { magma } from '../../theme/magma';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { omit, useGenerateId, usePrevious } from '../../utils';\r\nimport { ButtonColor, ButtonVariant } from '../Button';\r\nimport { Heading } from '../Heading';\r\nimport { IconButton } from '../IconButton';\r\nimport { Transition } from '../Transition';\r\nimport { TypographyVisualStyle } from '../Typography';\r\nexport var ModalSize;\r\n(function (ModalSize) {\r\n    ModalSize[\"large\"] = \"large\";\r\n    ModalSize[\"medium\"] = \"medium\";\r\n    ModalSize[\"small\"] = \"small\";\r\n})(ModalSize || (ModalSize = {}));\r\nconst ModalContainer = styled(Transition) `\n  position: fixed;\n  bottom: 0;\n  left: 0;\n  overflow-y: auto;\n  padding: ${props => props.theme.spaceScale.spacing03};\n  right: 0;\n  top: 0;\n  z-index: ${props => (props.modalCount >= 2 ? '999' : '998')};\n`;\r\nconst ModalBackdrop = styled(Transition) `\n  backdrop-filter: blur(3px);\n  background: ${props => transparentize(0.4, props.theme.colors.neutral900)};\n  bottom: 0;\n  left: 0;\n  right: 0;\n  top: 0;\n  z-index: 997;\n  position: fixed;\n`;\r\nconst ModalContent = styled.div `\n  background: ${props => props.isInverse\r\n    ? props.theme.colors.primary600\r\n    : props.theme.colors.neutral100};\n  border: ${props => {\r\n    if (!props.showBackgroundOverlay && props.isInverse) {\r\n        return `1px solid ${transparentize(0.5, props.theme.colors.tertiary)}`;\r\n    }\r\n    return 'none';\r\n}};\n  border-radius: ${props => props.theme.borderRadius};\n  box-shadow: ${props => {\r\n    const amount = props.isInverse ? 0.82 : 0.6;\r\n    return `0 2px 6px ${transparentize(amount, props.theme.colors.neutral900)}`;\r\n}};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0 auto;\n  position: relative;\n  z-index: 1000;\n\n  max-width: ${props => {\r\n    switch (props.size) {\r\n        case 'large':\r\n            return props.theme.modal.width.large;\r\n        case 'small':\r\n            return props.theme.modal.width.small;\r\n        default:\r\n            return props.theme.modal.width.medium;\r\n    }\r\n}};\n\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    margin: ${props => props.theme.spaceScale.spacing08} auto;\n  }\n`;\r\nconst ModalHeader = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05}\n    ${props => props.theme.spaceScale.spacing05} 0\n    ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06}\n      ${props => props.theme.spaceScale.spacing06} 0\n      ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst ModalWrapper = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst H2 = styled(Heading) `\n  font-size: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.fontSize};\n  line-height: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.lineHeight};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0;\n  padding-right: ${props => props.theme.spaceScale.spacing10};\n  font-weight: 600;\n`;\r\nconst CloseBtn = styled.span `\n  position: absolute;\n  top: 0;\n  right: 0;\n  margin: ${props => props.theme.spaceScale.spacing02};\n`;\r\nexport const Modal = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, children, closeAriaLabel, closeButtonSize, portalContainer, containerStyle, containerTransition, isBackgroundClickDisabled, isEscKeyDownDisabled, header, isCloseButtonHidden, isOpen, unmountOnExit = true, testId, isModalClosingControlledManually, headerRef, onClose, hasDrawerAnimation, showBackgroundOverlay = true, headerLevel = 2, ...rest } = props;\r\n    const lastFocus = React.useRef();\r\n    const headingRef = React.useRef();\r\n    const bodyRef = React.useRef();\r\n    const id = useGenerateId(props.id);\r\n    const headingId = `${id}_heading`;\r\n    const contentId = `${id}_content`;\r\n    const [isModalOpen, setIsModalOpen] = React.useState(isOpen);\r\n    const [currentTarget, setCurrentTarget] = React.useState(null);\r\n    const [modalCount, setModalCount] = React.useState(0);\r\n    const focusTrapElement = useFocusLock(isModalOpen, headingRef, bodyRef);\r\n    const prevOpen = usePrevious(isOpen);\r\n    React.useEffect(() => {\r\n        if (isModalClosingControlledManually &&\r\n            prevOpen &&\r\n            !isOpen &&\r\n            isModalOpen) {\r\n            setIsModalOpen(false);\r\n        }\r\n        else if (!prevOpen && isOpen) {\r\n            setIsModalOpen(true);\r\n            if (headerRef && typeof headerRef === 'function') {\r\n                headerRef(headingRef);\r\n            }\r\n        }\r\n        else if (prevOpen && !isOpen && isModalOpen) {\r\n            handleClose();\r\n        }\r\n    }, [isOpen]);\r\n    React.useEffect(() => {\r\n        if (isModalOpen) {\r\n            lastFocus.current = document.activeElement;\r\n            const count = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            setModalCount(count);\r\n            if (!isEscKeyDownDisabled) {\r\n                document.body.addEventListener('keydown', handleEscapeKeyDown, false);\r\n            }\r\n        }\r\n        return () => {\r\n            document.body.removeEventListener('keydown', handleEscapeKeyDown, false);\r\n        };\r\n    }, [isModalOpen]);\r\n    function handleModalClick(event) {\r\n        if (!document.getElementById(contentId).contains(event.target) &&\r\n            event.target === currentTarget) {\r\n            handleClose(event);\r\n        }\r\n    }\r\n    function handleModalOnMouseDown(event) {\r\n        setCurrentTarget(event.target);\r\n    }\r\n    function handleEscapeKeyDown(event) {\r\n        if (event.key === 'Escape') {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            props.onEscKeyDown &&\r\n                typeof props.onEscKeyDown === 'function' &&\r\n                props.onEscKeyDown(event);\r\n            //Supports nested modals\r\n            const modalsInDom = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            if (modalCount <= 1 && modalsInDom !== 1) {\r\n                if (document.getElementById(id).contains(event.target)) {\r\n                    handleClose(event);\r\n                }\r\n                else {\r\n                    headingRef.current.focus();\r\n                }\r\n            }\r\n            else {\r\n                handleClose(event);\r\n            }\r\n        }\r\n    }\r\n    function handleClose(event) {\r\n        event?.stopPropagation();\r\n        setTimeout(() => {\r\n            if (!props.isModalClosingControlledManually) {\r\n                setIsModalOpen(false);\r\n            }\r\n            lastFocus.current?.focus();\r\n            onClose && typeof onClose === 'function' && onClose();\r\n        }, 0);\r\n    }\r\n    const isInverse = useIsInverse(props.isInverse);\r\n    const other = omit(['onEscKeyDown'], rest);\r\n    const theme = React.useContext(ThemeContext);\r\n    const i18n = React.useContext(I18nContext);\r\n    const CloseIconButton = (React.createElement(CloseIcon, { size: magma.iconSizes[closeButtonSize]\r\n            ? magma.iconSizes[closeButtonSize]\r\n            : magma.iconSizes.medium }));\r\n    return isModalOpen\r\n        ? ReactDOM.createPortal(React.createElement(\"div\", { ref: focusTrapElement },\r\n            React.createElement(Global, { styles: css `\n                html {\n                  overflow: ${isOpen ? 'hidden' : 'auto'};\n                }\n              ` }),\r\n            React.createElement(ModalContainer, Object.assign({ \"aria-labelledby\": header ? headingId : null, \"aria-label\": !header ? ariaLabel : null, \"aria-modal\": true, \"data-testid\": testId, id: id, modalCount: modalCount, onClick: isBackgroundClickDisabled ? null : handleModalClick, onMouseDown: isBackgroundClickDisabled ? null : handleModalOnMouseDown, role: \"dialog\", style: containerStyle, theme: theme, isOpen: isModalOpen }, containerTransition, { unmountOnExit: unmountOnExit, hasDrawerAnimation: hasDrawerAnimation }),\r\n                React.createElement(ModalContent, Object.assign({}, other, { \"data-testid\": \"modal-content\", id: contentId, ref: ref, showBackgroundOverlay: showBackgroundOverlay, theme: theme }),\r\n                    header && (React.createElement(ModalHeader, { theme: theme }, header && (React.createElement(H2, { id: headingId, isInverse: isInverse, level: headerLevel, ref: headingRef, visualStyle: TypographyVisualStyle.headingSmall, tabIndex: -1, theme: theme }, header)))),\r\n                    !isCloseButtonHidden && (React.createElement(CloseBtn, { theme: theme },\r\n                        React.createElement(IconButton, { \"aria-label\": closeAriaLabel\r\n                                ? closeAriaLabel\r\n                                : i18n.modal.closeAriaLabel, color: ButtonColor.primary, icon: CloseIconButton, isInverse: isInverse, onClick: handleClose, testId: \"modal-closebtn\", variant: ButtonVariant.link }))),\r\n                    React.createElement(ModalWrapper, { ref: bodyRef, theme: theme }, children))),\r\n            showBackgroundOverlay && (React.createElement(ModalBackdrop, { \"data-testid\": \"modal-backdrop\", onMouseDown: isBackgroundClickDisabled ? undefined : handleClose, fade: hasDrawerAnimation, isOpen: isModalOpen, style: modalCount >= 2 && { zIndex: '998' }, unmountOnExit: true, theme: theme }))), portalContainer ?? document.getElementsByTagName('body')[0])\r\n        : null;\r\n});\r\n//# sourceMappingURL=Modal.js.map"]} */"));
|
|
18939
|
+
}, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Modal.tsx"],"names":[],"mappings":"AAgF+B","file":"Modal.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css, Global } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport ReactDOM from 'react-dom';\r\nimport { CloseIcon } from 'react-magma-icons';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { I18nContext } from '../../i18n';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { magma } from '../../theme/magma';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { omit, useGenerateId, usePrevious } from '../../utils';\r\nimport { ButtonColor, ButtonVariant } from '../Button';\r\nimport { Heading } from '../Heading';\r\nimport { IconButton } from '../IconButton';\r\nimport { Transition } from '../Transition';\r\nimport { TypographyVisualStyle } from '../Typography';\r\nexport var ModalSize;\r\n(function (ModalSize) {\r\n    ModalSize[\"large\"] = \"large\";\r\n    ModalSize[\"medium\"] = \"medium\";\r\n    ModalSize[\"small\"] = \"small\";\r\n})(ModalSize || (ModalSize = {}));\r\nconst ModalContainer = styled(Transition) `\n  position: fixed;\n  bottom: 0;\n  left: 0;\n  overflow-y: auto;\n  padding: ${props => props.theme.spaceScale.spacing03};\n  right: 0;\n  top: 0;\n  z-index: ${props => (props.modalCount >= 2 ? '999' : '998')};\n`;\r\nconst ModalBackdrop = styled(Transition) `\n  backdrop-filter: blur(3px);\n  background: ${props => transparentize(0.4, props.theme.colors.neutral900)};\n  bottom: 0;\n  left: 0;\n  right: 0;\n  top: 0;\n  z-index: 997;\n  position: fixed;\n`;\r\nconst ModalContent = styled.div `\n  background: ${props => props.isInverse\r\n    ? props.theme.colors.primary600\r\n    : props.theme.colors.neutral100};\n  border: ${props => {\r\n    if (!props.showBackgroundOverlay && props.isInverse) {\r\n        return `1px solid ${transparentize(0.5, props.theme.colors.tertiary)}`;\r\n    }\r\n    return 'none';\r\n}};\n  border-radius: ${props => props.theme.borderRadius};\n  box-shadow: ${props => {\r\n    const amount = props.isInverse ? 0.82 : 0.6;\r\n    return `0 2px 6px ${transparentize(amount, props.theme.colors.neutral900)}`;\r\n}};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0 auto;\n  position: relative;\n  z-index: 1000;\n\n  max-width: ${props => {\r\n    switch (props.size) {\r\n        case 'large':\r\n            return props.theme.modal.width.large;\r\n        case 'small':\r\n            return props.theme.modal.width.small;\r\n        default:\r\n            return props.theme.modal.width.medium;\r\n    }\r\n}};\n\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    margin: ${props => props.theme.spaceScale.spacing08} auto;\n  }\n`;\r\nconst ModalHeader = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05}\n    ${props => props.theme.spaceScale.spacing05} 0\n    ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06}\n      ${props => props.theme.spaceScale.spacing06} 0\n      ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst ModalWrapper = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst H2 = styled(Heading) `\n  font-size: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.fontSize};\n  line-height: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.lineHeight};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0;\n  padding-right: ${props => props.theme.spaceScale.spacing10};\n  font-weight: 600;\n`;\r\nconst CloseBtn = styled.span `\n  position: absolute;\n  top: 0;\n  right: 0;\n  margin: ${props => props.theme.spaceScale.spacing02};\n`;\r\nexport const Modal = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, children, closeAriaLabel, closeButtonSize, portalContainer, containerStyle, containerTransition, isBackgroundClickDisabled, isEscKeyDownDisabled, header, isCloseButtonHidden, isOpen, unmountOnExit = true, testId, isModalClosingControlledManually, headerRef, onClose, hasDrawerAnimation, showBackgroundOverlay = true, headerLevel = 2, ...rest } = props;\r\n    const lastFocus = React.useRef();\r\n    const headingRef = React.useRef();\r\n    const bodyRef = React.useRef();\r\n    const id = useGenerateId(props.id);\r\n    const headingId = `${id}_heading`;\r\n    const contentId = `${id}_content`;\r\n    const [isModalOpen, setIsModalOpen] = React.useState(isOpen);\r\n    const [currentTarget, setCurrentTarget] = React.useState(null);\r\n    const [modalCount, setModalCount] = React.useState(0);\r\n    const focusTrapElement = useFocusLock(isModalOpen, headingRef, bodyRef);\r\n    const prevOpen = usePrevious(isOpen);\r\n    React.useEffect(() => {\r\n        if (isModalClosingControlledManually &&\r\n            prevOpen &&\r\n            !isOpen &&\r\n            isModalOpen) {\r\n            setIsModalOpen(false);\r\n        }\r\n        else if (!prevOpen && isOpen) {\r\n            setIsModalOpen(true);\r\n            if (headerRef && typeof headerRef === 'function') {\r\n                headerRef(headingRef);\r\n            }\r\n        }\r\n        else if (prevOpen && !isOpen && isModalOpen) {\r\n            handleClose();\r\n        }\r\n    }, [isOpen]);\r\n    React.useEffect(() => {\r\n        if (isModalOpen) {\r\n            lastFocus.current = document.activeElement;\r\n            const count = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            setModalCount(count);\r\n            if (!isEscKeyDownDisabled) {\r\n                document.body.addEventListener('keydown', handleEscapeKeyDown, false);\r\n            }\r\n        }\r\n        return () => {\r\n            document.body.removeEventListener('keydown', handleEscapeKeyDown, false);\r\n        };\r\n    }, [isModalOpen, isEscKeyDownDisabled]);\r\n    function handleModalClick(event) {\r\n        const contentEl = document.getElementById(contentId);\r\n        if (contentEl &&\r\n            !contentEl.contains(event.target) &&\r\n            event.target === currentTarget) {\r\n            handleClose(event);\r\n        }\r\n    }\r\n    function handleModalOnMouseDown(event) {\r\n        setCurrentTarget(event.target);\r\n    }\r\n    function handleEscapeKeyDown(event) {\r\n        if (event.key === 'Escape') {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            props.onEscKeyDown &&\r\n                typeof props.onEscKeyDown === 'function' &&\r\n                props.onEscKeyDown(event);\r\n            //Supports nested modals\r\n            const modalsInDom = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            if (modalCount <= 1 && modalsInDom !== 1) {\r\n                const modalEl = document.getElementById(id);\r\n                if (modalEl && modalEl.contains(event.target)) {\r\n                    handleClose(event);\r\n                }\r\n                else {\r\n                    headingRef.current?.focus();\r\n                }\r\n            }\r\n            else {\r\n                handleClose(event);\r\n            }\r\n        }\r\n    }\r\n    function handleClose(event) {\r\n        event?.stopPropagation();\r\n        setTimeout(() => {\r\n            if (!props.isModalClosingControlledManually) {\r\n                setIsModalOpen(false);\r\n            }\r\n            lastFocus.current?.focus();\r\n            onClose && typeof onClose === 'function' && onClose();\r\n        }, 0);\r\n    }\r\n    const isInverse = useIsInverse(props.isInverse);\r\n    const other = omit(['onEscKeyDown'], rest);\r\n    const theme = React.useContext(ThemeContext);\r\n    const i18n = React.useContext(I18nContext);\r\n    const CloseIconButton = (React.createElement(CloseIcon, { size: magma.iconSizes[closeButtonSize]\r\n            ? magma.iconSizes[closeButtonSize]\r\n            : magma.iconSizes.medium }));\r\n    return isModalOpen\r\n        ? ReactDOM.createPortal(React.createElement(\"div\", { ref: focusTrapElement },\r\n            React.createElement(Global, { styles: css `\n                html {\n                  overflow: ${isOpen ? 'hidden' : 'auto'};\n                }\n              ` }),\r\n            React.createElement(ModalContainer, Object.assign({ \"aria-labelledby\": header ? headingId : null, \"aria-label\": !header ? ariaLabel : null, \"aria-modal\": true, \"data-testid\": testId, id: id, modalCount: modalCount, onClick: isBackgroundClickDisabled ? null : handleModalClick, onMouseDown: isBackgroundClickDisabled ? null : handleModalOnMouseDown, role: \"dialog\", style: containerStyle, theme: theme, isOpen: isModalOpen }, containerTransition, { unmountOnExit: unmountOnExit, hasDrawerAnimation: hasDrawerAnimation }),\r\n                React.createElement(ModalContent, Object.assign({}, other, { \"data-testid\": \"modal-content\", id: contentId, ref: ref, showBackgroundOverlay: showBackgroundOverlay, theme: theme }),\r\n                    header && (React.createElement(ModalHeader, { theme: theme }, header && (React.createElement(H2, { id: headingId, isInverse: isInverse, level: headerLevel, ref: headingRef, visualStyle: TypographyVisualStyle.headingSmall, tabIndex: -1, theme: theme }, header)))),\r\n                    !isCloseButtonHidden && (React.createElement(CloseBtn, { theme: theme },\r\n                        React.createElement(IconButton, { \"aria-label\": closeAriaLabel\r\n                                ? closeAriaLabel\r\n                                : i18n.modal.closeAriaLabel, color: ButtonColor.primary, icon: CloseIconButton, isInverse: isInverse, onClick: handleClose, testId: \"modal-closebtn\", variant: ButtonVariant.link }))),\r\n                    React.createElement(ModalWrapper, { ref: bodyRef, theme: theme }, children))),\r\n            showBackgroundOverlay && (React.createElement(ModalBackdrop, { \"data-testid\": \"modal-backdrop\", onMouseDown: isBackgroundClickDisabled ? undefined : handleClose, fade: hasDrawerAnimation, isOpen: isModalOpen, style: modalCount >= 2 && { zIndex: '998' }, unmountOnExit: true, theme: theme }))), portalContainer ?? document.getElementsByTagName('body')[0])\r\n        : null;\r\n});\r\n//# sourceMappingURL=Modal.js.map"]} */"));
|
|
18786
18940
|
var ModalWrapper = /*#__PURE__*/_styled("div", {
|
|
18787
18941
|
target: "e1pxemtw2",
|
|
18788
18942
|
label: "ModalWrapper"
|
|
@@ -18792,7 +18946,7 @@ var ModalWrapper = /*#__PURE__*/_styled("div", {
|
|
|
18792
18946
|
return props.theme.breakpoints.small;
|
|
18793
18947
|
}, "px){padding:", function (props) {
|
|
18794
18948
|
return props.theme.spaceScale.spacing06;
|
|
18795
|
-
}, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Modal.tsx"],"names":[],"mappings":"AA0FgC","file":"Modal.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css, Global } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport ReactDOM from 'react-dom';\r\nimport { CloseIcon } from 'react-magma-icons';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { I18nContext } from '../../i18n';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { magma } from '../../theme/magma';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { omit, useGenerateId, usePrevious } from '../../utils';\r\nimport { ButtonColor, ButtonVariant } from '../Button';\r\nimport { Heading } from '../Heading';\r\nimport { IconButton } from '../IconButton';\r\nimport { Transition } from '../Transition';\r\nimport { TypographyVisualStyle } from '../Typography';\r\nexport var ModalSize;\r\n(function (ModalSize) {\r\n    ModalSize[\"large\"] = \"large\";\r\n    ModalSize[\"medium\"] = \"medium\";\r\n    ModalSize[\"small\"] = \"small\";\r\n})(ModalSize || (ModalSize = {}));\r\nconst ModalContainer = styled(Transition) `\n  position: fixed;\n  bottom: 0;\n  left: 0;\n  overflow-y: auto;\n  padding: ${props => props.theme.spaceScale.spacing03};\n  right: 0;\n  top: 0;\n  z-index: ${props => (props.modalCount >= 2 ? '999' : '998')};\n`;\r\nconst ModalBackdrop = styled(Transition) `\n  backdrop-filter: blur(3px);\n  background: ${props => transparentize(0.4, props.theme.colors.neutral900)};\n  bottom: 0;\n  left: 0;\n  right: 0;\n  top: 0;\n  z-index: 997;\n  position: fixed;\n`;\r\nconst ModalContent = styled.div `\n  background: ${props => props.isInverse\r\n    ? props.theme.colors.primary600\r\n    : props.theme.colors.neutral100};\n  border: ${props => {\r\n    if (!props.showBackgroundOverlay && props.isInverse) {\r\n        return `1px solid ${transparentize(0.5, props.theme.colors.tertiary)}`;\r\n    }\r\n    return 'none';\r\n}};\n  border-radius: ${props => props.theme.borderRadius};\n  box-shadow: ${props => {\r\n    const amount = props.isInverse ? 0.82 : 0.6;\r\n    return `0 2px 6px ${transparentize(amount, props.theme.colors.neutral900)}`;\r\n}};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0 auto;\n  position: relative;\n  z-index: 1000;\n\n  max-width: ${props => {\r\n    switch (props.size) {\r\n        case 'large':\r\n            return props.theme.modal.width.large;\r\n        case 'small':\r\n            return props.theme.modal.width.small;\r\n        default:\r\n            return props.theme.modal.width.medium;\r\n    }\r\n}};\n\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    margin: ${props => props.theme.spaceScale.spacing08} auto;\n  }\n`;\r\nconst ModalHeader = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05}\n    ${props => props.theme.spaceScale.spacing05} 0\n    ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06}\n      ${props => props.theme.spaceScale.spacing06} 0\n      ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst ModalWrapper = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst H2 = styled(Heading) `\n  font-size: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.fontSize};\n  line-height: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.lineHeight};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0;\n  padding-right: ${props => props.theme.spaceScale.spacing10};\n  font-weight: 600;\n`;\r\nconst CloseBtn = styled.span `\n  position: absolute;\n  top: 0;\n  right: 0;\n  margin: ${props => props.theme.spaceScale.spacing02};\n`;\r\nexport const Modal = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, children, closeAriaLabel, closeButtonSize, portalContainer, containerStyle, containerTransition, isBackgroundClickDisabled, isEscKeyDownDisabled, header, isCloseButtonHidden, isOpen, unmountOnExit = true, testId, isModalClosingControlledManually, headerRef, onClose, hasDrawerAnimation, showBackgroundOverlay = true, headerLevel = 2, ...rest } = props;\r\n    const lastFocus = React.useRef();\r\n    const headingRef = React.useRef();\r\n    const bodyRef = React.useRef();\r\n    const id = useGenerateId(props.id);\r\n    const headingId = `${id}_heading`;\r\n    const contentId = `${id}_content`;\r\n    const [isModalOpen, setIsModalOpen] = React.useState(isOpen);\r\n    const [currentTarget, setCurrentTarget] = React.useState(null);\r\n    const [modalCount, setModalCount] = React.useState(0);\r\n    const focusTrapElement = useFocusLock(isModalOpen, headingRef, bodyRef);\r\n    const prevOpen = usePrevious(isOpen);\r\n    React.useEffect(() => {\r\n        if (isModalClosingControlledManually &&\r\n            prevOpen &&\r\n            !isOpen &&\r\n            isModalOpen) {\r\n            setIsModalOpen(false);\r\n        }\r\n        else if (!prevOpen && isOpen) {\r\n            setIsModalOpen(true);\r\n            if (headerRef && typeof headerRef === 'function') {\r\n                headerRef(headingRef);\r\n            }\r\n        }\r\n        else if (prevOpen && !isOpen && isModalOpen) {\r\n            handleClose();\r\n        }\r\n    }, [isOpen]);\r\n    React.useEffect(() => {\r\n        if (isModalOpen) {\r\n            lastFocus.current = document.activeElement;\r\n            const count = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            setModalCount(count);\r\n            if (!isEscKeyDownDisabled) {\r\n                document.body.addEventListener('keydown', handleEscapeKeyDown, false);\r\n            }\r\n        }\r\n        return () => {\r\n            document.body.removeEventListener('keydown', handleEscapeKeyDown, false);\r\n        };\r\n    }, [isModalOpen]);\r\n    function handleModalClick(event) {\r\n        if (!document.getElementById(contentId).contains(event.target) &&\r\n            event.target === currentTarget) {\r\n            handleClose(event);\r\n        }\r\n    }\r\n    function handleModalOnMouseDown(event) {\r\n        setCurrentTarget(event.target);\r\n    }\r\n    function handleEscapeKeyDown(event) {\r\n        if (event.key === 'Escape') {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            props.onEscKeyDown &&\r\n                typeof props.onEscKeyDown === 'function' &&\r\n                props.onEscKeyDown(event);\r\n            //Supports nested modals\r\n            const modalsInDom = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            if (modalCount <= 1 && modalsInDom !== 1) {\r\n                if (document.getElementById(id).contains(event.target)) {\r\n                    handleClose(event);\r\n                }\r\n                else {\r\n                    headingRef.current.focus();\r\n                }\r\n            }\r\n            else {\r\n                handleClose(event);\r\n            }\r\n        }\r\n    }\r\n    function handleClose(event) {\r\n        event?.stopPropagation();\r\n        setTimeout(() => {\r\n            if (!props.isModalClosingControlledManually) {\r\n                setIsModalOpen(false);\r\n            }\r\n            lastFocus.current?.focus();\r\n            onClose && typeof onClose === 'function' && onClose();\r\n        }, 0);\r\n    }\r\n    const isInverse = useIsInverse(props.isInverse);\r\n    const other = omit(['onEscKeyDown'], rest);\r\n    const theme = React.useContext(ThemeContext);\r\n    const i18n = React.useContext(I18nContext);\r\n    const CloseIconButton = (React.createElement(CloseIcon, { size: magma.iconSizes[closeButtonSize]\r\n            ? magma.iconSizes[closeButtonSize]\r\n            : magma.iconSizes.medium }));\r\n    return isModalOpen\r\n        ? ReactDOM.createPortal(React.createElement(\"div\", { ref: focusTrapElement },\r\n            React.createElement(Global, { styles: css `\n                html {\n                  overflow: ${isOpen ? 'hidden' : 'auto'};\n                }\n              ` }),\r\n            React.createElement(ModalContainer, Object.assign({ \"aria-labelledby\": header ? headingId : null, \"aria-label\": !header ? ariaLabel : null, \"aria-modal\": true, \"data-testid\": testId, id: id, modalCount: modalCount, onClick: isBackgroundClickDisabled ? null : handleModalClick, onMouseDown: isBackgroundClickDisabled ? null : handleModalOnMouseDown, role: \"dialog\", style: containerStyle, theme: theme, isOpen: isModalOpen }, containerTransition, { unmountOnExit: unmountOnExit, hasDrawerAnimation: hasDrawerAnimation }),\r\n                React.createElement(ModalContent, Object.assign({}, other, { \"data-testid\": \"modal-content\", id: contentId, ref: ref, showBackgroundOverlay: showBackgroundOverlay, theme: theme }),\r\n                    header && (React.createElement(ModalHeader, { theme: theme }, header && (React.createElement(H2, { id: headingId, isInverse: isInverse, level: headerLevel, ref: headingRef, visualStyle: TypographyVisualStyle.headingSmall, tabIndex: -1, theme: theme }, header)))),\r\n                    !isCloseButtonHidden && (React.createElement(CloseBtn, { theme: theme },\r\n                        React.createElement(IconButton, { \"aria-label\": closeAriaLabel\r\n                                ? closeAriaLabel\r\n                                : i18n.modal.closeAriaLabel, color: ButtonColor.primary, icon: CloseIconButton, isInverse: isInverse, onClick: handleClose, testId: \"modal-closebtn\", variant: ButtonVariant.link }))),\r\n                    React.createElement(ModalWrapper, { ref: bodyRef, theme: theme }, children))),\r\n            showBackgroundOverlay && (React.createElement(ModalBackdrop, { \"data-testid\": \"modal-backdrop\", onMouseDown: isBackgroundClickDisabled ? undefined : handleClose, fade: hasDrawerAnimation, isOpen: isModalOpen, style: modalCount >= 2 && { zIndex: '998' }, unmountOnExit: true, theme: theme }))), portalContainer ?? document.getElementsByTagName('body')[0])\r\n        : null;\r\n});\r\n//# sourceMappingURL=Modal.js.map"]} */"));
|
|
18949
|
+
}, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Modal.tsx"],"names":[],"mappings":"AA0FgC","file":"Modal.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css, Global } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport ReactDOM from 'react-dom';\r\nimport { CloseIcon } from 'react-magma-icons';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { I18nContext } from '../../i18n';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { magma } from '../../theme/magma';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { omit, useGenerateId, usePrevious } from '../../utils';\r\nimport { ButtonColor, ButtonVariant } from '../Button';\r\nimport { Heading } from '../Heading';\r\nimport { IconButton } from '../IconButton';\r\nimport { Transition } from '../Transition';\r\nimport { TypographyVisualStyle } from '../Typography';\r\nexport var ModalSize;\r\n(function (ModalSize) {\r\n    ModalSize[\"large\"] = \"large\";\r\n    ModalSize[\"medium\"] = \"medium\";\r\n    ModalSize[\"small\"] = \"small\";\r\n})(ModalSize || (ModalSize = {}));\r\nconst ModalContainer = styled(Transition) `\n  position: fixed;\n  bottom: 0;\n  left: 0;\n  overflow-y: auto;\n  padding: ${props => props.theme.spaceScale.spacing03};\n  right: 0;\n  top: 0;\n  z-index: ${props => (props.modalCount >= 2 ? '999' : '998')};\n`;\r\nconst ModalBackdrop = styled(Transition) `\n  backdrop-filter: blur(3px);\n  background: ${props => transparentize(0.4, props.theme.colors.neutral900)};\n  bottom: 0;\n  left: 0;\n  right: 0;\n  top: 0;\n  z-index: 997;\n  position: fixed;\n`;\r\nconst ModalContent = styled.div `\n  background: ${props => props.isInverse\r\n    ? props.theme.colors.primary600\r\n    : props.theme.colors.neutral100};\n  border: ${props => {\r\n    if (!props.showBackgroundOverlay && props.isInverse) {\r\n        return `1px solid ${transparentize(0.5, props.theme.colors.tertiary)}`;\r\n    }\r\n    return 'none';\r\n}};\n  border-radius: ${props => props.theme.borderRadius};\n  box-shadow: ${props => {\r\n    const amount = props.isInverse ? 0.82 : 0.6;\r\n    return `0 2px 6px ${transparentize(amount, props.theme.colors.neutral900)}`;\r\n}};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0 auto;\n  position: relative;\n  z-index: 1000;\n\n  max-width: ${props => {\r\n    switch (props.size) {\r\n        case 'large':\r\n            return props.theme.modal.width.large;\r\n        case 'small':\r\n            return props.theme.modal.width.small;\r\n        default:\r\n            return props.theme.modal.width.medium;\r\n    }\r\n}};\n\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    margin: ${props => props.theme.spaceScale.spacing08} auto;\n  }\n`;\r\nconst ModalHeader = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05}\n    ${props => props.theme.spaceScale.spacing05} 0\n    ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06}\n      ${props => props.theme.spaceScale.spacing06} 0\n      ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst ModalWrapper = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst H2 = styled(Heading) `\n  font-size: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.fontSize};\n  line-height: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.lineHeight};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0;\n  padding-right: ${props => props.theme.spaceScale.spacing10};\n  font-weight: 600;\n`;\r\nconst CloseBtn = styled.span `\n  position: absolute;\n  top: 0;\n  right: 0;\n  margin: ${props => props.theme.spaceScale.spacing02};\n`;\r\nexport const Modal = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, children, closeAriaLabel, closeButtonSize, portalContainer, containerStyle, containerTransition, isBackgroundClickDisabled, isEscKeyDownDisabled, header, isCloseButtonHidden, isOpen, unmountOnExit = true, testId, isModalClosingControlledManually, headerRef, onClose, hasDrawerAnimation, showBackgroundOverlay = true, headerLevel = 2, ...rest } = props;\r\n    const lastFocus = React.useRef();\r\n    const headingRef = React.useRef();\r\n    const bodyRef = React.useRef();\r\n    const id = useGenerateId(props.id);\r\n    const headingId = `${id}_heading`;\r\n    const contentId = `${id}_content`;\r\n    const [isModalOpen, setIsModalOpen] = React.useState(isOpen);\r\n    const [currentTarget, setCurrentTarget] = React.useState(null);\r\n    const [modalCount, setModalCount] = React.useState(0);\r\n    const focusTrapElement = useFocusLock(isModalOpen, headingRef, bodyRef);\r\n    const prevOpen = usePrevious(isOpen);\r\n    React.useEffect(() => {\r\n        if (isModalClosingControlledManually &&\r\n            prevOpen &&\r\n            !isOpen &&\r\n            isModalOpen) {\r\n            setIsModalOpen(false);\r\n        }\r\n        else if (!prevOpen && isOpen) {\r\n            setIsModalOpen(true);\r\n            if (headerRef && typeof headerRef === 'function') {\r\n                headerRef(headingRef);\r\n            }\r\n        }\r\n        else if (prevOpen && !isOpen && isModalOpen) {\r\n            handleClose();\r\n        }\r\n    }, [isOpen]);\r\n    React.useEffect(() => {\r\n        if (isModalOpen) {\r\n            lastFocus.current = document.activeElement;\r\n            const count = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            setModalCount(count);\r\n            if (!isEscKeyDownDisabled) {\r\n                document.body.addEventListener('keydown', handleEscapeKeyDown, false);\r\n            }\r\n        }\r\n        return () => {\r\n            document.body.removeEventListener('keydown', handleEscapeKeyDown, false);\r\n        };\r\n    }, [isModalOpen, isEscKeyDownDisabled]);\r\n    function handleModalClick(event) {\r\n        const contentEl = document.getElementById(contentId);\r\n        if (contentEl &&\r\n            !contentEl.contains(event.target) &&\r\n            event.target === currentTarget) {\r\n            handleClose(event);\r\n        }\r\n    }\r\n    function handleModalOnMouseDown(event) {\r\n        setCurrentTarget(event.target);\r\n    }\r\n    function handleEscapeKeyDown(event) {\r\n        if (event.key === 'Escape') {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            props.onEscKeyDown &&\r\n                typeof props.onEscKeyDown === 'function' &&\r\n                props.onEscKeyDown(event);\r\n            //Supports nested modals\r\n            const modalsInDom = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            if (modalCount <= 1 && modalsInDom !== 1) {\r\n                const modalEl = document.getElementById(id);\r\n                if (modalEl && modalEl.contains(event.target)) {\r\n                    handleClose(event);\r\n                }\r\n                else {\r\n                    headingRef.current?.focus();\r\n                }\r\n            }\r\n            else {\r\n                handleClose(event);\r\n            }\r\n        }\r\n    }\r\n    function handleClose(event) {\r\n        event?.stopPropagation();\r\n        setTimeout(() => {\r\n            if (!props.isModalClosingControlledManually) {\r\n                setIsModalOpen(false);\r\n            }\r\n            lastFocus.current?.focus();\r\n            onClose && typeof onClose === 'function' && onClose();\r\n        }, 0);\r\n    }\r\n    const isInverse = useIsInverse(props.isInverse);\r\n    const other = omit(['onEscKeyDown'], rest);\r\n    const theme = React.useContext(ThemeContext);\r\n    const i18n = React.useContext(I18nContext);\r\n    const CloseIconButton = (React.createElement(CloseIcon, { size: magma.iconSizes[closeButtonSize]\r\n            ? magma.iconSizes[closeButtonSize]\r\n            : magma.iconSizes.medium }));\r\n    return isModalOpen\r\n        ? ReactDOM.createPortal(React.createElement(\"div\", { ref: focusTrapElement },\r\n            React.createElement(Global, { styles: css `\n                html {\n                  overflow: ${isOpen ? 'hidden' : 'auto'};\n                }\n              ` }),\r\n            React.createElement(ModalContainer, Object.assign({ \"aria-labelledby\": header ? headingId : null, \"aria-label\": !header ? ariaLabel : null, \"aria-modal\": true, \"data-testid\": testId, id: id, modalCount: modalCount, onClick: isBackgroundClickDisabled ? null : handleModalClick, onMouseDown: isBackgroundClickDisabled ? null : handleModalOnMouseDown, role: \"dialog\", style: containerStyle, theme: theme, isOpen: isModalOpen }, containerTransition, { unmountOnExit: unmountOnExit, hasDrawerAnimation: hasDrawerAnimation }),\r\n                React.createElement(ModalContent, Object.assign({}, other, { \"data-testid\": \"modal-content\", id: contentId, ref: ref, showBackgroundOverlay: showBackgroundOverlay, theme: theme }),\r\n                    header && (React.createElement(ModalHeader, { theme: theme }, header && (React.createElement(H2, { id: headingId, isInverse: isInverse, level: headerLevel, ref: headingRef, visualStyle: TypographyVisualStyle.headingSmall, tabIndex: -1, theme: theme }, header)))),\r\n                    !isCloseButtonHidden && (React.createElement(CloseBtn, { theme: theme },\r\n                        React.createElement(IconButton, { \"aria-label\": closeAriaLabel\r\n                                ? closeAriaLabel\r\n                                : i18n.modal.closeAriaLabel, color: ButtonColor.primary, icon: CloseIconButton, isInverse: isInverse, onClick: handleClose, testId: \"modal-closebtn\", variant: ButtonVariant.link }))),\r\n                    React.createElement(ModalWrapper, { ref: bodyRef, theme: theme }, children))),\r\n            showBackgroundOverlay && (React.createElement(ModalBackdrop, { \"data-testid\": \"modal-backdrop\", onMouseDown: isBackgroundClickDisabled ? undefined : handleClose, fade: hasDrawerAnimation, isOpen: isModalOpen, style: modalCount >= 2 && { zIndex: '998' }, unmountOnExit: true, theme: theme }))), portalContainer ?? document.getElementsByTagName('body')[0])\r\n        : null;\r\n});\r\n//# sourceMappingURL=Modal.js.map"]} */"));
|
|
18796
18950
|
var H2 = /*#__PURE__*/_styled(Heading, {
|
|
18797
18951
|
target: "e1pxemtw1",
|
|
18798
18952
|
label: "H2"
|
|
@@ -18804,13 +18958,13 @@ var H2 = /*#__PURE__*/_styled(Heading, {
|
|
|
18804
18958
|
return props.isInverse ? props.theme.colors.neutral100 : props.theme.colors.neutral700;
|
|
18805
18959
|
}, ";margin:0;padding-right:", function (props) {
|
|
18806
18960
|
return props.theme.spaceScale.spacing10;
|
|
18807
|
-
}, ";font-weight:600;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Modal.tsx"],"names":[],"mappings":"AAgG2B","file":"Modal.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css, Global } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport ReactDOM from 'react-dom';\r\nimport { CloseIcon } from 'react-magma-icons';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { I18nContext } from '../../i18n';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { magma } from '../../theme/magma';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { omit, useGenerateId, usePrevious } from '../../utils';\r\nimport { ButtonColor, ButtonVariant } from '../Button';\r\nimport { Heading } from '../Heading';\r\nimport { IconButton } from '../IconButton';\r\nimport { Transition } from '../Transition';\r\nimport { TypographyVisualStyle } from '../Typography';\r\nexport var ModalSize;\r\n(function (ModalSize) {\r\n    ModalSize[\"large\"] = \"large\";\r\n    ModalSize[\"medium\"] = \"medium\";\r\n    ModalSize[\"small\"] = \"small\";\r\n})(ModalSize || (ModalSize = {}));\r\nconst ModalContainer = styled(Transition) `\n  position: fixed;\n  bottom: 0;\n  left: 0;\n  overflow-y: auto;\n  padding: ${props => props.theme.spaceScale.spacing03};\n  right: 0;\n  top: 0;\n  z-index: ${props => (props.modalCount >= 2 ? '999' : '998')};\n`;\r\nconst ModalBackdrop = styled(Transition) `\n  backdrop-filter: blur(3px);\n  background: ${props => transparentize(0.4, props.theme.colors.neutral900)};\n  bottom: 0;\n  left: 0;\n  right: 0;\n  top: 0;\n  z-index: 997;\n  position: fixed;\n`;\r\nconst ModalContent = styled.div `\n  background: ${props => props.isInverse\r\n    ? props.theme.colors.primary600\r\n    : props.theme.colors.neutral100};\n  border: ${props => {\r\n    if (!props.showBackgroundOverlay && props.isInverse) {\r\n        return `1px solid ${transparentize(0.5, props.theme.colors.tertiary)}`;\r\n    }\r\n    return 'none';\r\n}};\n  border-radius: ${props => props.theme.borderRadius};\n  box-shadow: ${props => {\r\n    const amount = props.isInverse ? 0.82 : 0.6;\r\n    return `0 2px 6px ${transparentize(amount, props.theme.colors.neutral900)}`;\r\n}};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0 auto;\n  position: relative;\n  z-index: 1000;\n\n  max-width: ${props => {\r\n    switch (props.size) {\r\n        case 'large':\r\n            return props.theme.modal.width.large;\r\n        case 'small':\r\n            return props.theme.modal.width.small;\r\n        default:\r\n            return props.theme.modal.width.medium;\r\n    }\r\n}};\n\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    margin: ${props => props.theme.spaceScale.spacing08} auto;\n  }\n`;\r\nconst ModalHeader = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05}\n    ${props => props.theme.spaceScale.spacing05} 0\n    ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06}\n      ${props => props.theme.spaceScale.spacing06} 0\n      ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst ModalWrapper = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst H2 = styled(Heading) `\n  font-size: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.fontSize};\n  line-height: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.lineHeight};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0;\n  padding-right: ${props => props.theme.spaceScale.spacing10};\n  font-weight: 600;\n`;\r\nconst CloseBtn = styled.span `\n  position: absolute;\n  top: 0;\n  right: 0;\n  margin: ${props => props.theme.spaceScale.spacing02};\n`;\r\nexport const Modal = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, children, closeAriaLabel, closeButtonSize, portalContainer, containerStyle, containerTransition, isBackgroundClickDisabled, isEscKeyDownDisabled, header, isCloseButtonHidden, isOpen, unmountOnExit = true, testId, isModalClosingControlledManually, headerRef, onClose, hasDrawerAnimation, showBackgroundOverlay = true, headerLevel = 2, ...rest } = props;\r\n    const lastFocus = React.useRef();\r\n    const headingRef = React.useRef();\r\n    const bodyRef = React.useRef();\r\n    const id = useGenerateId(props.id);\r\n    const headingId = `${id}_heading`;\r\n    const contentId = `${id}_content`;\r\n    const [isModalOpen, setIsModalOpen] = React.useState(isOpen);\r\n    const [currentTarget, setCurrentTarget] = React.useState(null);\r\n    const [modalCount, setModalCount] = React.useState(0);\r\n    const focusTrapElement = useFocusLock(isModalOpen, headingRef, bodyRef);\r\n    const prevOpen = usePrevious(isOpen);\r\n    React.useEffect(() => {\r\n        if (isModalClosingControlledManually &&\r\n            prevOpen &&\r\n            !isOpen &&\r\n            isModalOpen) {\r\n            setIsModalOpen(false);\r\n        }\r\n        else if (!prevOpen && isOpen) {\r\n            setIsModalOpen(true);\r\n            if (headerRef && typeof headerRef === 'function') {\r\n                headerRef(headingRef);\r\n            }\r\n        }\r\n        else if (prevOpen && !isOpen && isModalOpen) {\r\n            handleClose();\r\n        }\r\n    }, [isOpen]);\r\n    React.useEffect(() => {\r\n        if (isModalOpen) {\r\n            lastFocus.current = document.activeElement;\r\n            const count = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            setModalCount(count);\r\n            if (!isEscKeyDownDisabled) {\r\n                document.body.addEventListener('keydown', handleEscapeKeyDown, false);\r\n            }\r\n        }\r\n        return () => {\r\n            document.body.removeEventListener('keydown', handleEscapeKeyDown, false);\r\n        };\r\n    }, [isModalOpen]);\r\n    function handleModalClick(event) {\r\n        if (!document.getElementById(contentId).contains(event.target) &&\r\n            event.target === currentTarget) {\r\n            handleClose(event);\r\n        }\r\n    }\r\n    function handleModalOnMouseDown(event) {\r\n        setCurrentTarget(event.target);\r\n    }\r\n    function handleEscapeKeyDown(event) {\r\n        if (event.key === 'Escape') {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            props.onEscKeyDown &&\r\n                typeof props.onEscKeyDown === 'function' &&\r\n                props.onEscKeyDown(event);\r\n            //Supports nested modals\r\n            const modalsInDom = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            if (modalCount <= 1 && modalsInDom !== 1) {\r\n                if (document.getElementById(id).contains(event.target)) {\r\n                    handleClose(event);\r\n                }\r\n                else {\r\n                    headingRef.current.focus();\r\n                }\r\n            }\r\n            else {\r\n                handleClose(event);\r\n            }\r\n        }\r\n    }\r\n    function handleClose(event) {\r\n        event?.stopPropagation();\r\n        setTimeout(() => {\r\n            if (!props.isModalClosingControlledManually) {\r\n                setIsModalOpen(false);\r\n            }\r\n            lastFocus.current?.focus();\r\n            onClose && typeof onClose === 'function' && onClose();\r\n        }, 0);\r\n    }\r\n    const isInverse = useIsInverse(props.isInverse);\r\n    const other = omit(['onEscKeyDown'], rest);\r\n    const theme = React.useContext(ThemeContext);\r\n    const i18n = React.useContext(I18nContext);\r\n    const CloseIconButton = (React.createElement(CloseIcon, { size: magma.iconSizes[closeButtonSize]\r\n            ? magma.iconSizes[closeButtonSize]\r\n            : magma.iconSizes.medium }));\r\n    return isModalOpen\r\n        ? ReactDOM.createPortal(React.createElement(\"div\", { ref: focusTrapElement },\r\n            React.createElement(Global, { styles: css `\n                html {\n                  overflow: ${isOpen ? 'hidden' : 'auto'};\n                }\n              ` }),\r\n            React.createElement(ModalContainer, Object.assign({ \"aria-labelledby\": header ? headingId : null, \"aria-label\": !header ? ariaLabel : null, \"aria-modal\": true, \"data-testid\": testId, id: id, modalCount: modalCount, onClick: isBackgroundClickDisabled ? null : handleModalClick, onMouseDown: isBackgroundClickDisabled ? null : handleModalOnMouseDown, role: \"dialog\", style: containerStyle, theme: theme, isOpen: isModalOpen }, containerTransition, { unmountOnExit: unmountOnExit, hasDrawerAnimation: hasDrawerAnimation }),\r\n                React.createElement(ModalContent, Object.assign({}, other, { \"data-testid\": \"modal-content\", id: contentId, ref: ref, showBackgroundOverlay: showBackgroundOverlay, theme: theme }),\r\n                    header && (React.createElement(ModalHeader, { theme: theme }, header && (React.createElement(H2, { id: headingId, isInverse: isInverse, level: headerLevel, ref: headingRef, visualStyle: TypographyVisualStyle.headingSmall, tabIndex: -1, theme: theme }, header)))),\r\n                    !isCloseButtonHidden && (React.createElement(CloseBtn, { theme: theme },\r\n                        React.createElement(IconButton, { \"aria-label\": closeAriaLabel\r\n                                ? closeAriaLabel\r\n                                : i18n.modal.closeAriaLabel, color: ButtonColor.primary, icon: CloseIconButton, isInverse: isInverse, onClick: handleClose, testId: \"modal-closebtn\", variant: ButtonVariant.link }))),\r\n                    React.createElement(ModalWrapper, { ref: bodyRef, theme: theme }, children))),\r\n            showBackgroundOverlay && (React.createElement(ModalBackdrop, { \"data-testid\": \"modal-backdrop\", onMouseDown: isBackgroundClickDisabled ? undefined : handleClose, fade: hasDrawerAnimation, isOpen: isModalOpen, style: modalCount >= 2 && { zIndex: '998' }, unmountOnExit: true, theme: theme }))), portalContainer ?? document.getElementsByTagName('body')[0])\r\n        : null;\r\n});\r\n//# sourceMappingURL=Modal.js.map"]} */"));
|
|
18961
|
+
}, ";font-weight:600;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Modal.tsx"],"names":[],"mappings":"AAgG2B","file":"Modal.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css, Global } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport ReactDOM from 'react-dom';\r\nimport { CloseIcon } from 'react-magma-icons';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { I18nContext } from '../../i18n';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { magma } from '../../theme/magma';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { omit, useGenerateId, usePrevious } from '../../utils';\r\nimport { ButtonColor, ButtonVariant } from '../Button';\r\nimport { Heading } from '../Heading';\r\nimport { IconButton } from '../IconButton';\r\nimport { Transition } from '../Transition';\r\nimport { TypographyVisualStyle } from '../Typography';\r\nexport var ModalSize;\r\n(function (ModalSize) {\r\n    ModalSize[\"large\"] = \"large\";\r\n    ModalSize[\"medium\"] = \"medium\";\r\n    ModalSize[\"small\"] = \"small\";\r\n})(ModalSize || (ModalSize = {}));\r\nconst ModalContainer = styled(Transition) `\n  position: fixed;\n  bottom: 0;\n  left: 0;\n  overflow-y: auto;\n  padding: ${props => props.theme.spaceScale.spacing03};\n  right: 0;\n  top: 0;\n  z-index: ${props => (props.modalCount >= 2 ? '999' : '998')};\n`;\r\nconst ModalBackdrop = styled(Transition) `\n  backdrop-filter: blur(3px);\n  background: ${props => transparentize(0.4, props.theme.colors.neutral900)};\n  bottom: 0;\n  left: 0;\n  right: 0;\n  top: 0;\n  z-index: 997;\n  position: fixed;\n`;\r\nconst ModalContent = styled.div `\n  background: ${props => props.isInverse\r\n    ? props.theme.colors.primary600\r\n    : props.theme.colors.neutral100};\n  border: ${props => {\r\n    if (!props.showBackgroundOverlay && props.isInverse) {\r\n        return `1px solid ${transparentize(0.5, props.theme.colors.tertiary)}`;\r\n    }\r\n    return 'none';\r\n}};\n  border-radius: ${props => props.theme.borderRadius};\n  box-shadow: ${props => {\r\n    const amount = props.isInverse ? 0.82 : 0.6;\r\n    return `0 2px 6px ${transparentize(amount, props.theme.colors.neutral900)}`;\r\n}};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0 auto;\n  position: relative;\n  z-index: 1000;\n\n  max-width: ${props => {\r\n    switch (props.size) {\r\n        case 'large':\r\n            return props.theme.modal.width.large;\r\n        case 'small':\r\n            return props.theme.modal.width.small;\r\n        default:\r\n            return props.theme.modal.width.medium;\r\n    }\r\n}};\n\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    margin: ${props => props.theme.spaceScale.spacing08} auto;\n  }\n`;\r\nconst ModalHeader = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05}\n    ${props => props.theme.spaceScale.spacing05} 0\n    ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06}\n      ${props => props.theme.spaceScale.spacing06} 0\n      ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst ModalWrapper = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst H2 = styled(Heading) `\n  font-size: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.fontSize};\n  line-height: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.lineHeight};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0;\n  padding-right: ${props => props.theme.spaceScale.spacing10};\n  font-weight: 600;\n`;\r\nconst CloseBtn = styled.span `\n  position: absolute;\n  top: 0;\n  right: 0;\n  margin: ${props => props.theme.spaceScale.spacing02};\n`;\r\nexport const Modal = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, children, closeAriaLabel, closeButtonSize, portalContainer, containerStyle, containerTransition, isBackgroundClickDisabled, isEscKeyDownDisabled, header, isCloseButtonHidden, isOpen, unmountOnExit = true, testId, isModalClosingControlledManually, headerRef, onClose, hasDrawerAnimation, showBackgroundOverlay = true, headerLevel = 2, ...rest } = props;\r\n    const lastFocus = React.useRef();\r\n    const headingRef = React.useRef();\r\n    const bodyRef = React.useRef();\r\n    const id = useGenerateId(props.id);\r\n    const headingId = `${id}_heading`;\r\n    const contentId = `${id}_content`;\r\n    const [isModalOpen, setIsModalOpen] = React.useState(isOpen);\r\n    const [currentTarget, setCurrentTarget] = React.useState(null);\r\n    const [modalCount, setModalCount] = React.useState(0);\r\n    const focusTrapElement = useFocusLock(isModalOpen, headingRef, bodyRef);\r\n    const prevOpen = usePrevious(isOpen);\r\n    React.useEffect(() => {\r\n        if (isModalClosingControlledManually &&\r\n            prevOpen &&\r\n            !isOpen &&\r\n            isModalOpen) {\r\n            setIsModalOpen(false);\r\n        }\r\n        else if (!prevOpen && isOpen) {\r\n            setIsModalOpen(true);\r\n            if (headerRef && typeof headerRef === 'function') {\r\n                headerRef(headingRef);\r\n            }\r\n        }\r\n        else if (prevOpen && !isOpen && isModalOpen) {\r\n            handleClose();\r\n        }\r\n    }, [isOpen]);\r\n    React.useEffect(() => {\r\n        if (isModalOpen) {\r\n            lastFocus.current = document.activeElement;\r\n            const count = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            setModalCount(count);\r\n            if (!isEscKeyDownDisabled) {\r\n                document.body.addEventListener('keydown', handleEscapeKeyDown, false);\r\n            }\r\n        }\r\n        return () => {\r\n            document.body.removeEventListener('keydown', handleEscapeKeyDown, false);\r\n        };\r\n    }, [isModalOpen, isEscKeyDownDisabled]);\r\n    function handleModalClick(event) {\r\n        const contentEl = document.getElementById(contentId);\r\n        if (contentEl &&\r\n            !contentEl.contains(event.target) &&\r\n            event.target === currentTarget) {\r\n            handleClose(event);\r\n        }\r\n    }\r\n    function handleModalOnMouseDown(event) {\r\n        setCurrentTarget(event.target);\r\n    }\r\n    function handleEscapeKeyDown(event) {\r\n        if (event.key === 'Escape') {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            props.onEscKeyDown &&\r\n                typeof props.onEscKeyDown === 'function' &&\r\n                props.onEscKeyDown(event);\r\n            //Supports nested modals\r\n            const modalsInDom = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            if (modalCount <= 1 && modalsInDom !== 1) {\r\n                const modalEl = document.getElementById(id);\r\n                if (modalEl && modalEl.contains(event.target)) {\r\n                    handleClose(event);\r\n                }\r\n                else {\r\n                    headingRef.current?.focus();\r\n                }\r\n            }\r\n            else {\r\n                handleClose(event);\r\n            }\r\n        }\r\n    }\r\n    function handleClose(event) {\r\n        event?.stopPropagation();\r\n        setTimeout(() => {\r\n            if (!props.isModalClosingControlledManually) {\r\n                setIsModalOpen(false);\r\n            }\r\n            lastFocus.current?.focus();\r\n            onClose && typeof onClose === 'function' && onClose();\r\n        }, 0);\r\n    }\r\n    const isInverse = useIsInverse(props.isInverse);\r\n    const other = omit(['onEscKeyDown'], rest);\r\n    const theme = React.useContext(ThemeContext);\r\n    const i18n = React.useContext(I18nContext);\r\n    const CloseIconButton = (React.createElement(CloseIcon, { size: magma.iconSizes[closeButtonSize]\r\n            ? magma.iconSizes[closeButtonSize]\r\n            : magma.iconSizes.medium }));\r\n    return isModalOpen\r\n        ? ReactDOM.createPortal(React.createElement(\"div\", { ref: focusTrapElement },\r\n            React.createElement(Global, { styles: css `\n                html {\n                  overflow: ${isOpen ? 'hidden' : 'auto'};\n                }\n              ` }),\r\n            React.createElement(ModalContainer, Object.assign({ \"aria-labelledby\": header ? headingId : null, \"aria-label\": !header ? ariaLabel : null, \"aria-modal\": true, \"data-testid\": testId, id: id, modalCount: modalCount, onClick: isBackgroundClickDisabled ? null : handleModalClick, onMouseDown: isBackgroundClickDisabled ? null : handleModalOnMouseDown, role: \"dialog\", style: containerStyle, theme: theme, isOpen: isModalOpen }, containerTransition, { unmountOnExit: unmountOnExit, hasDrawerAnimation: hasDrawerAnimation }),\r\n                React.createElement(ModalContent, Object.assign({}, other, { \"data-testid\": \"modal-content\", id: contentId, ref: ref, showBackgroundOverlay: showBackgroundOverlay, theme: theme }),\r\n                    header && (React.createElement(ModalHeader, { theme: theme }, header && (React.createElement(H2, { id: headingId, isInverse: isInverse, level: headerLevel, ref: headingRef, visualStyle: TypographyVisualStyle.headingSmall, tabIndex: -1, theme: theme }, header)))),\r\n                    !isCloseButtonHidden && (React.createElement(CloseBtn, { theme: theme },\r\n                        React.createElement(IconButton, { \"aria-label\": closeAriaLabel\r\n                                ? closeAriaLabel\r\n                                : i18n.modal.closeAriaLabel, color: ButtonColor.primary, icon: CloseIconButton, isInverse: isInverse, onClick: handleClose, testId: \"modal-closebtn\", variant: ButtonVariant.link }))),\r\n                    React.createElement(ModalWrapper, { ref: bodyRef, theme: theme }, children))),\r\n            showBackgroundOverlay && (React.createElement(ModalBackdrop, { \"data-testid\": \"modal-backdrop\", onMouseDown: isBackgroundClickDisabled ? undefined : handleClose, fade: hasDrawerAnimation, isOpen: isModalOpen, style: modalCount >= 2 && { zIndex: '998' }, unmountOnExit: true, theme: theme }))), portalContainer ?? document.getElementsByTagName('body')[0])\r\n        : null;\r\n});\r\n//# sourceMappingURL=Modal.js.map"]} */"));
|
|
18808
18962
|
var CloseBtn = /*#__PURE__*/_styled("span", {
|
|
18809
18963
|
target: "e1pxemtw0",
|
|
18810
18964
|
label: "CloseBtn"
|
|
18811
18965
|
})("position:absolute;top:0;right:0;margin:", function (props) {
|
|
18812
18966
|
return props.theme.spaceScale.spacing02;
|
|
18813
|
-
}, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Modal.tsx"],"names":[],"mappings":"AA0G6B","file":"Modal.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css, Global } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport ReactDOM from 'react-dom';\r\nimport { CloseIcon } from 'react-magma-icons';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { I18nContext } from '../../i18n';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { magma } from '../../theme/magma';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { omit, useGenerateId, usePrevious } from '../../utils';\r\nimport { ButtonColor, ButtonVariant } from '../Button';\r\nimport { Heading } from '../Heading';\r\nimport { IconButton } from '../IconButton';\r\nimport { Transition } from '../Transition';\r\nimport { TypographyVisualStyle } from '../Typography';\r\nexport var ModalSize;\r\n(function (ModalSize) {\r\n    ModalSize[\"large\"] = \"large\";\r\n    ModalSize[\"medium\"] = \"medium\";\r\n    ModalSize[\"small\"] = \"small\";\r\n})(ModalSize || (ModalSize = {}));\r\nconst ModalContainer = styled(Transition) `\n  position: fixed;\n  bottom: 0;\n  left: 0;\n  overflow-y: auto;\n  padding: ${props => props.theme.spaceScale.spacing03};\n  right: 0;\n  top: 0;\n  z-index: ${props => (props.modalCount >= 2 ? '999' : '998')};\n`;\r\nconst ModalBackdrop = styled(Transition) `\n  backdrop-filter: blur(3px);\n  background: ${props => transparentize(0.4, props.theme.colors.neutral900)};\n  bottom: 0;\n  left: 0;\n  right: 0;\n  top: 0;\n  z-index: 997;\n  position: fixed;\n`;\r\nconst ModalContent = styled.div `\n  background: ${props => props.isInverse\r\n    ? props.theme.colors.primary600\r\n    : props.theme.colors.neutral100};\n  border: ${props => {\r\n    if (!props.showBackgroundOverlay && props.isInverse) {\r\n        return `1px solid ${transparentize(0.5, props.theme.colors.tertiary)}`;\r\n    }\r\n    return 'none';\r\n}};\n  border-radius: ${props => props.theme.borderRadius};\n  box-shadow: ${props => {\r\n    const amount = props.isInverse ? 0.82 : 0.6;\r\n    return `0 2px 6px ${transparentize(amount, props.theme.colors.neutral900)}`;\r\n}};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0 auto;\n  position: relative;\n  z-index: 1000;\n\n  max-width: ${props => {\r\n    switch (props.size) {\r\n        case 'large':\r\n            return props.theme.modal.width.large;\r\n        case 'small':\r\n            return props.theme.modal.width.small;\r\n        default:\r\n            return props.theme.modal.width.medium;\r\n    }\r\n}};\n\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    margin: ${props => props.theme.spaceScale.spacing08} auto;\n  }\n`;\r\nconst ModalHeader = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05}\n    ${props => props.theme.spaceScale.spacing05} 0\n    ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06}\n      ${props => props.theme.spaceScale.spacing06} 0\n      ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst ModalWrapper = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst H2 = styled(Heading) `\n  font-size: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.fontSize};\n  line-height: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.lineHeight};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0;\n  padding-right: ${props => props.theme.spaceScale.spacing10};\n  font-weight: 600;\n`;\r\nconst CloseBtn = styled.span `\n  position: absolute;\n  top: 0;\n  right: 0;\n  margin: ${props => props.theme.spaceScale.spacing02};\n`;\r\nexport const Modal = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, children, closeAriaLabel, closeButtonSize, portalContainer, containerStyle, containerTransition, isBackgroundClickDisabled, isEscKeyDownDisabled, header, isCloseButtonHidden, isOpen, unmountOnExit = true, testId, isModalClosingControlledManually, headerRef, onClose, hasDrawerAnimation, showBackgroundOverlay = true, headerLevel = 2, ...rest } = props;\r\n    const lastFocus = React.useRef();\r\n    const headingRef = React.useRef();\r\n    const bodyRef = React.useRef();\r\n    const id = useGenerateId(props.id);\r\n    const headingId = `${id}_heading`;\r\n    const contentId = `${id}_content`;\r\n    const [isModalOpen, setIsModalOpen] = React.useState(isOpen);\r\n    const [currentTarget, setCurrentTarget] = React.useState(null);\r\n    const [modalCount, setModalCount] = React.useState(0);\r\n    const focusTrapElement = useFocusLock(isModalOpen, headingRef, bodyRef);\r\n    const prevOpen = usePrevious(isOpen);\r\n    React.useEffect(() => {\r\n        if (isModalClosingControlledManually &&\r\n            prevOpen &&\r\n            !isOpen &&\r\n            isModalOpen) {\r\n            setIsModalOpen(false);\r\n        }\r\n        else if (!prevOpen && isOpen) {\r\n            setIsModalOpen(true);\r\n            if (headerRef && typeof headerRef === 'function') {\r\n                headerRef(headingRef);\r\n            }\r\n        }\r\n        else if (prevOpen && !isOpen && isModalOpen) {\r\n            handleClose();\r\n        }\r\n    }, [isOpen]);\r\n    React.useEffect(() => {\r\n        if (isModalOpen) {\r\n            lastFocus.current = document.activeElement;\r\n            const count = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            setModalCount(count);\r\n            if (!isEscKeyDownDisabled) {\r\n                document.body.addEventListener('keydown', handleEscapeKeyDown, false);\r\n            }\r\n        }\r\n        return () => {\r\n            document.body.removeEventListener('keydown', handleEscapeKeyDown, false);\r\n        };\r\n    }, [isModalOpen]);\r\n    function handleModalClick(event) {\r\n        if (!document.getElementById(contentId).contains(event.target) &&\r\n            event.target === currentTarget) {\r\n            handleClose(event);\r\n        }\r\n    }\r\n    function handleModalOnMouseDown(event) {\r\n        setCurrentTarget(event.target);\r\n    }\r\n    function handleEscapeKeyDown(event) {\r\n        if (event.key === 'Escape') {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            props.onEscKeyDown &&\r\n                typeof props.onEscKeyDown === 'function' &&\r\n                props.onEscKeyDown(event);\r\n            //Supports nested modals\r\n            const modalsInDom = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            if (modalCount <= 1 && modalsInDom !== 1) {\r\n                if (document.getElementById(id).contains(event.target)) {\r\n                    handleClose(event);\r\n                }\r\n                else {\r\n                    headingRef.current.focus();\r\n                }\r\n            }\r\n            else {\r\n                handleClose(event);\r\n            }\r\n        }\r\n    }\r\n    function handleClose(event) {\r\n        event?.stopPropagation();\r\n        setTimeout(() => {\r\n            if (!props.isModalClosingControlledManually) {\r\n                setIsModalOpen(false);\r\n            }\r\n            lastFocus.current?.focus();\r\n            onClose && typeof onClose === 'function' && onClose();\r\n        }, 0);\r\n    }\r\n    const isInverse = useIsInverse(props.isInverse);\r\n    const other = omit(['onEscKeyDown'], rest);\r\n    const theme = React.useContext(ThemeContext);\r\n    const i18n = React.useContext(I18nContext);\r\n    const CloseIconButton = (React.createElement(CloseIcon, { size: magma.iconSizes[closeButtonSize]\r\n            ? magma.iconSizes[closeButtonSize]\r\n            : magma.iconSizes.medium }));\r\n    return isModalOpen\r\n        ? ReactDOM.createPortal(React.createElement(\"div\", { ref: focusTrapElement },\r\n            React.createElement(Global, { styles: css `\n                html {\n                  overflow: ${isOpen ? 'hidden' : 'auto'};\n                }\n              ` }),\r\n            React.createElement(ModalContainer, Object.assign({ \"aria-labelledby\": header ? headingId : null, \"aria-label\": !header ? ariaLabel : null, \"aria-modal\": true, \"data-testid\": testId, id: id, modalCount: modalCount, onClick: isBackgroundClickDisabled ? null : handleModalClick, onMouseDown: isBackgroundClickDisabled ? null : handleModalOnMouseDown, role: \"dialog\", style: containerStyle, theme: theme, isOpen: isModalOpen }, containerTransition, { unmountOnExit: unmountOnExit, hasDrawerAnimation: hasDrawerAnimation }),\r\n                React.createElement(ModalContent, Object.assign({}, other, { \"data-testid\": \"modal-content\", id: contentId, ref: ref, showBackgroundOverlay: showBackgroundOverlay, theme: theme }),\r\n                    header && (React.createElement(ModalHeader, { theme: theme }, header && (React.createElement(H2, { id: headingId, isInverse: isInverse, level: headerLevel, ref: headingRef, visualStyle: TypographyVisualStyle.headingSmall, tabIndex: -1, theme: theme }, header)))),\r\n                    !isCloseButtonHidden && (React.createElement(CloseBtn, { theme: theme },\r\n                        React.createElement(IconButton, { \"aria-label\": closeAriaLabel\r\n                                ? closeAriaLabel\r\n                                : i18n.modal.closeAriaLabel, color: ButtonColor.primary, icon: CloseIconButton, isInverse: isInverse, onClick: handleClose, testId: \"modal-closebtn\", variant: ButtonVariant.link }))),\r\n                    React.createElement(ModalWrapper, { ref: bodyRef, theme: theme }, children))),\r\n            showBackgroundOverlay && (React.createElement(ModalBackdrop, { \"data-testid\": \"modal-backdrop\", onMouseDown: isBackgroundClickDisabled ? undefined : handleClose, fade: hasDrawerAnimation, isOpen: isModalOpen, style: modalCount >= 2 && { zIndex: '998' }, unmountOnExit: true, theme: theme }))), portalContainer ?? document.getElementsByTagName('body')[0])\r\n        : null;\r\n});\r\n//# sourceMappingURL=Modal.js.map"]} */"));
|
|
18967
|
+
}, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Modal.tsx"],"names":[],"mappings":"AA0G6B","file":"Modal.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css, Global } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport ReactDOM from 'react-dom';\r\nimport { CloseIcon } from 'react-magma-icons';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { I18nContext } from '../../i18n';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { magma } from '../../theme/magma';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { omit, useGenerateId, usePrevious } from '../../utils';\r\nimport { ButtonColor, ButtonVariant } from '../Button';\r\nimport { Heading } from '../Heading';\r\nimport { IconButton } from '../IconButton';\r\nimport { Transition } from '../Transition';\r\nimport { TypographyVisualStyle } from '../Typography';\r\nexport var ModalSize;\r\n(function (ModalSize) {\r\n    ModalSize[\"large\"] = \"large\";\r\n    ModalSize[\"medium\"] = \"medium\";\r\n    ModalSize[\"small\"] = \"small\";\r\n})(ModalSize || (ModalSize = {}));\r\nconst ModalContainer = styled(Transition) `\n  position: fixed;\n  bottom: 0;\n  left: 0;\n  overflow-y: auto;\n  padding: ${props => props.theme.spaceScale.spacing03};\n  right: 0;\n  top: 0;\n  z-index: ${props => (props.modalCount >= 2 ? '999' : '998')};\n`;\r\nconst ModalBackdrop = styled(Transition) `\n  backdrop-filter: blur(3px);\n  background: ${props => transparentize(0.4, props.theme.colors.neutral900)};\n  bottom: 0;\n  left: 0;\n  right: 0;\n  top: 0;\n  z-index: 997;\n  position: fixed;\n`;\r\nconst ModalContent = styled.div `\n  background: ${props => props.isInverse\r\n    ? props.theme.colors.primary600\r\n    : props.theme.colors.neutral100};\n  border: ${props => {\r\n    if (!props.showBackgroundOverlay && props.isInverse) {\r\n        return `1px solid ${transparentize(0.5, props.theme.colors.tertiary)}`;\r\n    }\r\n    return 'none';\r\n}};\n  border-radius: ${props => props.theme.borderRadius};\n  box-shadow: ${props => {\r\n    const amount = props.isInverse ? 0.82 : 0.6;\r\n    return `0 2px 6px ${transparentize(amount, props.theme.colors.neutral900)}`;\r\n}};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0 auto;\n  position: relative;\n  z-index: 1000;\n\n  max-width: ${props => {\r\n    switch (props.size) {\r\n        case 'large':\r\n            return props.theme.modal.width.large;\r\n        case 'small':\r\n            return props.theme.modal.width.small;\r\n        default:\r\n            return props.theme.modal.width.medium;\r\n    }\r\n}};\n\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    margin: ${props => props.theme.spaceScale.spacing08} auto;\n  }\n`;\r\nconst ModalHeader = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05}\n    ${props => props.theme.spaceScale.spacing05} 0\n    ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06}\n      ${props => props.theme.spaceScale.spacing06} 0\n      ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst ModalWrapper = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst H2 = styled(Heading) `\n  font-size: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.fontSize};\n  line-height: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.lineHeight};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0;\n  padding-right: ${props => props.theme.spaceScale.spacing10};\n  font-weight: 600;\n`;\r\nconst CloseBtn = styled.span `\n  position: absolute;\n  top: 0;\n  right: 0;\n  margin: ${props => props.theme.spaceScale.spacing02};\n`;\r\nexport const Modal = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, children, closeAriaLabel, closeButtonSize, portalContainer, containerStyle, containerTransition, isBackgroundClickDisabled, isEscKeyDownDisabled, header, isCloseButtonHidden, isOpen, unmountOnExit = true, testId, isModalClosingControlledManually, headerRef, onClose, hasDrawerAnimation, showBackgroundOverlay = true, headerLevel = 2, ...rest } = props;\r\n    const lastFocus = React.useRef();\r\n    const headingRef = React.useRef();\r\n    const bodyRef = React.useRef();\r\n    const id = useGenerateId(props.id);\r\n    const headingId = `${id}_heading`;\r\n    const contentId = `${id}_content`;\r\n    const [isModalOpen, setIsModalOpen] = React.useState(isOpen);\r\n    const [currentTarget, setCurrentTarget] = React.useState(null);\r\n    const [modalCount, setModalCount] = React.useState(0);\r\n    const focusTrapElement = useFocusLock(isModalOpen, headingRef, bodyRef);\r\n    const prevOpen = usePrevious(isOpen);\r\n    React.useEffect(() => {\r\n        if (isModalClosingControlledManually &&\r\n            prevOpen &&\r\n            !isOpen &&\r\n            isModalOpen) {\r\n            setIsModalOpen(false);\r\n        }\r\n        else if (!prevOpen && isOpen) {\r\n            setIsModalOpen(true);\r\n            if (headerRef && typeof headerRef === 'function') {\r\n                headerRef(headingRef);\r\n            }\r\n        }\r\n        else if (prevOpen && !isOpen && isModalOpen) {\r\n            handleClose();\r\n        }\r\n    }, [isOpen]);\r\n    React.useEffect(() => {\r\n        if (isModalOpen) {\r\n            lastFocus.current = document.activeElement;\r\n            const count = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            setModalCount(count);\r\n            if (!isEscKeyDownDisabled) {\r\n                document.body.addEventListener('keydown', handleEscapeKeyDown, false);\r\n            }\r\n        }\r\n        return () => {\r\n            document.body.removeEventListener('keydown', handleEscapeKeyDown, false);\r\n        };\r\n    }, [isModalOpen, isEscKeyDownDisabled]);\r\n    function handleModalClick(event) {\r\n        const contentEl = document.getElementById(contentId);\r\n        if (contentEl &&\r\n            !contentEl.contains(event.target) &&\r\n            event.target === currentTarget) {\r\n            handleClose(event);\r\n        }\r\n    }\r\n    function handleModalOnMouseDown(event) {\r\n        setCurrentTarget(event.target);\r\n    }\r\n    function handleEscapeKeyDown(event) {\r\n        if (event.key === 'Escape') {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            props.onEscKeyDown &&\r\n                typeof props.onEscKeyDown === 'function' &&\r\n                props.onEscKeyDown(event);\r\n            //Supports nested modals\r\n            const modalsInDom = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            if (modalCount <= 1 && modalsInDom !== 1) {\r\n                const modalEl = document.getElementById(id);\r\n                if (modalEl && modalEl.contains(event.target)) {\r\n                    handleClose(event);\r\n                }\r\n                else {\r\n                    headingRef.current?.focus();\r\n                }\r\n            }\r\n            else {\r\n                handleClose(event);\r\n            }\r\n        }\r\n    }\r\n    function handleClose(event) {\r\n        event?.stopPropagation();\r\n        setTimeout(() => {\r\n            if (!props.isModalClosingControlledManually) {\r\n                setIsModalOpen(false);\r\n            }\r\n            lastFocus.current?.focus();\r\n            onClose && typeof onClose === 'function' && onClose();\r\n        }, 0);\r\n    }\r\n    const isInverse = useIsInverse(props.isInverse);\r\n    const other = omit(['onEscKeyDown'], rest);\r\n    const theme = React.useContext(ThemeContext);\r\n    const i18n = React.useContext(I18nContext);\r\n    const CloseIconButton = (React.createElement(CloseIcon, { size: magma.iconSizes[closeButtonSize]\r\n            ? magma.iconSizes[closeButtonSize]\r\n            : magma.iconSizes.medium }));\r\n    return isModalOpen\r\n        ? ReactDOM.createPortal(React.createElement(\"div\", { ref: focusTrapElement },\r\n            React.createElement(Global, { styles: css `\n                html {\n                  overflow: ${isOpen ? 'hidden' : 'auto'};\n                }\n              ` }),\r\n            React.createElement(ModalContainer, Object.assign({ \"aria-labelledby\": header ? headingId : null, \"aria-label\": !header ? ariaLabel : null, \"aria-modal\": true, \"data-testid\": testId, id: id, modalCount: modalCount, onClick: isBackgroundClickDisabled ? null : handleModalClick, onMouseDown: isBackgroundClickDisabled ? null : handleModalOnMouseDown, role: \"dialog\", style: containerStyle, theme: theme, isOpen: isModalOpen }, containerTransition, { unmountOnExit: unmountOnExit, hasDrawerAnimation: hasDrawerAnimation }),\r\n                React.createElement(ModalContent, Object.assign({}, other, { \"data-testid\": \"modal-content\", id: contentId, ref: ref, showBackgroundOverlay: showBackgroundOverlay, theme: theme }),\r\n                    header && (React.createElement(ModalHeader, { theme: theme }, header && (React.createElement(H2, { id: headingId, isInverse: isInverse, level: headerLevel, ref: headingRef, visualStyle: TypographyVisualStyle.headingSmall, tabIndex: -1, theme: theme }, header)))),\r\n                    !isCloseButtonHidden && (React.createElement(CloseBtn, { theme: theme },\r\n                        React.createElement(IconButton, { \"aria-label\": closeAriaLabel\r\n                                ? closeAriaLabel\r\n                                : i18n.modal.closeAriaLabel, color: ButtonColor.primary, icon: CloseIconButton, isInverse: isInverse, onClick: handleClose, testId: \"modal-closebtn\", variant: ButtonVariant.link }))),\r\n                    React.createElement(ModalWrapper, { ref: bodyRef, theme: theme }, children))),\r\n            showBackgroundOverlay && (React.createElement(ModalBackdrop, { \"data-testid\": \"modal-backdrop\", onMouseDown: isBackgroundClickDisabled ? undefined : handleClose, fade: hasDrawerAnimation, isOpen: isModalOpen, style: modalCount >= 2 && { zIndex: '998' }, unmountOnExit: true, theme: theme }))), portalContainer ?? document.getElementsByTagName('body')[0])\r\n        : null;\r\n});\r\n//# sourceMappingURL=Modal.js.map"]} */"));
|
|
18814
18968
|
var Modal = /*#__PURE__*/forwardRef(function (props, ref) {
|
|
18815
18969
|
var ariaLabel = props.ariaLabel,
|
|
18816
18970
|
children = props.children,
|
|
@@ -18877,9 +19031,10 @@ var Modal = /*#__PURE__*/forwardRef(function (props, ref) {
|
|
|
18877
19031
|
return function () {
|
|
18878
19032
|
document.body.removeEventListener('keydown', handleEscapeKeyDown, false);
|
|
18879
19033
|
};
|
|
18880
|
-
}, [isModalOpen]);
|
|
19034
|
+
}, [isModalOpen, isEscKeyDownDisabled]);
|
|
18881
19035
|
function handleModalClick(event) {
|
|
18882
|
-
|
|
19036
|
+
var contentEl = document.getElementById(contentId);
|
|
19037
|
+
if (contentEl && !contentEl.contains(event.target) && event.target === currentTarget) {
|
|
18883
19038
|
handleClose(event);
|
|
18884
19039
|
}
|
|
18885
19040
|
}
|
|
@@ -18894,10 +19049,12 @@ var Modal = /*#__PURE__*/forwardRef(function (props, ref) {
|
|
|
18894
19049
|
//Supports nested modals
|
|
18895
19050
|
var modalsInDom = document.querySelectorAll('[aria-modal="true"]').length;
|
|
18896
19051
|
if (modalCount <= 1 && modalsInDom !== 1) {
|
|
18897
|
-
|
|
19052
|
+
var modalEl = document.getElementById(id);
|
|
19053
|
+
if (modalEl && modalEl.contains(event.target)) {
|
|
18898
19054
|
handleClose(event);
|
|
18899
19055
|
} else {
|
|
18900
|
-
|
|
19056
|
+
var _headingRef$current;
|
|
19057
|
+
(_headingRef$current = headingRef.current) == null || _headingRef$current.focus();
|
|
18901
19058
|
}
|
|
18902
19059
|
} else {
|
|
18903
19060
|
handleClose(event);
|
|
@@ -18925,7 +19082,7 @@ var Modal = /*#__PURE__*/forwardRef(function (props, ref) {
|
|
|
18925
19082
|
return isModalOpen ? ReactDOM__default.createPortal(createElement("div", {
|
|
18926
19083
|
ref: focusTrapElement
|
|
18927
19084
|
}, createElement(Global, {
|
|
18928
|
-
styles: /*#__PURE__*/css("html{overflow:", isOpen ? 'hidden' : 'auto', ";};label:styles;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Modal.tsx"],"names":[],"mappings":"AA6MsD","file":"Modal.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css, Global } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport ReactDOM from 'react-dom';\r\nimport { CloseIcon } from 'react-magma-icons';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { I18nContext } from '../../i18n';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { magma } from '../../theme/magma';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { omit, useGenerateId, usePrevious } from '../../utils';\r\nimport { ButtonColor, ButtonVariant } from '../Button';\r\nimport { Heading } from '../Heading';\r\nimport { IconButton } from '../IconButton';\r\nimport { Transition } from '../Transition';\r\nimport { TypographyVisualStyle } from '../Typography';\r\nexport var ModalSize;\r\n(function (ModalSize) {\r\n    ModalSize[\"large\"] = \"large\";\r\n    ModalSize[\"medium\"] = \"medium\";\r\n    ModalSize[\"small\"] = \"small\";\r\n})(ModalSize || (ModalSize = {}));\r\nconst ModalContainer = styled(Transition) `\n  position: fixed;\n  bottom: 0;\n  left: 0;\n  overflow-y: auto;\n  padding: ${props => props.theme.spaceScale.spacing03};\n  right: 0;\n  top: 0;\n  z-index: ${props => (props.modalCount >= 2 ? '999' : '998')};\n`;\r\nconst ModalBackdrop = styled(Transition) `\n  backdrop-filter: blur(3px);\n  background: ${props => transparentize(0.4, props.theme.colors.neutral900)};\n  bottom: 0;\n  left: 0;\n  right: 0;\n  top: 0;\n  z-index: 997;\n  position: fixed;\n`;\r\nconst ModalContent = styled.div `\n  background: ${props => props.isInverse\r\n    ? props.theme.colors.primary600\r\n    : props.theme.colors.neutral100};\n  border: ${props => {\r\n    if (!props.showBackgroundOverlay && props.isInverse) {\r\n        return `1px solid ${transparentize(0.5, props.theme.colors.tertiary)}`;\r\n    }\r\n    return 'none';\r\n}};\n  border-radius: ${props => props.theme.borderRadius};\n  box-shadow: ${props => {\r\n    const amount = props.isInverse ? 0.82 : 0.6;\r\n    return `0 2px 6px ${transparentize(amount, props.theme.colors.neutral900)}`;\r\n}};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0 auto;\n  position: relative;\n  z-index: 1000;\n\n  max-width: ${props => {\r\n    switch (props.size) {\r\n        case 'large':\r\n            return props.theme.modal.width.large;\r\n        case 'small':\r\n            return props.theme.modal.width.small;\r\n        default:\r\n            return props.theme.modal.width.medium;\r\n    }\r\n}};\n\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    margin: ${props => props.theme.spaceScale.spacing08} auto;\n  }\n`;\r\nconst ModalHeader = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05}\n    ${props => props.theme.spaceScale.spacing05} 0\n    ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06}\n      ${props => props.theme.spaceScale.spacing06} 0\n      ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst ModalWrapper = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst H2 = styled(Heading) `\n  font-size: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.fontSize};\n  line-height: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.lineHeight};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0;\n  padding-right: ${props => props.theme.spaceScale.spacing10};\n  font-weight: 600;\n`;\r\nconst CloseBtn = styled.span `\n  position: absolute;\n  top: 0;\n  right: 0;\n  margin: ${props => props.theme.spaceScale.spacing02};\n`;\r\nexport const Modal = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, children, closeAriaLabel, closeButtonSize, portalContainer, containerStyle, containerTransition, isBackgroundClickDisabled, isEscKeyDownDisabled, header, isCloseButtonHidden, isOpen, unmountOnExit = true, testId, isModalClosingControlledManually, headerRef, onClose, hasDrawerAnimation, showBackgroundOverlay = true, headerLevel = 2, ...rest } = props;\r\n    const lastFocus = React.useRef();\r\n    const headingRef = React.useRef();\r\n    const bodyRef = React.useRef();\r\n    const id = useGenerateId(props.id);\r\n    const headingId = `${id}_heading`;\r\n    const contentId = `${id}_content`;\r\n    const [isModalOpen, setIsModalOpen] = React.useState(isOpen);\r\n    const [currentTarget, setCurrentTarget] = React.useState(null);\r\n    const [modalCount, setModalCount] = React.useState(0);\r\n    const focusTrapElement = useFocusLock(isModalOpen, headingRef, bodyRef);\r\n    const prevOpen = usePrevious(isOpen);\r\n    React.useEffect(() => {\r\n        if (isModalClosingControlledManually &&\r\n            prevOpen &&\r\n            !isOpen &&\r\n            isModalOpen) {\r\n            setIsModalOpen(false);\r\n        }\r\n        else if (!prevOpen && isOpen) {\r\n            setIsModalOpen(true);\r\n            if (headerRef && typeof headerRef === 'function') {\r\n                headerRef(headingRef);\r\n            }\r\n        }\r\n        else if (prevOpen && !isOpen && isModalOpen) {\r\n            handleClose();\r\n        }\r\n    }, [isOpen]);\r\n    React.useEffect(() => {\r\n        if (isModalOpen) {\r\n            lastFocus.current = document.activeElement;\r\n            const count = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            setModalCount(count);\r\n            if (!isEscKeyDownDisabled) {\r\n                document.body.addEventListener('keydown', handleEscapeKeyDown, false);\r\n            }\r\n        }\r\n        return () => {\r\n            document.body.removeEventListener('keydown', handleEscapeKeyDown, false);\r\n        };\r\n    }, [isModalOpen]);\r\n    function handleModalClick(event) {\r\n        if (!document.getElementById(contentId).contains(event.target) &&\r\n            event.target === currentTarget) {\r\n            handleClose(event);\r\n        }\r\n    }\r\n    function handleModalOnMouseDown(event) {\r\n        setCurrentTarget(event.target);\r\n    }\r\n    function handleEscapeKeyDown(event) {\r\n        if (event.key === 'Escape') {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            props.onEscKeyDown &&\r\n                typeof props.onEscKeyDown === 'function' &&\r\n                props.onEscKeyDown(event);\r\n            //Supports nested modals\r\n            const modalsInDom = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            if (modalCount <= 1 && modalsInDom !== 1) {\r\n                if (document.getElementById(id).contains(event.target)) {\r\n                    handleClose(event);\r\n                }\r\n                else {\r\n                    headingRef.current.focus();\r\n                }\r\n            }\r\n            else {\r\n                handleClose(event);\r\n            }\r\n        }\r\n    }\r\n    function handleClose(event) {\r\n        event?.stopPropagation();\r\n        setTimeout(() => {\r\n            if (!props.isModalClosingControlledManually) {\r\n                setIsModalOpen(false);\r\n            }\r\n            lastFocus.current?.focus();\r\n            onClose && typeof onClose === 'function' && onClose();\r\n        }, 0);\r\n    }\r\n    const isInverse = useIsInverse(props.isInverse);\r\n    const other = omit(['onEscKeyDown'], rest);\r\n    const theme = React.useContext(ThemeContext);\r\n    const i18n = React.useContext(I18nContext);\r\n    const CloseIconButton = (React.createElement(CloseIcon, { size: magma.iconSizes[closeButtonSize]\r\n            ? magma.iconSizes[closeButtonSize]\r\n            : magma.iconSizes.medium }));\r\n    return isModalOpen\r\n        ? ReactDOM.createPortal(React.createElement(\"div\", { ref: focusTrapElement },\r\n            React.createElement(Global, { styles: css `\n                html {\n                  overflow: ${isOpen ? 'hidden' : 'auto'};\n                }\n              ` }),\r\n            React.createElement(ModalContainer, Object.assign({ \"aria-labelledby\": header ? headingId : null, \"aria-label\": !header ? ariaLabel : null, \"aria-modal\": true, \"data-testid\": testId, id: id, modalCount: modalCount, onClick: isBackgroundClickDisabled ? null : handleModalClick, onMouseDown: isBackgroundClickDisabled ? null : handleModalOnMouseDown, role: \"dialog\", style: containerStyle, theme: theme, isOpen: isModalOpen }, containerTransition, { unmountOnExit: unmountOnExit, hasDrawerAnimation: hasDrawerAnimation }),\r\n                React.createElement(ModalContent, Object.assign({}, other, { \"data-testid\": \"modal-content\", id: contentId, ref: ref, showBackgroundOverlay: showBackgroundOverlay, theme: theme }),\r\n                    header && (React.createElement(ModalHeader, { theme: theme }, header && (React.createElement(H2, { id: headingId, isInverse: isInverse, level: headerLevel, ref: headingRef, visualStyle: TypographyVisualStyle.headingSmall, tabIndex: -1, theme: theme }, header)))),\r\n                    !isCloseButtonHidden && (React.createElement(CloseBtn, { theme: theme },\r\n                        React.createElement(IconButton, { \"aria-label\": closeAriaLabel\r\n                                ? closeAriaLabel\r\n                                : i18n.modal.closeAriaLabel, color: ButtonColor.primary, icon: CloseIconButton, isInverse: isInverse, onClick: handleClose, testId: \"modal-closebtn\", variant: ButtonVariant.link }))),\r\n                    React.createElement(ModalWrapper, { ref: bodyRef, theme: theme }, children))),\r\n            showBackgroundOverlay && (React.createElement(ModalBackdrop, { \"data-testid\": \"modal-backdrop\", onMouseDown: isBackgroundClickDisabled ? undefined : handleClose, fade: hasDrawerAnimation, isOpen: isModalOpen, style: modalCount >= 2 && { zIndex: '998' }, unmountOnExit: true, theme: theme }))), portalContainer ?? document.getElementsByTagName('body')[0])\r\n        : null;\r\n});\r\n//# sourceMappingURL=Modal.js.map"]} */"))
|
|
19085
|
+
styles: /*#__PURE__*/css("html{overflow:", isOpen ? 'hidden' : 'auto', ";};label:styles;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Modal.tsx"],"names":[],"mappings":"AAgNsD","file":"Modal.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css, Global } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport ReactDOM from 'react-dom';\r\nimport { CloseIcon } from 'react-magma-icons';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { I18nContext } from '../../i18n';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { magma } from '../../theme/magma';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { omit, useGenerateId, usePrevious } from '../../utils';\r\nimport { ButtonColor, ButtonVariant } from '../Button';\r\nimport { Heading } from '../Heading';\r\nimport { IconButton } from '../IconButton';\r\nimport { Transition } from '../Transition';\r\nimport { TypographyVisualStyle } from '../Typography';\r\nexport var ModalSize;\r\n(function (ModalSize) {\r\n    ModalSize[\"large\"] = \"large\";\r\n    ModalSize[\"medium\"] = \"medium\";\r\n    ModalSize[\"small\"] = \"small\";\r\n})(ModalSize || (ModalSize = {}));\r\nconst ModalContainer = styled(Transition) `\n  position: fixed;\n  bottom: 0;\n  left: 0;\n  overflow-y: auto;\n  padding: ${props => props.theme.spaceScale.spacing03};\n  right: 0;\n  top: 0;\n  z-index: ${props => (props.modalCount >= 2 ? '999' : '998')};\n`;\r\nconst ModalBackdrop = styled(Transition) `\n  backdrop-filter: blur(3px);\n  background: ${props => transparentize(0.4, props.theme.colors.neutral900)};\n  bottom: 0;\n  left: 0;\n  right: 0;\n  top: 0;\n  z-index: 997;\n  position: fixed;\n`;\r\nconst ModalContent = styled.div `\n  background: ${props => props.isInverse\r\n    ? props.theme.colors.primary600\r\n    : props.theme.colors.neutral100};\n  border: ${props => {\r\n    if (!props.showBackgroundOverlay && props.isInverse) {\r\n        return `1px solid ${transparentize(0.5, props.theme.colors.tertiary)}`;\r\n    }\r\n    return 'none';\r\n}};\n  border-radius: ${props => props.theme.borderRadius};\n  box-shadow: ${props => {\r\n    const amount = props.isInverse ? 0.82 : 0.6;\r\n    return `0 2px 6px ${transparentize(amount, props.theme.colors.neutral900)}`;\r\n}};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0 auto;\n  position: relative;\n  z-index: 1000;\n\n  max-width: ${props => {\r\n    switch (props.size) {\r\n        case 'large':\r\n            return props.theme.modal.width.large;\r\n        case 'small':\r\n            return props.theme.modal.width.small;\r\n        default:\r\n            return props.theme.modal.width.medium;\r\n    }\r\n}};\n\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    margin: ${props => props.theme.spaceScale.spacing08} auto;\n  }\n`;\r\nconst ModalHeader = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05}\n    ${props => props.theme.spaceScale.spacing05} 0\n    ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06}\n      ${props => props.theme.spaceScale.spacing06} 0\n      ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst ModalWrapper = styled.div `\n  padding: ${props => props.theme.spaceScale.spacing05};\n  @media (min-width: ${props => props.theme.breakpoints.small}px) {\n    padding: ${props => props.theme.spaceScale.spacing06};\n  }\n`;\r\nconst H2 = styled(Heading) `\n  font-size: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.fontSize};\n  line-height: ${props => props.theme.typographyVisualStyles.headingSmall.desktop.lineHeight};\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  margin: 0;\n  padding-right: ${props => props.theme.spaceScale.spacing10};\n  font-weight: 600;\n`;\r\nconst CloseBtn = styled.span `\n  position: absolute;\n  top: 0;\n  right: 0;\n  margin: ${props => props.theme.spaceScale.spacing02};\n`;\r\nexport const Modal = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, children, closeAriaLabel, closeButtonSize, portalContainer, containerStyle, containerTransition, isBackgroundClickDisabled, isEscKeyDownDisabled, header, isCloseButtonHidden, isOpen, unmountOnExit = true, testId, isModalClosingControlledManually, headerRef, onClose, hasDrawerAnimation, showBackgroundOverlay = true, headerLevel = 2, ...rest } = props;\r\n    const lastFocus = React.useRef();\r\n    const headingRef = React.useRef();\r\n    const bodyRef = React.useRef();\r\n    const id = useGenerateId(props.id);\r\n    const headingId = `${id}_heading`;\r\n    const contentId = `${id}_content`;\r\n    const [isModalOpen, setIsModalOpen] = React.useState(isOpen);\r\n    const [currentTarget, setCurrentTarget] = React.useState(null);\r\n    const [modalCount, setModalCount] = React.useState(0);\r\n    const focusTrapElement = useFocusLock(isModalOpen, headingRef, bodyRef);\r\n    const prevOpen = usePrevious(isOpen);\r\n    React.useEffect(() => {\r\n        if (isModalClosingControlledManually &&\r\n            prevOpen &&\r\n            !isOpen &&\r\n            isModalOpen) {\r\n            setIsModalOpen(false);\r\n        }\r\n        else if (!prevOpen && isOpen) {\r\n            setIsModalOpen(true);\r\n            if (headerRef && typeof headerRef === 'function') {\r\n                headerRef(headingRef);\r\n            }\r\n        }\r\n        else if (prevOpen && !isOpen && isModalOpen) {\r\n            handleClose();\r\n        }\r\n    }, [isOpen]);\r\n    React.useEffect(() => {\r\n        if (isModalOpen) {\r\n            lastFocus.current = document.activeElement;\r\n            const count = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            setModalCount(count);\r\n            if (!isEscKeyDownDisabled) {\r\n                document.body.addEventListener('keydown', handleEscapeKeyDown, false);\r\n            }\r\n        }\r\n        return () => {\r\n            document.body.removeEventListener('keydown', handleEscapeKeyDown, false);\r\n        };\r\n    }, [isModalOpen, isEscKeyDownDisabled]);\r\n    function handleModalClick(event) {\r\n        const contentEl = document.getElementById(contentId);\r\n        if (contentEl &&\r\n            !contentEl.contains(event.target) &&\r\n            event.target === currentTarget) {\r\n            handleClose(event);\r\n        }\r\n    }\r\n    function handleModalOnMouseDown(event) {\r\n        setCurrentTarget(event.target);\r\n    }\r\n    function handleEscapeKeyDown(event) {\r\n        if (event.key === 'Escape') {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            props.onEscKeyDown &&\r\n                typeof props.onEscKeyDown === 'function' &&\r\n                props.onEscKeyDown(event);\r\n            //Supports nested modals\r\n            const modalsInDom = document.querySelectorAll('[aria-modal=\"true\"]').length;\r\n            if (modalCount <= 1 && modalsInDom !== 1) {\r\n                const modalEl = document.getElementById(id);\r\n                if (modalEl && modalEl.contains(event.target)) {\r\n                    handleClose(event);\r\n                }\r\n                else {\r\n                    headingRef.current?.focus();\r\n                }\r\n            }\r\n            else {\r\n                handleClose(event);\r\n            }\r\n        }\r\n    }\r\n    function handleClose(event) {\r\n        event?.stopPropagation();\r\n        setTimeout(() => {\r\n            if (!props.isModalClosingControlledManually) {\r\n                setIsModalOpen(false);\r\n            }\r\n            lastFocus.current?.focus();\r\n            onClose && typeof onClose === 'function' && onClose();\r\n        }, 0);\r\n    }\r\n    const isInverse = useIsInverse(props.isInverse);\r\n    const other = omit(['onEscKeyDown'], rest);\r\n    const theme = React.useContext(ThemeContext);\r\n    const i18n = React.useContext(I18nContext);\r\n    const CloseIconButton = (React.createElement(CloseIcon, { size: magma.iconSizes[closeButtonSize]\r\n            ? magma.iconSizes[closeButtonSize]\r\n            : magma.iconSizes.medium }));\r\n    return isModalOpen\r\n        ? ReactDOM.createPortal(React.createElement(\"div\", { ref: focusTrapElement },\r\n            React.createElement(Global, { styles: css `\n                html {\n                  overflow: ${isOpen ? 'hidden' : 'auto'};\n                }\n              ` }),\r\n            React.createElement(ModalContainer, Object.assign({ \"aria-labelledby\": header ? headingId : null, \"aria-label\": !header ? ariaLabel : null, \"aria-modal\": true, \"data-testid\": testId, id: id, modalCount: modalCount, onClick: isBackgroundClickDisabled ? null : handleModalClick, onMouseDown: isBackgroundClickDisabled ? null : handleModalOnMouseDown, role: \"dialog\", style: containerStyle, theme: theme, isOpen: isModalOpen }, containerTransition, { unmountOnExit: unmountOnExit, hasDrawerAnimation: hasDrawerAnimation }),\r\n                React.createElement(ModalContent, Object.assign({}, other, { \"data-testid\": \"modal-content\", id: contentId, ref: ref, showBackgroundOverlay: showBackgroundOverlay, theme: theme }),\r\n                    header && (React.createElement(ModalHeader, { theme: theme }, header && (React.createElement(H2, { id: headingId, isInverse: isInverse, level: headerLevel, ref: headingRef, visualStyle: TypographyVisualStyle.headingSmall, tabIndex: -1, theme: theme }, header)))),\r\n                    !isCloseButtonHidden && (React.createElement(CloseBtn, { theme: theme },\r\n                        React.createElement(IconButton, { \"aria-label\": closeAriaLabel\r\n                                ? closeAriaLabel\r\n                                : i18n.modal.closeAriaLabel, color: ButtonColor.primary, icon: CloseIconButton, isInverse: isInverse, onClick: handleClose, testId: \"modal-closebtn\", variant: ButtonVariant.link }))),\r\n                    React.createElement(ModalWrapper, { ref: bodyRef, theme: theme }, children))),\r\n            showBackgroundOverlay && (React.createElement(ModalBackdrop, { \"data-testid\": \"modal-backdrop\", onMouseDown: isBackgroundClickDisabled ? undefined : handleClose, fade: hasDrawerAnimation, isOpen: isModalOpen, style: modalCount >= 2 && { zIndex: '998' }, unmountOnExit: true, theme: theme }))), portalContainer ?? document.getElementsByTagName('body')[0])\r\n        : null;\r\n});\r\n//# sourceMappingURL=Modal.js.map"]} */"))
|
|
18929
19086
|
}), createElement(ModalContainer, Object.assign({
|
|
18930
19087
|
"aria-labelledby": header ? headingId : null,
|
|
18931
19088
|
"aria-label": !header ? ariaLabel : null,
|