bareframe 0.1.0 → 0.1.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +179 -0
- package/dist/bareframe.min.js +119 -0
- package/dist/components/accordion.js +66 -0
- package/dist/components/autocomplete.css +78 -15
- package/dist/components/autocomplete.js +220 -10
- package/dist/components/avatar.css +129 -17
- package/dist/components/avatar.js +47 -10
- package/dist/components/breadcrumb.css +63 -17
- package/dist/components/breadcrumb.js +140 -5
- package/dist/components/button.css +4 -0
- package/dist/components/button.js +95 -15
- package/dist/components/chart.css +163 -14
- package/dist/components/chart.js +59 -4
- package/dist/components/checkbox.css +43 -1
- package/dist/components/checkbox.js +98 -5
- package/dist/components/dialog.css +95 -0
- package/dist/components/dialog.js +172 -4
- package/dist/components/divider.css +18 -22
- package/dist/components/divider.js +31 -3
- package/dist/components/drawer.css +68 -18
- package/dist/components/drawer.js +84 -4
- package/dist/components/edge.css +54 -0
- package/dist/components/edge.js +55 -0
- package/dist/components/file-upload.css +72 -3
- package/dist/components/file-upload.js +186 -4
- package/dist/components/input.css +59 -0
- package/dist/components/input.js +369 -4
- package/dist/components/list.css +11 -0
- package/dist/components/list.js +45 -0
- package/dist/components/menu.css +20 -0
- package/dist/components/menu.js +144 -0
- package/dist/components/modal.css +30 -17
- package/dist/components/modal.js +68 -4
- package/dist/components/nav.css +39 -0
- package/dist/components/progress.css +196 -0
- package/dist/components/progress.js +304 -0
- package/dist/components/radio.css +35 -1
- package/dist/components/radio.js +86 -5
- package/dist/components/range.css +91 -0
- package/dist/components/range.js +250 -0
- package/dist/components/select.css +35 -1
- package/dist/components/select.js +255 -4
- package/dist/components/skeleton.css +108 -21
- package/dist/components/skeleton.js +57 -4
- package/dist/components/tab.css +9 -1
- package/dist/components/tab.js +66 -1
- package/dist/components/tag.css +36 -3
- package/dist/components/tag.js +32 -0
- package/dist/components/toast.css +113 -0
- package/dist/components/toast.js +265 -4
- package/dist/components/toggle.css +53 -0
- package/dist/components/toggle.js +73 -5
- package/dist/components/wizard.css +79 -14
- package/dist/components/wizard.js +141 -4
- package/dist/index.js +5147 -110
- package/dist/manifest.json +5 -42
- package/dist/themes/aurora.css +47 -0
- package/dist/themes/dark.css +12 -2
- package/dist/themes/desert.css +37 -0
- package/dist/themes/future.css +47 -0
- package/dist/themes/layout.css +191 -0
- package/dist/themes/light.css +12 -0
- package/dist/themes/matrix.css +37 -0
- package/dist/themes/modern.css +64 -0
- package/dist/themes/nature.css +47 -0
- package/dist/themes/nebula.css +37 -0
- package/dist/themes/noir.css +37 -0
- package/dist/themes/oceanic.css +37 -0
- package/dist/themes/retro.css +47 -0
- package/dist/themes/simple.css +47 -0
- package/dist/themes/sprint.css +12 -0
- package/dist/themes/sunrise.css +37 -0
- package/dist/themes/system.css +13 -0
- package/package.json +9 -2
- package/dist/components/alert.css +0 -30
- package/dist/components/alert.js +0 -31
- package/dist/components/badge.css +0 -30
- package/dist/components/badge.js +0 -31
- package/dist/components/banner.css +0 -30
- package/dist/components/banner.js +0 -31
- package/dist/components/bar-chart.css +0 -30
- package/dist/components/bar-chart.js +0 -31
- package/dist/components/bottom-sheet.css +0 -30
- package/dist/components/bottom-sheet.js +0 -31
- package/dist/components/button-group.css +0 -30
- package/dist/components/button-group.js +0 -31
- package/dist/components/chip.css +0 -30
- package/dist/components/chip.js +0 -31
- package/dist/components/color-picker.css +0 -30
- package/dist/components/color-picker.js +0 -31
- package/dist/components/context-menu.css +0 -30
- package/dist/components/context-menu.js +0 -31
- package/dist/components/donut-chart.css +0 -30
- package/dist/components/donut-chart.js +0 -31
- package/dist/components/expanded-panel.css +0 -30
- package/dist/components/expanded-panel.js +0 -31
- package/dist/components/footer.css +0 -30
- package/dist/components/footer.js +0 -31
- package/dist/components/gantt-chart.css +0 -30
- package/dist/components/gantt-chart.js +0 -31
- package/dist/components/gauge.css +0 -30
- package/dist/components/gauge.js +0 -31
- package/dist/components/graph.css +0 -30
- package/dist/components/graph.js +0 -31
- package/dist/components/header.css +0 -30
- package/dist/components/header.js +0 -31
- package/dist/components/heatmap.css +0 -30
- package/dist/components/heatmap.js +0 -31
- package/dist/components/line-chart.css +0 -30
- package/dist/components/line-chart.js +0 -31
- package/dist/components/list-item.css +0 -30
- package/dist/components/list-item.js +0 -31
- package/dist/components/menu-item.css +0 -30
- package/dist/components/menu-item.js +0 -31
- package/dist/components/multi-select.css +0 -30
- package/dist/components/multi-select.js +0 -31
- package/dist/components/notification.css +0 -30
- package/dist/components/notification.js +0 -31
- package/dist/components/pie-chart.css +0 -30
- package/dist/components/pie-chart.js +0 -31
- package/dist/components/popover.css +0 -30
- package/dist/components/popover.js +0 -31
- package/dist/components/progress-bar.css +0 -30
- package/dist/components/progress-bar.js +0 -31
- package/dist/components/progress-circle.css +0 -30
- package/dist/components/progress-circle.js +0 -31
- package/dist/components/radio-group.css +0 -30
- package/dist/components/radio-group.js +0 -31
- package/dist/components/range-slider.css +0 -30
- package/dist/components/range-slider.js +0 -31
- package/dist/components/rating.css +0 -30
- package/dist/components/rating.js +0 -31
- package/dist/components/sheet.css +0 -30
- package/dist/components/sheet.js +0 -31
- package/dist/components/slider.css +0 -30
- package/dist/components/slider.js +0 -31
- package/dist/components/snackbar.css +0 -30
- package/dist/components/snackbar.js +0 -31
- package/dist/components/sparkline.css +0 -30
- package/dist/components/sparkline.js +0 -31
- package/dist/components/stepper.css +0 -30
- package/dist/components/stepper.js +0 -31
- package/dist/components/switch.css +0 -30
- package/dist/components/switch.js +0 -31
- package/dist/components/tab-group.css +0 -30
- package/dist/components/tab-group.js +0 -31
- package/dist/components/textfield.css +0 -30
- package/dist/components/textfield.js +0 -31
- package/dist/components/tooltip.css +0 -30
- package/dist/components/tooltip.js +0 -31
- package/dist/components/treemap.css +0 -30
- package/dist/components/treemap.js +0 -31
- package/dist/components/upload-dropzone.css +0 -30
- package/dist/components/upload-dropzone.js +0 -31
|
@@ -1,8 +1,11 @@
|
|
|
1
1
|
class BfAccordion extends HTMLElement {
|
|
2
|
+
static observedAttributes = ['active-id'];
|
|
3
|
+
|
|
2
4
|
constructor() {
|
|
3
5
|
super();
|
|
4
6
|
this.attachShadow({ mode: 'open' });
|
|
5
7
|
this._details = [];
|
|
8
|
+
this._itemIds = [];
|
|
6
9
|
}
|
|
7
10
|
|
|
8
11
|
connectedCallback() {
|
|
@@ -31,12 +34,14 @@ class BfAccordion extends HTMLElement {
|
|
|
31
34
|
`Section ${index + 1}`;
|
|
32
35
|
const slotName = `item-${index + 1}`;
|
|
33
36
|
const openByDefault = item.hasAttribute('open');
|
|
37
|
+
const itemId = this._ensureItemId(item, index);
|
|
34
38
|
|
|
35
39
|
item.setAttribute('slot', slotName);
|
|
36
40
|
|
|
37
41
|
const details = document.createElement('details');
|
|
38
42
|
details.className = 'accordion-item';
|
|
39
43
|
details.setAttribute('part', 'item');
|
|
44
|
+
details.dataset.itemId = itemId;
|
|
40
45
|
if (openByDefault) {
|
|
41
46
|
details.open = true;
|
|
42
47
|
}
|
|
@@ -57,6 +62,7 @@ class BfAccordion extends HTMLElement {
|
|
|
57
62
|
details.append(summary, panel);
|
|
58
63
|
container.append(details);
|
|
59
64
|
this._details.push(details);
|
|
65
|
+
this._itemIds.push(itemId);
|
|
60
66
|
|
|
61
67
|
details.addEventListener('toggle', () => {
|
|
62
68
|
if (!this.multiple && details.open) {
|
|
@@ -73,6 +79,7 @@ class BfAccordion extends HTMLElement {
|
|
|
73
79
|
composed: true,
|
|
74
80
|
detail: {
|
|
75
81
|
index,
|
|
82
|
+
id: itemId,
|
|
76
83
|
title,
|
|
77
84
|
open: details.open,
|
|
78
85
|
},
|
|
@@ -82,11 +89,70 @@ class BfAccordion extends HTMLElement {
|
|
|
82
89
|
});
|
|
83
90
|
|
|
84
91
|
this.shadowRoot.replaceChildren(link, container);
|
|
92
|
+
const activeId = this.getAttribute('active-id');
|
|
93
|
+
if (activeId) {
|
|
94
|
+
this.openItem(activeId);
|
|
95
|
+
}
|
|
85
96
|
}
|
|
86
97
|
|
|
87
98
|
get multiple() {
|
|
88
99
|
return this.hasAttribute('multiple');
|
|
89
100
|
}
|
|
101
|
+
|
|
102
|
+
attributeChangedCallback(name, oldValue, newValue) {
|
|
103
|
+
if (!this._initialized) {
|
|
104
|
+
return;
|
|
105
|
+
}
|
|
106
|
+
if (name === 'active-id' && oldValue !== newValue && newValue) {
|
|
107
|
+
this.openItem(newValue);
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
openItem(itemId) {
|
|
112
|
+
const details = this._details.find((entry) => entry.dataset.itemId === itemId);
|
|
113
|
+
if (!details) {
|
|
114
|
+
return;
|
|
115
|
+
}
|
|
116
|
+
details.open = true;
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
closeItem(itemId) {
|
|
120
|
+
const details = this._details.find((entry) => entry.dataset.itemId === itemId);
|
|
121
|
+
if (!details) {
|
|
122
|
+
return;
|
|
123
|
+
}
|
|
124
|
+
details.open = false;
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
toggleItem(itemId) {
|
|
128
|
+
const details = this._details.find((entry) => entry.dataset.itemId === itemId);
|
|
129
|
+
if (!details) {
|
|
130
|
+
return;
|
|
131
|
+
}
|
|
132
|
+
details.open = !details.open;
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
_ensureItemId(item, index) {
|
|
136
|
+
const explicit = item.getAttribute('id');
|
|
137
|
+
if (explicit && !this._itemIds.includes(explicit)) {
|
|
138
|
+
return explicit;
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
const hostId = this.getAttribute('id') || 'bf-accordion';
|
|
142
|
+
let counter = index + 1;
|
|
143
|
+
let candidate = `${hostId}-item-${counter}`;
|
|
144
|
+
while (
|
|
145
|
+
this._itemIds.includes(candidate) ||
|
|
146
|
+
(this.querySelector(`#${CSS.escape(candidate)}`) &&
|
|
147
|
+
this.querySelector(`#${CSS.escape(candidate)}`) !== item)
|
|
148
|
+
) {
|
|
149
|
+
counter += 1;
|
|
150
|
+
candidate = `${hostId}-item-${counter}`;
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
item.setAttribute('id', candidate);
|
|
154
|
+
return candidate;
|
|
155
|
+
}
|
|
90
156
|
}
|
|
91
157
|
|
|
92
158
|
customElements.define('bf-accordion', BfAccordion);
|
|
@@ -3,28 +3,91 @@
|
|
|
3
3
|
--bf-autocomplete-radius: var(--bf-theme-radius-md, 8px);
|
|
4
4
|
--bf-autocomplete-border-width: var(--bf-theme-border-width, 1px);
|
|
5
5
|
--bf-autocomplete-border-style: var(--bf-theme-border-style, solid);
|
|
6
|
-
--bf-autocomplete-border-color: var(
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
--bf-autocomplete-
|
|
11
|
-
|
|
12
|
-
var(--bf-theme-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
6
|
+
--bf-autocomplete-border-color: var(
|
|
7
|
+
--bf-theme-autocomplete-border-color,
|
|
8
|
+
var(--bf-theme-border-1, #d1d5db)
|
|
9
|
+
);
|
|
10
|
+
--bf-autocomplete-bg: var(
|
|
11
|
+
--bf-theme-autocomplete-bg,
|
|
12
|
+
var(--bf-theme-surface-1, #ffffff)
|
|
13
|
+
);
|
|
14
|
+
--bf-autocomplete-color: var(
|
|
15
|
+
--bf-theme-autocomplete-color,
|
|
16
|
+
var(--bf-theme-text-1, #111827)
|
|
17
|
+
);
|
|
18
|
+
--bf-autocomplete-placeholder: var(--bf-theme-text-2, #6b7280);
|
|
19
|
+
--bf-autocomplete-focus: var(--bf-theme-focus-ring-color, #9ca3af);
|
|
20
|
+
--bf-autocomplete-list-max-height: 220px;
|
|
21
|
+
--bf-autocomplete-item-hover-bg: var(--bf-theme-surface-2, #f3f4f6);
|
|
22
|
+
--bf-autocomplete-item-active-bg: var(--bf-theme-surface-2, #e5e7eb);
|
|
23
|
+
--bf-autocomplete-shadow: 0 8px 26px rgba(15, 23, 42, 0.12);
|
|
24
|
+
|
|
25
|
+
display: inline-block;
|
|
26
|
+
width: min(100%, 420px);
|
|
17
27
|
font: var(--bf-autocomplete-font);
|
|
18
|
-
color: var(--bf-autocomplete-color);
|
|
19
28
|
}
|
|
20
29
|
|
|
21
30
|
.root {
|
|
22
|
-
|
|
31
|
+
position: relative;
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
.input {
|
|
35
|
+
width: 100%;
|
|
36
|
+
box-sizing: border-box;
|
|
37
|
+
font: inherit;
|
|
23
38
|
color: var(--bf-autocomplete-color);
|
|
39
|
+
background: var(--bf-autocomplete-bg);
|
|
24
40
|
border-width: var(--bf-autocomplete-border-width);
|
|
25
41
|
border-style: var(--bf-autocomplete-border-style);
|
|
26
42
|
border-color: var(--bf-autocomplete-border-color);
|
|
27
43
|
border-radius: var(--bf-autocomplete-radius);
|
|
28
|
-
padding:
|
|
29
|
-
|
|
44
|
+
padding: 0.62rem 0.8rem;
|
|
45
|
+
outline: none;
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
.input::placeholder {
|
|
49
|
+
color: var(--bf-autocomplete-placeholder);
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
.input:focus-visible {
|
|
53
|
+
border-color: var(--bf-autocomplete-focus);
|
|
54
|
+
box-shadow: 0 0 0 2px color-mix(in srgb, var(--bf-autocomplete-focus) 30%, transparent);
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
.list {
|
|
58
|
+
position: absolute;
|
|
59
|
+
top: calc(100% + 0.35rem);
|
|
60
|
+
left: 0;
|
|
61
|
+
right: 0;
|
|
62
|
+
margin: 0;
|
|
63
|
+
padding: 0.35rem;
|
|
64
|
+
list-style: none;
|
|
65
|
+
border-width: var(--bf-autocomplete-border-width);
|
|
66
|
+
border-style: var(--bf-autocomplete-border-style);
|
|
67
|
+
border-color: var(--bf-autocomplete-border-color);
|
|
68
|
+
border-radius: calc(var(--bf-autocomplete-radius) - 2px);
|
|
69
|
+
background: var(--bf-autocomplete-bg);
|
|
70
|
+
max-height: var(--bf-autocomplete-list-max-height);
|
|
71
|
+
overflow: auto;
|
|
72
|
+
box-shadow: var(--bf-autocomplete-shadow);
|
|
73
|
+
z-index: 20;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
.item {
|
|
77
|
+
padding: 0.48rem 0.55rem;
|
|
78
|
+
border-radius: calc(var(--bf-autocomplete-radius) - 4px);
|
|
79
|
+
cursor: pointer;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
.item:hover {
|
|
83
|
+
background: var(--bf-autocomplete-item-hover-bg);
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
.item.active {
|
|
87
|
+
background: var(--bf-autocomplete-item-active-bg);
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
.item.empty {
|
|
91
|
+
cursor: default;
|
|
92
|
+
color: var(--bf-autocomplete-placeholder);
|
|
30
93
|
}
|
|
@@ -2,6 +2,11 @@ class BfAutocomplete extends HTMLElement {
|
|
|
2
2
|
constructor() {
|
|
3
3
|
super();
|
|
4
4
|
this.attachShadow({ mode: 'open' });
|
|
5
|
+
this._items = [];
|
|
6
|
+
this._filtered = [];
|
|
7
|
+
this._activeIndex = -1;
|
|
8
|
+
this._listOpen = false;
|
|
9
|
+
this._boundOnDocumentClick = this._onDocumentClick.bind(this);
|
|
5
10
|
}
|
|
6
11
|
|
|
7
12
|
connectedCallback() {
|
|
@@ -11,20 +16,225 @@ class BfAutocomplete extends HTMLElement {
|
|
|
11
16
|
this._initialized = true;
|
|
12
17
|
|
|
13
18
|
const cssUrl = new URL('./autocomplete.css', import.meta.url);
|
|
14
|
-
const
|
|
15
|
-
|
|
16
|
-
|
|
19
|
+
const placeholder = this.getAttribute('placeholder') || 'Search...';
|
|
20
|
+
const value = this.getAttribute('value') || '';
|
|
21
|
+
const listId = `ac-list-${Math.random().toString(36).slice(2, 10)}`;
|
|
17
22
|
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
23
|
+
this.shadowRoot.innerHTML = `
|
|
24
|
+
<link rel="stylesheet" href="${cssUrl.href}" />
|
|
25
|
+
<div class="root" part="root">
|
|
26
|
+
<input
|
|
27
|
+
class="input"
|
|
28
|
+
part="input"
|
|
29
|
+
type="text"
|
|
30
|
+
placeholder="${placeholder.replace(/"/g, '"')}"
|
|
31
|
+
autocomplete="off"
|
|
32
|
+
spellcheck="false"
|
|
33
|
+
role="combobox"
|
|
34
|
+
aria-expanded="false"
|
|
35
|
+
aria-controls="${listId}"
|
|
36
|
+
aria-autocomplete="list"
|
|
37
|
+
/>
|
|
38
|
+
<ul class="list" part="list" id="${listId}" role="listbox" hidden></ul>
|
|
39
|
+
</div>
|
|
40
|
+
`;
|
|
22
41
|
|
|
23
|
-
|
|
24
|
-
|
|
42
|
+
this._input = this.shadowRoot.querySelector('.input');
|
|
43
|
+
this._list = this.shadowRoot.querySelector('.list');
|
|
44
|
+
this._input.value = value;
|
|
45
|
+
this._setItemsFromAttributes();
|
|
46
|
+
this._filtered = [...this._items];
|
|
47
|
+
this._renderList();
|
|
48
|
+
|
|
49
|
+
this._input.addEventListener('focus', () => {
|
|
50
|
+
this._filter(this._input.value);
|
|
51
|
+
this._openList();
|
|
52
|
+
});
|
|
53
|
+
|
|
54
|
+
this._input.addEventListener('input', () => {
|
|
55
|
+
const text = this._input.value;
|
|
56
|
+
this._filter(text);
|
|
57
|
+
this._openList();
|
|
58
|
+
this.dispatchEvent(
|
|
59
|
+
new CustomEvent('bf-change', {
|
|
60
|
+
bubbles: true,
|
|
61
|
+
composed: true,
|
|
62
|
+
detail: { value: text },
|
|
63
|
+
}),
|
|
64
|
+
);
|
|
65
|
+
});
|
|
66
|
+
|
|
67
|
+
this._input.addEventListener('keydown', (event) => {
|
|
68
|
+
if (!this._listOpen && (event.key === 'ArrowDown' || event.key === 'ArrowUp')) {
|
|
69
|
+
this._openList();
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
if (event.key === 'ArrowDown') {
|
|
73
|
+
event.preventDefault();
|
|
74
|
+
this._moveActive(1);
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
if (event.key === 'ArrowUp') {
|
|
78
|
+
event.preventDefault();
|
|
79
|
+
this._moveActive(-1);
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
if (event.key === 'Enter' && this._activeIndex >= 0) {
|
|
83
|
+
event.preventDefault();
|
|
84
|
+
const item = this._filtered[this._activeIndex];
|
|
85
|
+
if (item) {
|
|
86
|
+
this._select(item);
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
if (event.key === 'Escape') {
|
|
91
|
+
this._closeList();
|
|
92
|
+
}
|
|
93
|
+
});
|
|
94
|
+
|
|
95
|
+
document.addEventListener('click', this._boundOnDocumentClick);
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
disconnectedCallback() {
|
|
99
|
+
document.removeEventListener('click', this._boundOnDocumentClick);
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
_setItemsFromAttributes() {
|
|
103
|
+
const json = this.getAttribute('options');
|
|
104
|
+
if (json) {
|
|
105
|
+
try {
|
|
106
|
+
const parsed = JSON.parse(json);
|
|
107
|
+
if (Array.isArray(parsed)) {
|
|
108
|
+
this._items = parsed.map((item) => String(item));
|
|
109
|
+
return;
|
|
110
|
+
}
|
|
111
|
+
} catch {
|
|
112
|
+
// Fall through to csv parsing.
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
const csv = this.getAttribute('options-csv');
|
|
117
|
+
if (csv) {
|
|
118
|
+
this._items = csv
|
|
119
|
+
.split(',')
|
|
120
|
+
.map((part) => part.trim())
|
|
121
|
+
.filter(Boolean);
|
|
122
|
+
return;
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
this._items = [
|
|
126
|
+
'Accessibility',
|
|
127
|
+
'Accordion',
|
|
128
|
+
'Autocomplete',
|
|
129
|
+
'Button',
|
|
130
|
+
'Card',
|
|
131
|
+
'Color Picker',
|
|
132
|
+
'Data Grid',
|
|
133
|
+
'Dropdown',
|
|
134
|
+
'Modal',
|
|
135
|
+
'Typography',
|
|
136
|
+
];
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
_filter(query) {
|
|
140
|
+
const lower = query.trim().toLowerCase();
|
|
141
|
+
if (!lower) {
|
|
142
|
+
this._filtered = [...this._items];
|
|
143
|
+
} else {
|
|
144
|
+
this._filtered = this._items.filter((item) =>
|
|
145
|
+
item.toLowerCase().includes(lower),
|
|
146
|
+
);
|
|
147
|
+
}
|
|
148
|
+
this._activeIndex = -1;
|
|
149
|
+
this._renderList();
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
_renderList() {
|
|
153
|
+
this._list.innerHTML = '';
|
|
154
|
+
|
|
155
|
+
if (!this._filtered.length) {
|
|
156
|
+
const li = document.createElement('li');
|
|
157
|
+
li.className = 'item empty';
|
|
158
|
+
li.textContent = 'No matches';
|
|
159
|
+
li.setAttribute('part', 'item empty');
|
|
160
|
+
this._list.append(li);
|
|
161
|
+
return;
|
|
25
162
|
}
|
|
26
163
|
|
|
27
|
-
this.
|
|
164
|
+
this._filtered.forEach((item, index) => {
|
|
165
|
+
const li = document.createElement('li');
|
|
166
|
+
li.className = 'item';
|
|
167
|
+
li.textContent = item;
|
|
168
|
+
li.role = 'option';
|
|
169
|
+
li.id = `ac-opt-${index}`;
|
|
170
|
+
li.setAttribute('part', 'item');
|
|
171
|
+
li.addEventListener('mousedown', (event) => {
|
|
172
|
+
event.preventDefault();
|
|
173
|
+
this._select(item);
|
|
174
|
+
});
|
|
175
|
+
this._list.append(li);
|
|
176
|
+
});
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
_moveActive(step) {
|
|
180
|
+
if (!this._filtered.length) {
|
|
181
|
+
return;
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
const next = this._activeIndex + step;
|
|
185
|
+
if (next < 0) {
|
|
186
|
+
this._activeIndex = this._filtered.length - 1;
|
|
187
|
+
} else if (next >= this._filtered.length) {
|
|
188
|
+
this._activeIndex = 0;
|
|
189
|
+
} else {
|
|
190
|
+
this._activeIndex = next;
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
const items = [...this.shadowRoot.querySelectorAll('.item:not(.empty)')];
|
|
194
|
+
items.forEach((itemEl, idx) => {
|
|
195
|
+
itemEl.classList.toggle('active', idx === this._activeIndex);
|
|
196
|
+
});
|
|
197
|
+
|
|
198
|
+
const activeEl = items[this._activeIndex];
|
|
199
|
+
if (activeEl) {
|
|
200
|
+
this._input.setAttribute('aria-activedescendant', activeEl.id);
|
|
201
|
+
activeEl.scrollIntoView({ block: 'nearest' });
|
|
202
|
+
}
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
_select(value) {
|
|
206
|
+
this._input.value = value;
|
|
207
|
+
this._closeList();
|
|
208
|
+
this.dispatchEvent(
|
|
209
|
+
new CustomEvent('bf-select', {
|
|
210
|
+
bubbles: true,
|
|
211
|
+
composed: true,
|
|
212
|
+
detail: { value },
|
|
213
|
+
}),
|
|
214
|
+
);
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
_openList() {
|
|
218
|
+
this._listOpen = true;
|
|
219
|
+
this._input.setAttribute('aria-expanded', 'true');
|
|
220
|
+
this._list.hidden = false;
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
_closeList() {
|
|
224
|
+
this._listOpen = false;
|
|
225
|
+
this._activeIndex = -1;
|
|
226
|
+
this._input.setAttribute('aria-expanded', 'false');
|
|
227
|
+
this._input.removeAttribute('aria-activedescendant');
|
|
228
|
+
this._list.hidden = true;
|
|
229
|
+
this.shadowRoot.querySelectorAll('.item.active').forEach((itemEl) => {
|
|
230
|
+
itemEl.classList.remove('active');
|
|
231
|
+
});
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
_onDocumentClick(event) {
|
|
235
|
+
if (!this.contains(event.target) && !this.shadowRoot.contains(event.target)) {
|
|
236
|
+
this._closeList();
|
|
237
|
+
}
|
|
28
238
|
}
|
|
29
239
|
}
|
|
30
240
|
|
|
@@ -1,30 +1,142 @@
|
|
|
1
1
|
:host {
|
|
2
2
|
--bf-avatar-font: var(--bf-theme-font-family, inherit);
|
|
3
|
-
--bf-avatar-
|
|
3
|
+
--bf-avatar-size-sm: 2rem;
|
|
4
|
+
--bf-avatar-size-md: 2.5rem;
|
|
5
|
+
--bf-avatar-size-lg: 3rem;
|
|
6
|
+
--bf-avatar-size-xl: 3.75rem;
|
|
7
|
+
--bf-avatar-radius: 999px;
|
|
4
8
|
--bf-avatar-border-width: var(--bf-theme-border-width, 1px);
|
|
5
9
|
--bf-avatar-border-style: var(--bf-theme-border-style, solid);
|
|
6
|
-
--bf-avatar-border-color: var(
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
--bf-avatar-
|
|
11
|
-
|
|
12
|
-
var(--bf-theme-
|
|
13
|
-
|
|
14
|
-
|
|
10
|
+
--bf-avatar-border-color: var(
|
|
11
|
+
--bf-theme-avatar-border-color,
|
|
12
|
+
var(--bf-theme-border-1, #d1d5db)
|
|
13
|
+
);
|
|
14
|
+
--bf-avatar-bg: var(
|
|
15
|
+
--bf-theme-avatar-bg,
|
|
16
|
+
var(--bf-theme-surface-2, #f3f4f6)
|
|
17
|
+
);
|
|
18
|
+
--bf-avatar-color: var(
|
|
19
|
+
--bf-theme-avatar-color,
|
|
20
|
+
var(--bf-theme-text-1, #111827)
|
|
21
|
+
);
|
|
22
|
+
--bf-avatar-fallback-bg: var(
|
|
23
|
+
--bf-theme-avatar-fallback-bg,
|
|
24
|
+
color-mix(in srgb, var(--bf-avatar-bg) 75%, #94a3b8 25%)
|
|
25
|
+
);
|
|
26
|
+
--bf-avatar-fallback-color: var(
|
|
27
|
+
--bf-theme-avatar-fallback-color,
|
|
28
|
+
var(--bf-avatar-color)
|
|
29
|
+
);
|
|
30
|
+
--bf-avatar-shadow: var(--bf-theme-avatar-shadow, 0 1px 2px rgba(15, 23, 42, 0.08));
|
|
31
|
+
--bf-avatar-status-size: 0.62rem;
|
|
32
|
+
--bf-avatar-status-ring: var(--bf-theme-surface-1, #ffffff);
|
|
33
|
+
--bf-avatar-status-online: #16a34a;
|
|
34
|
+
--bf-avatar-status-away: #f59e0b;
|
|
35
|
+
--bf-avatar-status-busy: #ef4444;
|
|
36
|
+
--bf-avatar-status-offline: #64748b;
|
|
15
37
|
|
|
16
|
-
display: block;
|
|
38
|
+
display: inline-block;
|
|
17
39
|
font: var(--bf-avatar-font);
|
|
18
|
-
color: var(--bf-avatar-color);
|
|
19
40
|
}
|
|
20
41
|
|
|
21
42
|
.root {
|
|
22
|
-
|
|
23
|
-
|
|
43
|
+
position: relative;
|
|
44
|
+
width: var(--bf-avatar-size-md);
|
|
45
|
+
height: var(--bf-avatar-size-md);
|
|
46
|
+
border-radius: var(--bf-avatar-radius);
|
|
24
47
|
border-width: var(--bf-avatar-border-width);
|
|
25
48
|
border-style: var(--bf-avatar-border-style);
|
|
26
49
|
border-color: var(--bf-avatar-border-color);
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
50
|
+
overflow: hidden;
|
|
51
|
+
background: var(--bf-avatar-bg);
|
|
52
|
+
color: var(--bf-avatar-color);
|
|
53
|
+
box-shadow: var(--bf-avatar-shadow);
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
.root.size-sm {
|
|
57
|
+
width: var(--bf-avatar-size-sm);
|
|
58
|
+
height: var(--bf-avatar-size-sm);
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
.root.size-md {
|
|
62
|
+
width: var(--bf-avatar-size-md);
|
|
63
|
+
height: var(--bf-avatar-size-md);
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
.root.size-lg {
|
|
67
|
+
width: var(--bf-avatar-size-lg);
|
|
68
|
+
height: var(--bf-avatar-size-lg);
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
.root.size-xl {
|
|
72
|
+
width: var(--bf-avatar-size-xl);
|
|
73
|
+
height: var(--bf-avatar-size-xl);
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
.image,
|
|
77
|
+
.fallback {
|
|
78
|
+
width: 100%;
|
|
79
|
+
height: 100%;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
.image {
|
|
83
|
+
display: block;
|
|
84
|
+
object-fit: cover;
|
|
85
|
+
background: var(--bf-avatar-bg);
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
.fallback {
|
|
89
|
+
display: grid;
|
|
90
|
+
place-items: center;
|
|
91
|
+
font-weight: 700;
|
|
92
|
+
letter-spacing: 0.02em;
|
|
93
|
+
background: var(--bf-avatar-fallback-bg);
|
|
94
|
+
color: var(--bf-avatar-fallback-color);
|
|
95
|
+
user-select: none;
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
.root.size-sm .fallback {
|
|
99
|
+
font-size: 0.7rem;
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
.root.size-md .fallback {
|
|
103
|
+
font-size: 0.85rem;
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
.root.size-lg .fallback {
|
|
107
|
+
font-size: 1rem;
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
.root.size-xl .fallback {
|
|
111
|
+
font-size: 1.15rem;
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
.hidden {
|
|
115
|
+
display: none;
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
.status {
|
|
119
|
+
position: absolute;
|
|
120
|
+
right: 0.02rem;
|
|
121
|
+
bottom: 0.02rem;
|
|
122
|
+
width: var(--bf-avatar-status-size);
|
|
123
|
+
height: var(--bf-avatar-status-size);
|
|
124
|
+
border-radius: 999px;
|
|
125
|
+
border: 2px solid var(--bf-avatar-status-ring);
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
.status.online {
|
|
129
|
+
background: var(--bf-avatar-status-online);
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
.status.away {
|
|
133
|
+
background: var(--bf-avatar-status-away);
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
.status.busy {
|
|
137
|
+
background: var(--bf-avatar-status-busy);
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
.status.offline {
|
|
141
|
+
background: var(--bf-avatar-status-offline);
|
|
30
142
|
}
|
|
@@ -1,4 +1,8 @@
|
|
|
1
1
|
class BfAvatar extends HTMLElement {
|
|
2
|
+
static get observedAttributes() {
|
|
3
|
+
return ['name', 'src', 'alt', 'initials', 'size', 'status'];
|
|
4
|
+
}
|
|
5
|
+
|
|
2
6
|
constructor() {
|
|
3
7
|
super();
|
|
4
8
|
this.attachShadow({ mode: 'open' });
|
|
@@ -9,22 +13,55 @@ class BfAvatar extends HTMLElement {
|
|
|
9
13
|
return;
|
|
10
14
|
}
|
|
11
15
|
this._initialized = true;
|
|
16
|
+
this._render();
|
|
17
|
+
}
|
|
12
18
|
|
|
19
|
+
attributeChangedCallback() {
|
|
20
|
+
if (!this._initialized) {
|
|
21
|
+
return;
|
|
22
|
+
}
|
|
23
|
+
this._render();
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
_render() {
|
|
13
27
|
const cssUrl = new URL('./avatar.css', import.meta.url);
|
|
14
|
-
const
|
|
15
|
-
|
|
16
|
-
|
|
28
|
+
const name = this.getAttribute('name') || '';
|
|
29
|
+
const src = this.getAttribute('src') || '';
|
|
30
|
+
const alt = this.getAttribute('alt') || name || 'Avatar';
|
|
31
|
+
const explicitInitials = this.getAttribute('initials');
|
|
32
|
+
const initials = explicitInitials || this._buildInitials(name);
|
|
33
|
+
const size = this.getAttribute('size') || 'md';
|
|
34
|
+
const status = this.getAttribute('status') || '';
|
|
35
|
+
const showStatus = status && ['online', 'away', 'busy', 'offline'].includes(status);
|
|
17
36
|
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
37
|
+
this.shadowRoot.innerHTML = `
|
|
38
|
+
<link rel="stylesheet" href="${cssUrl.href}" />
|
|
39
|
+
<div class="root size-${size}" part="root">
|
|
40
|
+
${src ? `<img class="image" part="image" src="${src}" alt="${alt.replace(/"/g, '"')}" />` : ''}
|
|
41
|
+
<div class="fallback ${src ? 'hidden' : ''}" part="fallback" aria-label="${alt.replace(/"/g, '"')}">${initials || '?'}</div>
|
|
42
|
+
${showStatus ? `<span class="status ${status}" part="status"></span>` : ''}
|
|
43
|
+
</div>
|
|
44
|
+
`;
|
|
22
45
|
|
|
23
|
-
if (
|
|
24
|
-
|
|
46
|
+
if (src) {
|
|
47
|
+
const image = this.shadowRoot.querySelector('.image');
|
|
48
|
+
const fallback = this.shadowRoot.querySelector('.fallback');
|
|
49
|
+
image.addEventListener('error', () => {
|
|
50
|
+
image.classList.add('hidden');
|
|
51
|
+
fallback.classList.remove('hidden');
|
|
52
|
+
});
|
|
25
53
|
}
|
|
54
|
+
}
|
|
26
55
|
|
|
27
|
-
|
|
56
|
+
_buildInitials(name) {
|
|
57
|
+
if (!name.trim()) {
|
|
58
|
+
return '';
|
|
59
|
+
}
|
|
60
|
+
const parts = name
|
|
61
|
+
.trim()
|
|
62
|
+
.split(/\s+/)
|
|
63
|
+
.slice(0, 2);
|
|
64
|
+
return parts.map((part) => part.charAt(0).toUpperCase()).join('');
|
|
28
65
|
}
|
|
29
66
|
}
|
|
30
67
|
|