@carbon/ibm-products-web-components 0.1.0 → 0.1.1-canary.3605
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/.storybook/main.ts +15 -1
- package/.storybook/preview-head.html +5 -1
- package/.storybook/{theme.ts → theme.js} +2 -2
- package/es/components/side-panel/side-panel.d.ts +61 -80
- package/es/components/side-panel/side-panel.js +5 -3
- package/es/components/side-panel/side-panel.js.map +1 -1
- package/es/components/side-panel/side-panel.scss.js +1 -1
- package/es/components/tearsheet/defs.d.ts +26 -0
- package/es/components/tearsheet/defs.js +39 -0
- package/es/components/tearsheet/defs.js.map +1 -0
- package/es/components/tearsheet/index.d.ts +9 -0
- package/es/components/tearsheet/index.js +9 -0
- package/es/components/tearsheet/index.js.map +1 -0
- package/es/components/tearsheet/tearsheet.d.ts +490 -0
- package/es/components/tearsheet/tearsheet.js +685 -0
- package/es/components/tearsheet/tearsheet.js.map +1 -0
- package/es/components/tearsheet/tearsheet.scss.js +13 -0
- package/es/components/tearsheet/tearsheet.scss.js.map +1 -0
- package/es/components/tearsheet/tearsheet.test.d.ts +7 -0
- package/es/components/tearsheet/tearsheet.test.js +122 -0
- package/es/components/tearsheet/tearsheet.test.js.map +1 -0
- package/es/index.d.ts +1 -0
- package/es/index.js +1 -0
- package/es/index.js.map +1 -1
- package/lib/components/side-panel/side-panel.d.ts +61 -80
- package/lib/components/tearsheet/defs.d.ts +26 -0
- package/lib/components/tearsheet/defs.js +39 -0
- package/lib/components/tearsheet/defs.js.map +1 -0
- package/lib/components/tearsheet/index.d.ts +9 -0
- package/lib/components/tearsheet/tearsheet.d.ts +490 -0
- package/lib/components/tearsheet/tearsheet.test.d.ts +7 -0
- package/lib/index.d.ts +1 -0
- package/package.json +17 -17
- package/scss/components/tearsheet/story-styles.scss +23 -0
- package/scss/components/tearsheet/tearsheet.scss +318 -0
- package/src/components/side-panel/side-panel.ts +5 -5
- package/src/components/tearsheet/defs.ts +30 -0
- package/src/components/tearsheet/index.ts +10 -0
- package/src/components/tearsheet/story-styles.scss +23 -0
- package/src/components/tearsheet/tearsheet.mdx +101 -0
- package/src/components/tearsheet/tearsheet.scss +318 -0
- package/src/components/tearsheet/tearsheet.stories.ts +703 -0
- package/src/components/tearsheet/tearsheet.test.ts +118 -0
- package/src/components/tearsheet/tearsheet.ts +792 -0
- package/src/index.ts +1 -0
- package/netlify.toml +0 -8
- /package/.storybook/{Preview.ts → preview.ts} +0 -0
@@ -0,0 +1,792 @@
|
|
1
|
+
/**
|
2
|
+
* @license
|
3
|
+
*
|
4
|
+
* Copyright IBM Corp. 2023, 2024
|
5
|
+
*
|
6
|
+
* This source code is licensed under the Apache-2.0 license found in the
|
7
|
+
* LICENSE file in the root directory of this source tree.
|
8
|
+
*/
|
9
|
+
|
10
|
+
import { LitElement, html } from 'lit';
|
11
|
+
import {
|
12
|
+
property,
|
13
|
+
query,
|
14
|
+
queryAssignedElements,
|
15
|
+
state,
|
16
|
+
} from 'lit/decorators.js';
|
17
|
+
import { prefix, carbonPrefix } from '../../globals/settings';
|
18
|
+
import HostListener from '@carbon/web-components/es/globals/decorators/host-listener.js';
|
19
|
+
import HostListenerMixin from '@carbon/web-components/es/globals/mixins/host-listener.js';
|
20
|
+
import styles from './tearsheet.scss?lit';
|
21
|
+
import { selectorTabbable } from '../../globals/settings';
|
22
|
+
import { carbonElement as customElement } from '@carbon/web-components/es/globals/decorators/carbon-element.js';
|
23
|
+
import '@carbon/web-components/es/components/button/index.js';
|
24
|
+
import '@carbon/web-components/es/components/layer/index.js';
|
25
|
+
import '@carbon/web-components/es/components/button/button-set-base.js';
|
26
|
+
import '@carbon/web-components/es/components/modal/index.js';
|
27
|
+
import {
|
28
|
+
TEARSHEET_INFLUENCER_PLACEMENT,
|
29
|
+
TEARSHEET_INFLUENCER_WIDTH,
|
30
|
+
TEARSHEET_WIDTH,
|
31
|
+
} from './defs';
|
32
|
+
|
33
|
+
export {
|
34
|
+
TEARSHEET_INFLUENCER_PLACEMENT,
|
35
|
+
TEARSHEET_INFLUENCER_WIDTH,
|
36
|
+
TEARSHEET_WIDTH,
|
37
|
+
};
|
38
|
+
|
39
|
+
const maxStackDepth = 3;
|
40
|
+
type StackHandler = (newDepth: number, newPosition: number) => void;
|
41
|
+
interface StackState {
|
42
|
+
open: StackHandler[];
|
43
|
+
all: StackHandler[];
|
44
|
+
}
|
45
|
+
|
46
|
+
// eslint-disable-next-line no-bitwise
|
47
|
+
const PRECEDING =
|
48
|
+
Node.DOCUMENT_POSITION_PRECEDING | Node.DOCUMENT_POSITION_CONTAINS;
|
49
|
+
// eslint-disable-next-line no-bitwise
|
50
|
+
const FOLLOWING =
|
51
|
+
Node.DOCUMENT_POSITION_FOLLOWING | Node.DOCUMENT_POSITION_CONTAINED_BY;
|
52
|
+
|
53
|
+
const blockClass = `${prefix}--tearsheet`;
|
54
|
+
const blockClassModalHeader = `${carbonPrefix}--modal-header`;
|
55
|
+
const blockClassActionSet = `${prefix}--action-set`;
|
56
|
+
|
57
|
+
/**
|
58
|
+
* Tries to focus on the given elements and bails out if one of them is successful.
|
59
|
+
*
|
60
|
+
* @param elements The elements.
|
61
|
+
* @param reverse `true` to go through the list in reverse order.
|
62
|
+
* @returns `true` if one of the attempts is successful, `false` otherwise.
|
63
|
+
*/
|
64
|
+
function tryFocusElements(elements: NodeListOf<HTMLElement>, reverse: boolean) {
|
65
|
+
if (!reverse) {
|
66
|
+
for (let i = 0; i < elements.length; ++i) {
|
67
|
+
const elem = elements[i];
|
68
|
+
elem.focus();
|
69
|
+
if (elem.ownerDocument!.activeElement === elem) {
|
70
|
+
return true;
|
71
|
+
}
|
72
|
+
}
|
73
|
+
} else {
|
74
|
+
for (let i = elements.length - 1; i >= 0; --i) {
|
75
|
+
const elem = elements[i];
|
76
|
+
elem.focus();
|
77
|
+
if (elem.ownerDocument!.activeElement === elem) {
|
78
|
+
return true;
|
79
|
+
}
|
80
|
+
}
|
81
|
+
}
|
82
|
+
return false;
|
83
|
+
}
|
84
|
+
|
85
|
+
/**
|
86
|
+
* Tearsheet.
|
87
|
+
*
|
88
|
+
* @element c4p-tearsheet
|
89
|
+
* @csspart dialog The dialog.
|
90
|
+
* @fires c4p-tearsheet-beingclosed
|
91
|
+
* The custom event fired before this tearsheet is being closed upon a user gesture.
|
92
|
+
* Cancellation of this event stops the user-initiated action of closing this tearsheet.
|
93
|
+
* @fires c4p-tearsheet-closed - The custom event fired after this tearsheet is closed upon a user gesture.
|
94
|
+
*/
|
95
|
+
@customElement(`${prefix}-tearsheet`)
|
96
|
+
class CDSTearsheet extends HostListenerMixin(LitElement) {
|
97
|
+
/**
|
98
|
+
* The element that had focus before this tearsheet gets open.
|
99
|
+
*/
|
100
|
+
private _launcher: Element | null = null;
|
101
|
+
|
102
|
+
/**
|
103
|
+
* Node to track focus going outside of tearsheet content.
|
104
|
+
*/
|
105
|
+
@query('#start-sentinel')
|
106
|
+
private _startSentinelNode!: HTMLAnchorElement;
|
107
|
+
|
108
|
+
/**
|
109
|
+
* Node to track focus going outside of tearsheet content.
|
110
|
+
*/
|
111
|
+
@query('#end-sentinel')
|
112
|
+
private _endSentinelNode!: HTMLAnchorElement;
|
113
|
+
|
114
|
+
/**
|
115
|
+
* Node to track tearsheet.
|
116
|
+
*/
|
117
|
+
@query(`.${blockClass}__container`)
|
118
|
+
private _tearsheet!: HTMLDivElement;
|
119
|
+
|
120
|
+
@queryAssignedElements({
|
121
|
+
slot: 'actions',
|
122
|
+
selector: `${carbonPrefix}-button`,
|
123
|
+
})
|
124
|
+
private _actions!: Array<HTMLElement>;
|
125
|
+
|
126
|
+
@state()
|
127
|
+
_actionsCount = 0;
|
128
|
+
|
129
|
+
@state()
|
130
|
+
_hasHeaderActions = false;
|
131
|
+
|
132
|
+
@state()
|
133
|
+
_hasLabel = false;
|
134
|
+
|
135
|
+
@state()
|
136
|
+
_hasSlug = false;
|
137
|
+
|
138
|
+
@state()
|
139
|
+
_hasTitle = false;
|
140
|
+
|
141
|
+
@state()
|
142
|
+
_hasDescription = false;
|
143
|
+
|
144
|
+
@state()
|
145
|
+
_hasInfluencerLeft = false;
|
146
|
+
|
147
|
+
@state()
|
148
|
+
_hasInfluencerRight = false;
|
149
|
+
|
150
|
+
@state()
|
151
|
+
_isOpen = false;
|
152
|
+
|
153
|
+
@state()
|
154
|
+
_hasHeaderNavigation = false;
|
155
|
+
|
156
|
+
/**
|
157
|
+
* Handles `click` event on this element.
|
158
|
+
*
|
159
|
+
* @param event The event.
|
160
|
+
*/
|
161
|
+
@HostListener('click')
|
162
|
+
// @ts-ignore: The decorator refers to this method but TS thinks this method is not referred to
|
163
|
+
private _handleClick = (event: MouseEvent) => {
|
164
|
+
if (
|
165
|
+
event.composedPath().indexOf(this.shadowRoot!) < 0 &&
|
166
|
+
!this.preventCloseOnClickOutside
|
167
|
+
) {
|
168
|
+
this._handleUserInitiatedClose(event.target);
|
169
|
+
}
|
170
|
+
};
|
171
|
+
|
172
|
+
/**
|
173
|
+
* Handles `blur` event on this element.
|
174
|
+
*
|
175
|
+
* @param event The event.
|
176
|
+
* @param event.target The event target.
|
177
|
+
* @param event.relatedTarget The event relatedTarget.
|
178
|
+
*/
|
179
|
+
@HostListener('shadowRoot:focusout')
|
180
|
+
// @ts-ignore: The decorator refers to this method but TS thinks this method is not referred to
|
181
|
+
private _handleBlur = async ({ target, relatedTarget }: FocusEvent) => {
|
182
|
+
if (!this._topOfStack()) {
|
183
|
+
return;
|
184
|
+
}
|
185
|
+
|
186
|
+
const {
|
187
|
+
// condensedActions,
|
188
|
+
open,
|
189
|
+
_startSentinelNode: startSentinelNode,
|
190
|
+
_endSentinelNode: endSentinelNode,
|
191
|
+
} = this;
|
192
|
+
|
193
|
+
const oldContains = target !== this && this.contains(target as Node);
|
194
|
+
const currentContains =
|
195
|
+
relatedTarget !== this &&
|
196
|
+
(this.contains(relatedTarget as Node) ||
|
197
|
+
(this.shadowRoot?.contains(relatedTarget as Node) &&
|
198
|
+
relatedTarget !== (startSentinelNode as Node) &&
|
199
|
+
relatedTarget !== (endSentinelNode as Node)));
|
200
|
+
|
201
|
+
// Performs focus wrapping if _all_ of the following is met:
|
202
|
+
// * This tearsheet is open
|
203
|
+
// * The viewport still has focus
|
204
|
+
// * Tearsheet body used to have focus but no longer has focus
|
205
|
+
const { selectorTabbable: selectorTabbableForTearsheet } = this
|
206
|
+
.constructor as typeof CDSTearsheet;
|
207
|
+
|
208
|
+
if (open && relatedTarget && oldContains && !currentContains) {
|
209
|
+
const comparisonResult = (target as Node).compareDocumentPosition(
|
210
|
+
relatedTarget as Node
|
211
|
+
);
|
212
|
+
// eslint-disable-next-line no-bitwise
|
213
|
+
if (relatedTarget === startSentinelNode || comparisonResult & PRECEDING) {
|
214
|
+
await (this.constructor as typeof CDSTearsheet)._delay();
|
215
|
+
if (
|
216
|
+
!tryFocusElements(
|
217
|
+
this.querySelectorAll(selectorTabbableForTearsheet),
|
218
|
+
true
|
219
|
+
) &&
|
220
|
+
relatedTarget !== this
|
221
|
+
) {
|
222
|
+
this.focus();
|
223
|
+
}
|
224
|
+
}
|
225
|
+
// eslint-disable-next-line no-bitwise
|
226
|
+
else if (
|
227
|
+
relatedTarget === endSentinelNode ||
|
228
|
+
comparisonResult & FOLLOWING
|
229
|
+
) {
|
230
|
+
await (this.constructor as typeof CDSTearsheet)._delay();
|
231
|
+
if (
|
232
|
+
!tryFocusElements(
|
233
|
+
this.querySelectorAll(selectorTabbableForTearsheet),
|
234
|
+
true
|
235
|
+
)
|
236
|
+
) {
|
237
|
+
this.focus();
|
238
|
+
}
|
239
|
+
}
|
240
|
+
}
|
241
|
+
};
|
242
|
+
|
243
|
+
@HostListener('document:keydown')
|
244
|
+
// @ts-ignore: The decorator refers to this method but TS thinks this method is not referred to
|
245
|
+
private _handleKeydown = ({ key, target }: KeyboardEvent) => {
|
246
|
+
if ((key === 'Esc' || key === 'Escape') && this._topOfStack()) {
|
247
|
+
this._handleUserInitiatedClose(target);
|
248
|
+
}
|
249
|
+
};
|
250
|
+
|
251
|
+
private _checkSetHasSlot(e: Event) {
|
252
|
+
const t = e.target as HTMLSlotElement;
|
253
|
+
const dataPostfix = t.getAttribute('data-postfix');
|
254
|
+
const postfix = dataPostfix ? `-${dataPostfix}` : '';
|
255
|
+
|
256
|
+
// snake `ab-cd-ef` to _has camel case _hasAbCdEf
|
257
|
+
const hasName = `_has-${t.name}${postfix}`.replace(/-./g, (c) =>
|
258
|
+
c[1].toUpperCase()
|
259
|
+
);
|
260
|
+
this[hasName] = (t?.assignedElements()?.length ?? 0) > 0;
|
261
|
+
}
|
262
|
+
|
263
|
+
/**
|
264
|
+
* Handles `click` event on the modal container.
|
265
|
+
*
|
266
|
+
* @param event The event.
|
267
|
+
*/
|
268
|
+
private _handleClickContainer(event: MouseEvent) {
|
269
|
+
if (
|
270
|
+
(event.target as Element).matches(
|
271
|
+
(this.constructor as typeof CDSTearsheet).selectorCloseButton
|
272
|
+
)
|
273
|
+
) {
|
274
|
+
this._handleUserInitiatedClose(event.target);
|
275
|
+
}
|
276
|
+
}
|
277
|
+
|
278
|
+
/**
|
279
|
+
* Handles user-initiated close request of this tearsheet.
|
280
|
+
*
|
281
|
+
* @param triggeredBy The element that triggered this close request.
|
282
|
+
*/
|
283
|
+
private _handleUserInitiatedClose(triggeredBy: EventTarget | null) {
|
284
|
+
if (this.open) {
|
285
|
+
const init = {
|
286
|
+
bubbles: true,
|
287
|
+
cancelable: true,
|
288
|
+
composed: true,
|
289
|
+
detail: {
|
290
|
+
triggeredBy,
|
291
|
+
},
|
292
|
+
};
|
293
|
+
if (
|
294
|
+
this.dispatchEvent(
|
295
|
+
new CustomEvent(
|
296
|
+
(this.constructor as typeof CDSTearsheet).eventBeforeClose,
|
297
|
+
init
|
298
|
+
)
|
299
|
+
)
|
300
|
+
) {
|
301
|
+
this.open = false;
|
302
|
+
this.dispatchEvent(
|
303
|
+
new CustomEvent(
|
304
|
+
(this.constructor as typeof CDSTearsheet).eventClose,
|
305
|
+
init
|
306
|
+
)
|
307
|
+
);
|
308
|
+
}
|
309
|
+
}
|
310
|
+
}
|
311
|
+
|
312
|
+
private _handleSlugChange(e: Event) {
|
313
|
+
const childItems = (e.target as HTMLSlotElement).assignedElements();
|
314
|
+
|
315
|
+
this._hasSlug = childItems.length > 0;
|
316
|
+
if (this._hasSlug) {
|
317
|
+
childItems[0].setAttribute('size', 'lg');
|
318
|
+
this.setAttribute('slug', '');
|
319
|
+
} else {
|
320
|
+
this.removeAttribute('slug');
|
321
|
+
}
|
322
|
+
}
|
323
|
+
|
324
|
+
/**
|
325
|
+
* Optional aria label for the tearsheet
|
326
|
+
*/
|
327
|
+
@property({ reflect: true, attribute: 'aria-label' })
|
328
|
+
ariaLabel = '';
|
329
|
+
|
330
|
+
/**
|
331
|
+
* Sets the close button icon description
|
332
|
+
*/
|
333
|
+
@property({ reflect: true, attribute: 'close-icon-description' })
|
334
|
+
closeIconDescription = 'Close';
|
335
|
+
|
336
|
+
/**
|
337
|
+
* Enable a close icon ('x') in the header area of the tearsheet. By default,
|
338
|
+
* (when this prop is omitted, or undefined or null) a tearsheet does not
|
339
|
+
* display a close icon if there are navigation actions ("transactional
|
340
|
+
* tearsheet") and displays one if there are no navigation actions ("passive
|
341
|
+
* tearsheet"), and that behavior can be overridden if required by setting
|
342
|
+
* this prop to either true or false.
|
343
|
+
*/
|
344
|
+
|
345
|
+
@property({ reflect: true, type: Boolean, attribute: 'has-close-icon' })
|
346
|
+
hasCloseIcon = false;
|
347
|
+
|
348
|
+
/**
|
349
|
+
* The placement of the influencer section, 'left' or 'right'.
|
350
|
+
*/
|
351
|
+
@property({ reflect: true, attribute: 'influencer-placement' })
|
352
|
+
influencerPlacement = TEARSHEET_INFLUENCER_PLACEMENT.RIGHT;
|
353
|
+
|
354
|
+
/**
|
355
|
+
* The width of the influencer section, 'narrow' or 'wide'.
|
356
|
+
*/
|
357
|
+
@property({ reflect: true, attribute: 'influencer-width' })
|
358
|
+
influencerWidth = TEARSHEET_INFLUENCER_WIDTH.NARROW;
|
359
|
+
|
360
|
+
/**
|
361
|
+
* `true` if the tearsheet should be open.
|
362
|
+
*/
|
363
|
+
@property({ type: Boolean, reflect: true })
|
364
|
+
open = false;
|
365
|
+
|
366
|
+
/**
|
367
|
+
* Prevent closing on click outside of tearsheet
|
368
|
+
*/
|
369
|
+
@property({ type: Boolean, attribute: 'prevent-close-on-click-outside' })
|
370
|
+
preventCloseOnClickOutside = false;
|
371
|
+
|
372
|
+
/**
|
373
|
+
* The initial location of focus in the side panel
|
374
|
+
*/
|
375
|
+
@property({
|
376
|
+
reflect: true,
|
377
|
+
attribute: 'selector-initial-focus',
|
378
|
+
type: String,
|
379
|
+
})
|
380
|
+
selectorInitialFocus;
|
381
|
+
|
382
|
+
/**
|
383
|
+
* The width of the influencer section, 'narrow' or 'wide'.
|
384
|
+
*/
|
385
|
+
@property({ reflect: true, attribute: 'width' })
|
386
|
+
width = TEARSHEET_WIDTH.NARROW;
|
387
|
+
|
388
|
+
private _checkUpdateActionSizes = () => {
|
389
|
+
if (this._actions) {
|
390
|
+
for (let i = 0; i < this._actions.length; i++) {
|
391
|
+
this._actions[i].setAttribute(
|
392
|
+
'size',
|
393
|
+
this.width === 'wide' ? '2xl' : 'xl'
|
394
|
+
);
|
395
|
+
}
|
396
|
+
}
|
397
|
+
};
|
398
|
+
|
399
|
+
private _maxActions = 4;
|
400
|
+
private _handleActionsChange(e: Event) {
|
401
|
+
const target = e.target as HTMLSlotElement;
|
402
|
+
const actions = target?.assignedElements();
|
403
|
+
const actionsCount = actions?.length ?? 0;
|
404
|
+
|
405
|
+
if (actionsCount > this._maxActions) {
|
406
|
+
this._actionsCount = this._maxActions;
|
407
|
+
console.error(`Too many tearsheet actions, max ${this._maxActions}.`);
|
408
|
+
} else {
|
409
|
+
this._actionsCount = actionsCount;
|
410
|
+
}
|
411
|
+
|
412
|
+
for (let i = 0; i < actions?.length; i++) {
|
413
|
+
if (i + 1 > this._maxActions) {
|
414
|
+
// hide excessive tearsheet actions
|
415
|
+
actions[i].setAttribute('hidden', 'true');
|
416
|
+
actions[i].setAttribute(
|
417
|
+
`data-actions-limit-${this._maxActions}-exceeded`,
|
418
|
+
`${actions.length}`
|
419
|
+
);
|
420
|
+
} else {
|
421
|
+
actions[i].classList.add(`${blockClassActionSet}__action-button`);
|
422
|
+
}
|
423
|
+
}
|
424
|
+
this._checkUpdateActionSizes();
|
425
|
+
}
|
426
|
+
|
427
|
+
// Data structure to communicate the state of tearsheet stacking
|
428
|
+
// (i.e. when more than one tearsheet is open). Each tearsheet supplies a
|
429
|
+
// handler to be called whenever the stacking of the tearsheets changes, which
|
430
|
+
// happens when a tearsheet opens or closes. The 'open' array contains one
|
431
|
+
// handler per OPEN tearsheet ordered from lowest to highest in visual z-order.
|
432
|
+
// The 'all' array contains all the handlers for open and closed tearsheets.
|
433
|
+
|
434
|
+
@state()
|
435
|
+
_stackDepth = -1;
|
436
|
+
|
437
|
+
@state()
|
438
|
+
_stackPosition = -1;
|
439
|
+
|
440
|
+
private _topOfStack = () => {
|
441
|
+
return this._stackDepth === this._stackPosition;
|
442
|
+
};
|
443
|
+
|
444
|
+
private static _stack: StackState = {
|
445
|
+
open: [],
|
446
|
+
all: [],
|
447
|
+
};
|
448
|
+
private _notifyStack = () => {
|
449
|
+
CDSTearsheet._stack.all.forEach(
|
450
|
+
(handler: (stackSize: number, position: number) => void) => {
|
451
|
+
handler(
|
452
|
+
Math.min(CDSTearsheet._stack.open.length, maxStackDepth),
|
453
|
+
CDSTearsheet._stack.open.indexOf(handler) + 1
|
454
|
+
);
|
455
|
+
}
|
456
|
+
);
|
457
|
+
};
|
458
|
+
|
459
|
+
private _handleStackChange: StackHandler = (newDepth, newPosition) => {
|
460
|
+
this._stackDepth = newDepth;
|
461
|
+
this._stackPosition = newPosition;
|
462
|
+
if (this._stackDepth > 1 && this._stackPosition > 0) {
|
463
|
+
this.setAttribute('stack-position', `${newPosition}`);
|
464
|
+
this.setAttribute('stack-depth', `${this._stackDepth}`);
|
465
|
+
} else {
|
466
|
+
this.removeAttribute('stack-position');
|
467
|
+
this.removeAttribute('stack-depth');
|
468
|
+
}
|
469
|
+
};
|
470
|
+
|
471
|
+
private _updateStack = () => {
|
472
|
+
if (this.open) {
|
473
|
+
CDSTearsheet._stack.open.push(this._handleStackChange);
|
474
|
+
} else {
|
475
|
+
const indexOpen = CDSTearsheet._stack.open.indexOf(
|
476
|
+
this._handleStackChange
|
477
|
+
);
|
478
|
+
if (indexOpen >= 0) {
|
479
|
+
CDSTearsheet._stack.open.splice(indexOpen, 1);
|
480
|
+
}
|
481
|
+
}
|
482
|
+
this._notifyStack();
|
483
|
+
};
|
484
|
+
|
485
|
+
actionsMultiple = ['', 'single', 'double', 'triple'][this._actionsCount];
|
486
|
+
|
487
|
+
connectedCallback() {
|
488
|
+
super.connectedCallback();
|
489
|
+
|
490
|
+
CDSTearsheet._stack.all.push(this._handleStackChange);
|
491
|
+
}
|
492
|
+
|
493
|
+
disconnectedCallback() {
|
494
|
+
super.disconnectedCallback();
|
495
|
+
|
496
|
+
const indexAll = CDSTearsheet._stack.all.indexOf(this._handleStackChange);
|
497
|
+
CDSTearsheet._stack.all.splice(indexAll, 1);
|
498
|
+
const indexOpen = CDSTearsheet._stack.all.indexOf(this._handleStackChange);
|
499
|
+
CDSTearsheet._stack.open.splice(indexOpen, 1);
|
500
|
+
}
|
501
|
+
|
502
|
+
render() {
|
503
|
+
const {
|
504
|
+
closeIconDescription,
|
505
|
+
influencerPlacement,
|
506
|
+
influencerWidth,
|
507
|
+
open,
|
508
|
+
width,
|
509
|
+
} = this;
|
510
|
+
|
511
|
+
const actionsMultiple = ['', 'single', 'double', 'triple'][
|
512
|
+
this._actionsCount
|
513
|
+
];
|
514
|
+
const headerFieldsTemplate = html`<div
|
515
|
+
class=${`${blockClass}__header-fields`}
|
516
|
+
>
|
517
|
+
<h2 class=${`${blockClassModalHeader}__label`} ?hidden=${!this._hasLabel}>
|
518
|
+
<slot name="label" @slotchange=${this._checkSetHasSlot}></slot>
|
519
|
+
</h2>
|
520
|
+
<h3
|
521
|
+
class=${`${blockClassModalHeader}__heading ${blockClass}__heading`}
|
522
|
+
?hidden=${!this._hasTitle}
|
523
|
+
>
|
524
|
+
<slot name="title" @slotchange=${this._checkSetHasSlot}></slot>
|
525
|
+
</h3>
|
526
|
+
<div
|
527
|
+
class=${`${blockClass}__header-description`}
|
528
|
+
?hidden=${!this._hasDescription}
|
529
|
+
>
|
530
|
+
<slot name="description" @slotchange=${this._checkSetHasSlot}></slot>
|
531
|
+
</div>
|
532
|
+
</div>`;
|
533
|
+
|
534
|
+
const headerActionsTemplate = html` <div
|
535
|
+
class=${`${blockClass}__header-actions`}
|
536
|
+
?hidden=${!this._hasHeaderActions || this.width === 'narrow'}
|
537
|
+
>
|
538
|
+
<slot name="header-actions" @slotchange=${this._checkSetHasSlot}></slot>
|
539
|
+
</div>`;
|
540
|
+
|
541
|
+
const headerTemplate = html` <cds-modal-header
|
542
|
+
class=${`${blockClass}__header`}
|
543
|
+
?has-close-icon=${this.hasCloseIcon || this?._actionsCount === 0}
|
544
|
+
?has-navigation=${this._hasHeaderNavigation && this.width === 'wide'}
|
545
|
+
?has-header-actions=${this._hasHeaderActions && this.width === 'wide'}
|
546
|
+
?has-actions=${this?._actionsCount > 0}
|
547
|
+
?has-slug=${this?._hasSlug}
|
548
|
+
width=${width}
|
549
|
+
>
|
550
|
+
${this.width === TEARSHEET_WIDTH.WIDE
|
551
|
+
? html`<cds-layer level="1" class=${`${blockClass}__header-content`}
|
552
|
+
>${headerFieldsTemplate}${headerActionsTemplate}</cds-layer
|
553
|
+
>`
|
554
|
+
: html`<div>${headerFieldsTemplate}${headerActionsTemplate}</div>`}
|
555
|
+
|
556
|
+
<div
|
557
|
+
class=${`${blockClass}__header-navigation`}
|
558
|
+
?hidden=${!this._hasHeaderNavigation || this.width === 'narrow'}
|
559
|
+
>
|
560
|
+
<slot
|
561
|
+
name="header-navigation"
|
562
|
+
@slotchange=${this._checkSetHasSlot}
|
563
|
+
></slot>
|
564
|
+
</div>
|
565
|
+
<slot name="slug" @slotchange=${this._handleSlugChange}></slot>
|
566
|
+
${this.hasCloseIcon || this?._actionsCount === 0
|
567
|
+
? html`<cds-modal-close-button
|
568
|
+
close-button-label=${closeIconDescription}
|
569
|
+
@click=${this._handleUserInitiatedClose}
|
570
|
+
></cds-modal-close-button>`
|
571
|
+
: ''}
|
572
|
+
</cds-modal-header>`;
|
573
|
+
|
574
|
+
return html`
|
575
|
+
<a
|
576
|
+
id="start-sentinel"
|
577
|
+
class="${prefix}--visually-hidden"
|
578
|
+
href="javascript:void 0"
|
579
|
+
role="navigation"
|
580
|
+
></a>
|
581
|
+
<div
|
582
|
+
aria-label=${this.ariaLabel}
|
583
|
+
class=${`${blockClass}__container ${carbonPrefix}--modal-container ${carbonPrefix}--modal-container--sm`}
|
584
|
+
part="dialog"
|
585
|
+
role="complementary"
|
586
|
+
?open=${this._isOpen}
|
587
|
+
?opening=${open && !this._isOpen}
|
588
|
+
?closing=${!open && this._isOpen}
|
589
|
+
width=${width}
|
590
|
+
stack-position=${this._stackPosition}
|
591
|
+
stack-depth=${this._stackDepth}
|
592
|
+
@click=${this._handleClickContainer}
|
593
|
+
>
|
594
|
+
<!-- Header -->
|
595
|
+
${headerTemplate}
|
596
|
+
|
597
|
+
<!-- Body -->
|
598
|
+
<cds-modal-body class=${`${blockClass}__body`} width=${width}>
|
599
|
+
<!-- Influencer when on left -->
|
600
|
+
${influencerPlacement !== TEARSHEET_INFLUENCER_PLACEMENT.RIGHT
|
601
|
+
? html`<div
|
602
|
+
class=${`${blockClass}__influencer`}
|
603
|
+
?wide=${influencerWidth === 'wide'}
|
604
|
+
?hidden=${!this._hasInfluencerLeft ||
|
605
|
+
this.width === TEARSHEET_WIDTH.NARROW}
|
606
|
+
>
|
607
|
+
<slot
|
608
|
+
name="influencer"
|
609
|
+
data-postfix="left"
|
610
|
+
@slotchange=${this._checkSetHasSlot}
|
611
|
+
></slot>
|
612
|
+
</div>`
|
613
|
+
: ''}
|
614
|
+
|
615
|
+
<div class=${`${blockClass}__right`}>
|
616
|
+
<div class=${`${blockClass}__main`}>
|
617
|
+
<div class=${`${blockClass}__content`}>
|
618
|
+
<cds-layer level="0">
|
619
|
+
<slot></slot>
|
620
|
+
</cds-layer>
|
621
|
+
</div>
|
622
|
+
|
623
|
+
<!-- Influencer when on right -->
|
624
|
+
${influencerPlacement === TEARSHEET_INFLUENCER_PLACEMENT.RIGHT
|
625
|
+
? html`<div
|
626
|
+
class=${`${blockClass}__influencer`}
|
627
|
+
?wide=${influencerWidth}
|
628
|
+
?hidden=${!this._hasInfluencerRight ||
|
629
|
+
this.width === TEARSHEET_WIDTH.NARROW}
|
630
|
+
>
|
631
|
+
<slot
|
632
|
+
name="influencer"
|
633
|
+
data-postfix="right"
|
634
|
+
@slotchange=${this._checkSetHasSlot}
|
635
|
+
></slot>
|
636
|
+
</div>`
|
637
|
+
: ''}
|
638
|
+
</div>
|
639
|
+
<!-- Action buttons -->
|
640
|
+
<cds-button-set-base
|
641
|
+
class=${`${blockClass}__buttons ${blockClass}__button-container`}
|
642
|
+
actions-multiple=${actionsMultiple}
|
643
|
+
?tearsheet-wide=${width === 'wide'}
|
644
|
+
?hidden=${this._actionsCount === 0}
|
645
|
+
>
|
646
|
+
<slot
|
647
|
+
name="actions"
|
648
|
+
@slotchange=${this._handleActionsChange}
|
649
|
+
></slot>
|
650
|
+
</cds-button-set-base>
|
651
|
+
</div>
|
652
|
+
</cds-modal-body>
|
653
|
+
</div>
|
654
|
+
<a
|
655
|
+
id="end-sentinel"
|
656
|
+
class="${prefix}--visually-hidden"
|
657
|
+
href="javascript:void 0"
|
658
|
+
role="navigation"
|
659
|
+
></a>
|
660
|
+
`;
|
661
|
+
}
|
662
|
+
|
663
|
+
_checkSetOpen = () => {
|
664
|
+
const { _tearsheet: tearsheet } = this;
|
665
|
+
if (tearsheet && this._isOpen) {
|
666
|
+
// wait until the tearsheet has transitioned off the screen to remove
|
667
|
+
tearsheet.addEventListener('transitionend', () => {
|
668
|
+
this._isOpen = false;
|
669
|
+
});
|
670
|
+
} else {
|
671
|
+
// allow the html to render before animating in the tearsheet
|
672
|
+
window.requestAnimationFrame(() => {
|
673
|
+
this._isOpen = this.open;
|
674
|
+
});
|
675
|
+
}
|
676
|
+
};
|
677
|
+
|
678
|
+
async updated(changedProperties) {
|
679
|
+
if (changedProperties.has('width')) {
|
680
|
+
this._checkUpdateActionSizes();
|
681
|
+
}
|
682
|
+
|
683
|
+
if (
|
684
|
+
process.env.NODE_ENV === 'development' &&
|
685
|
+
(changedProperties.has('width') ||
|
686
|
+
changedProperties.has('_hasHeaderNavigation') ||
|
687
|
+
changedProperties.has('_hasInfluencerLeft') ||
|
688
|
+
changedProperties.has('_hasInfluencerRight') ||
|
689
|
+
changedProperties.has('_hasHeaderActions'))
|
690
|
+
) {
|
691
|
+
if (this.width === 'narrow') {
|
692
|
+
if (this._hasHeaderNavigation) {
|
693
|
+
console.error(
|
694
|
+
`Header navigation is not permitted in narrow Tearsheet.`
|
695
|
+
);
|
696
|
+
}
|
697
|
+
if (this._hasInfluencerLeft || this._hasInfluencerRight) {
|
698
|
+
console.error(`Influencer is not permitted in narrow Tearsheet.`);
|
699
|
+
}
|
700
|
+
if (this._hasHeaderActions) {
|
701
|
+
console.error(
|
702
|
+
`Header actions are not permitted in narrow Tearsheet.`
|
703
|
+
);
|
704
|
+
}
|
705
|
+
}
|
706
|
+
}
|
707
|
+
|
708
|
+
if (changedProperties.has('open')) {
|
709
|
+
this._updateStack();
|
710
|
+
|
711
|
+
this._checkSetOpen();
|
712
|
+
if (this.open) {
|
713
|
+
this._launcher = this.ownerDocument!.activeElement;
|
714
|
+
const focusNode =
|
715
|
+
this.selectorInitialFocus &&
|
716
|
+
this.querySelector(this.selectorInitialFocus);
|
717
|
+
|
718
|
+
await (this.constructor as typeof CDSTearsheet)._delay();
|
719
|
+
if (focusNode) {
|
720
|
+
// For cases where a `carbon-web-components` component (e.g. `<cds-button>`) being `primaryFocusNode`,
|
721
|
+
// where its first update/render cycle that makes it focusable happens after `<c4p-tearsheet>`'s first update/render cycle
|
722
|
+
(focusNode as HTMLElement).focus();
|
723
|
+
} else if (
|
724
|
+
!tryFocusElements(
|
725
|
+
this.querySelectorAll(
|
726
|
+
(this.constructor as typeof CDSTearsheet).selectorTabbable
|
727
|
+
),
|
728
|
+
true
|
729
|
+
)
|
730
|
+
) {
|
731
|
+
this.focus();
|
732
|
+
}
|
733
|
+
} else if (
|
734
|
+
this._launcher &&
|
735
|
+
typeof (this._launcher as HTMLElement).focus === 'function'
|
736
|
+
) {
|
737
|
+
(this._launcher as HTMLElement).focus();
|
738
|
+
this._launcher = null;
|
739
|
+
}
|
740
|
+
}
|
741
|
+
}
|
742
|
+
|
743
|
+
/**
|
744
|
+
* @param ms The number of milliseconds.
|
745
|
+
* @returns A promise that is resolves after the given milliseconds.
|
746
|
+
*/
|
747
|
+
private static _delay(ms = 0) {
|
748
|
+
return new Promise((resolve) => {
|
749
|
+
setTimeout(resolve, ms);
|
750
|
+
});
|
751
|
+
}
|
752
|
+
|
753
|
+
/**
|
754
|
+
* A selector selecting buttons that should close this modal.
|
755
|
+
*/
|
756
|
+
static get selectorCloseButton() {
|
757
|
+
return `[data-modal-close],${carbonPrefix}-modal-close-button`;
|
758
|
+
}
|
759
|
+
|
760
|
+
/**
|
761
|
+
* A selector selecting tabbable nodes.
|
762
|
+
*/
|
763
|
+
static get selectorTabbable() {
|
764
|
+
return selectorTabbable;
|
765
|
+
}
|
766
|
+
|
767
|
+
/**
|
768
|
+
* The name of the custom event fired before this tearsheet is being closed upon a user gesture.
|
769
|
+
* Cancellation of this event stops the user-initiated action of closing this tearsheet.
|
770
|
+
*/
|
771
|
+
static get eventBeforeClose() {
|
772
|
+
return `${prefix}-tearsheet-beingclosed`;
|
773
|
+
}
|
774
|
+
|
775
|
+
/**
|
776
|
+
* The name of the custom event fired after this tearsheet is closed upon a user gesture.
|
777
|
+
*/
|
778
|
+
static get eventClose() {
|
779
|
+
return `${prefix}-tearsheet-closed`;
|
780
|
+
}
|
781
|
+
|
782
|
+
/**
|
783
|
+
* The name of the custom event fired on clicking the navigate back button
|
784
|
+
*/
|
785
|
+
static get eventNavigateBack() {
|
786
|
+
return `${prefix}-tearsheet-header-navigate-back`;
|
787
|
+
}
|
788
|
+
|
789
|
+
static styles = styles; // `styles` here is a `CSSResult` generated by custom WebPack loader
|
790
|
+
}
|
791
|
+
|
792
|
+
export default CDSTearsheet;
|