@operato/popup 0.2.35 → 0.2.42
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/CHANGELOG.md +45 -0
- package/demo/index.html +2 -2
- package/demo/ox-popup-list.html +2 -2
- package/demo/ox-popup-menu.html +2 -2
- package/demo/ox-popup.html +2 -2
- package/dist/src/ox-popup-list.d.ts +21 -9
- package/dist/src/ox-popup-list.js +114 -61
- package/dist/src/ox-popup-list.js.map +1 -1
- package/dist/src/ox-popup-menu.d.ts +2 -2
- package/dist/src/ox-popup-menu.js +21 -17
- package/dist/src/ox-popup-menu.js.map +1 -1
- package/dist/src/ox-popup-menuitem.d.ts +3 -3
- package/dist/src/ox-popup-menuitem.js +11 -11
- package/dist/src/ox-popup-menuitem.js.map +1 -1
- package/dist/src/ox-popup.d.ts +9 -5
- package/dist/src/ox-popup.js +25 -17
- package/dist/src/ox-popup.js.map +1 -1
- package/dist/test/ox-popup-menu.test.js +2 -2
- package/dist/test/ox-popup-menu.test.js.map +1 -1
- package/dist/test/ox-popup.test.js +1 -1
- package/dist/test/ox-popup.test.js.map +1 -1
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +3 -2
- package/src/ox-popup-list.ts +127 -64
- package/src/ox-popup-menu.ts +21 -17
- package/src/ox-popup-menuitem.ts +11 -11
- package/src/ox-popup.ts +45 -25
- package/test/ox-popup-menu.test.ts +7 -8
- package/test/ox-popup.test.ts +6 -7
package/CHANGELOG.md
CHANGED
|
@@ -3,6 +3,51 @@
|
|
|
3
3
|
All notable changes to this project will be documented in this file.
|
|
4
4
|
See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
|
|
5
5
|
|
|
6
|
+
### [0.2.42](https://github.com/hatiolab/operato/compare/v0.2.41...v0.2.42) (2021-12-07)
|
|
7
|
+
|
|
8
|
+
**Note:** Version bump only for package @operato/popup
|
|
9
|
+
|
|
10
|
+
|
|
11
|
+
|
|
12
|
+
|
|
13
|
+
|
|
14
|
+
### [0.2.41](https://github.com/hatiolab/operato/compare/v0.2.40...v0.2.41) (2021-12-07)
|
|
15
|
+
|
|
16
|
+
|
|
17
|
+
### :bug: Bug Fix
|
|
18
|
+
|
|
19
|
+
* popup and input style ([11c5b3e](https://github.com/hatiolab/operato/commit/11c5b3e4bf3f08d18673ada585e50a69b9c88b5b))
|
|
20
|
+
* popup style ([43ee008](https://github.com/hatiolab/operato/commit/43ee008afbbd817bb002d2e3c420044f939c9415))
|
|
21
|
+
|
|
22
|
+
|
|
23
|
+
|
|
24
|
+
### [0.2.38](https://github.com/hatiolab/operato/compare/v0.2.37...v0.2.38) (2021-12-07)
|
|
25
|
+
|
|
26
|
+
**Note:** Version bump only for package @operato/popup
|
|
27
|
+
|
|
28
|
+
|
|
29
|
+
|
|
30
|
+
|
|
31
|
+
|
|
32
|
+
### [0.2.36](https://github.com/hatiolab/operato/compare/v0.2.35...v0.2.36) (2021-12-06)
|
|
33
|
+
|
|
34
|
+
|
|
35
|
+
### :bug: Bug Fix
|
|
36
|
+
|
|
37
|
+
* apply class name prefix Ox for operato/popup ([d528848](https://github.com/hatiolab/operato/commit/d5288486ea0e1bec84fff3c5b6bac696eb701e5d))
|
|
38
|
+
* operato/input to support formfield ([21b4d46](https://github.com/hatiolab/operato/commit/21b4d46d750ba833c571b5caeed27eebd8aa03ca))
|
|
39
|
+
* operato/input to support formfield ([544bf92](https://github.com/hatiolab/operato/commit/544bf928946d878aa967f33b20efdf90eb8e82f5))
|
|
40
|
+
* operato/ox-popup-list rewritten ([f78b3e9](https://github.com/hatiolab/operato/commit/f78b3e9623faac5ea653442ced653173d630d9e6))
|
|
41
|
+
* operato/popup, input sample ([116b037](https://github.com/hatiolab/operato/commit/116b0371446cb24d8863c2bdea876b97f6e11dfd))
|
|
42
|
+
* operato/popup, input sample ([31fb99f](https://github.com/hatiolab/operato/commit/31fb99fc4e37a26c38826dd3554f92ded0632e82))
|
|
43
|
+
|
|
44
|
+
|
|
45
|
+
### :rocket: New Features
|
|
46
|
+
|
|
47
|
+
* adding filter options for data-grist ([3620e8b](https://github.com/hatiolab/operato/commit/3620e8b5c39e7e66f8e7ab39b2ae7200f7f7afb5))
|
|
48
|
+
|
|
49
|
+
|
|
50
|
+
|
|
6
51
|
### [0.2.35](https://github.com/hatiolab/operato/compare/v0.2.34...v0.2.35) (2021-12-03)
|
|
7
52
|
|
|
8
53
|
**Note:** Version bump only for package @operato/popup
|
package/demo/index.html
CHANGED
|
@@ -28,12 +28,12 @@
|
|
|
28
28
|
import '@material/mwc-icon'
|
|
29
29
|
import '@operato/input'
|
|
30
30
|
|
|
31
|
-
import {
|
|
31
|
+
import { OxPopupMenu } from '../dist/src/ox-popup-menu.js'
|
|
32
32
|
|
|
33
33
|
const parent = document.querySelector('#demo')
|
|
34
34
|
|
|
35
35
|
parent.addEventListener('click', function (e) {
|
|
36
|
-
|
|
36
|
+
OxPopupMenu.open({
|
|
37
37
|
template: html`
|
|
38
38
|
<ox-popup-menuitem
|
|
39
39
|
label="article - click me"
|
package/demo/ox-popup-list.html
CHANGED
|
@@ -27,7 +27,7 @@
|
|
|
27
27
|
import '@material/mwc-icon'
|
|
28
28
|
import '@operato/input'
|
|
29
29
|
|
|
30
|
-
import {
|
|
30
|
+
import { OxPopupList } from '../dist/src/ox-popup-list.js'
|
|
31
31
|
|
|
32
32
|
const parent = document.querySelector('#demo')
|
|
33
33
|
|
|
@@ -43,7 +43,7 @@
|
|
|
43
43
|
html`
|
|
44
44
|
<span>clike anywhere in this box to ox-select.</span>
|
|
45
45
|
|
|
46
|
-
<ox-popup-list id="popup-list"
|
|
46
|
+
<ox-popup-list id="popup-list" @select=${e => console.log('select', e.target)} multiple>
|
|
47
47
|
<div option>Plain Text</div>
|
|
48
48
|
|
|
49
49
|
<div option>
|
package/demo/ox-popup-menu.html
CHANGED
|
@@ -28,12 +28,12 @@
|
|
|
28
28
|
import '@material/mwc-icon'
|
|
29
29
|
import '@operato/input'
|
|
30
30
|
|
|
31
|
-
import {
|
|
31
|
+
import { OxPopupMenu } from '../dist/src/ox-popup-menu.js'
|
|
32
32
|
|
|
33
33
|
const parent = document.querySelector('#demo')
|
|
34
34
|
|
|
35
35
|
parent.addEventListener('click', function (e) {
|
|
36
|
-
|
|
36
|
+
OxPopupMenu.open({
|
|
37
37
|
template: html`
|
|
38
38
|
<ox-popup-menuitem
|
|
39
39
|
label="article - click me"
|
package/demo/ox-popup.html
CHANGED
|
@@ -22,11 +22,11 @@
|
|
|
22
22
|
<script type="module">
|
|
23
23
|
import { html, render } from 'lit'
|
|
24
24
|
import '../dist/src/ox-popup.js'
|
|
25
|
-
import {
|
|
25
|
+
import { OxPopup } from '../dist/src/ox-popup.js'
|
|
26
26
|
|
|
27
27
|
const parent = document.querySelector('#demo')
|
|
28
28
|
parent.addEventListener('click', function (e) {
|
|
29
|
-
|
|
29
|
+
OxPopup.open({
|
|
30
30
|
template: html`<img
|
|
31
31
|
src=""
|
|
32
32
|
/>`,
|
|
@@ -1,24 +1,36 @@
|
|
|
1
1
|
import { PropertyValues } from 'lit';
|
|
2
|
-
import {
|
|
3
|
-
export declare class
|
|
2
|
+
import { OxPopup } from './ox-popup';
|
|
3
|
+
export declare class OxPopupList extends OxPopup {
|
|
4
4
|
static styles: import("lit").CSSResult[];
|
|
5
|
-
|
|
5
|
+
multiple: boolean;
|
|
6
|
+
attrSelected?: string;
|
|
7
|
+
activeIndex?: number;
|
|
6
8
|
render(): import("lit-html").TemplateResult<1>;
|
|
7
9
|
protected _onkeydown: (e: KeyboardEvent) => void;
|
|
8
10
|
protected _onfocusout: (e: FocusEvent) => void;
|
|
9
11
|
protected _onclick: (e: MouseEvent) => void;
|
|
10
12
|
updated(changes: PropertyValues<this>): void;
|
|
11
|
-
select(
|
|
12
|
-
setActive(active: number | Element | null): void;
|
|
13
|
+
select(): Promise<void>;
|
|
14
|
+
setActive(active: number | Element | null, withSelect?: boolean): void;
|
|
15
|
+
open(params: {
|
|
16
|
+
left?: number;
|
|
17
|
+
top?: number;
|
|
18
|
+
right?: number;
|
|
19
|
+
bottom?: number;
|
|
20
|
+
silent?: boolean;
|
|
21
|
+
}): void;
|
|
22
|
+
close(): void;
|
|
13
23
|
/**
|
|
14
|
-
* Open
|
|
24
|
+
* Open OxPopup
|
|
15
25
|
*
|
|
16
26
|
* @param {PopupOpenOptions}
|
|
17
27
|
*/
|
|
18
|
-
static open({ template, top, left, parent }: {
|
|
28
|
+
static open({ template, top, left, right, bottom, parent }: {
|
|
19
29
|
template: unknown;
|
|
20
|
-
top
|
|
21
|
-
left
|
|
30
|
+
top?: number;
|
|
31
|
+
left?: number;
|
|
32
|
+
right?: number;
|
|
33
|
+
bottom?: number;
|
|
22
34
|
parent?: Element | null;
|
|
23
35
|
}): void;
|
|
24
36
|
}
|
|
@@ -1,20 +1,25 @@
|
|
|
1
1
|
import { __decorate } from "tslib";
|
|
2
2
|
import { css, html } from 'lit';
|
|
3
|
-
import { customElement, property } from 'lit/decorators.js';
|
|
4
|
-
import { Popup } from './ox-popup';
|
|
5
3
|
import { render } from 'lit-html';
|
|
6
|
-
|
|
7
|
-
|
|
4
|
+
import { customElement, property, state } from 'lit/decorators.js';
|
|
5
|
+
import { OxPopup } from './ox-popup';
|
|
6
|
+
function guaranteeFocus(element) {
|
|
7
|
+
// 1. 옵션 엘리먼트의 하위 첫번째 focusible 엘리먼트에 focus 기회를 준다.
|
|
8
|
+
const focusible = element.querySelector('button, [href], input, select, textarea, [tabindex]:not([tabindex="-1"])');
|
|
9
|
+
if (focusible) {
|
|
10
|
+
;
|
|
11
|
+
focusible.focus();
|
|
12
|
+
return;
|
|
13
|
+
}
|
|
14
|
+
// 2. 자신을 포함해서 가장 가까운 부모에게 focus 기회를 준다.
|
|
8
15
|
const closest = element.closest('button, [href], input, select, textarea, [tabindex]:not([tabindex="-1"])');
|
|
9
16
|
closest === null || closest === void 0 ? void 0 : closest.focus();
|
|
10
|
-
return closest;
|
|
11
17
|
}
|
|
12
|
-
let
|
|
18
|
+
let OxPopupList = class OxPopupList extends OxPopup {
|
|
13
19
|
constructor() {
|
|
14
20
|
super(...arguments);
|
|
15
|
-
this.
|
|
21
|
+
this.multiple = false;
|
|
16
22
|
this._onkeydown = function (e) {
|
|
17
|
-
var _a;
|
|
18
23
|
e.stopPropagation();
|
|
19
24
|
switch (e.key) {
|
|
20
25
|
case 'Esc': // for IE/Edge
|
|
@@ -34,29 +39,19 @@ let PopupList = class PopupList extends Popup {
|
|
|
34
39
|
this.activeIndex++;
|
|
35
40
|
break;
|
|
36
41
|
case 'Enter':
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
}
|
|
42
|
+
case ' ':
|
|
43
|
+
case 'Spacebar': // for old firefox
|
|
44
|
+
this.setActive(this.activeIndex, true);
|
|
45
|
+
this.select();
|
|
42
46
|
break;
|
|
43
47
|
}
|
|
44
48
|
}.bind(this);
|
|
45
49
|
this._onfocusout = function (e) {
|
|
46
|
-
const target = e.target;
|
|
47
50
|
const to = e.relatedTarget;
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
this.setActive(this.activeIndex);
|
|
53
|
-
}
|
|
54
|
-
else {
|
|
55
|
-
if (!this.contains(to)) {
|
|
56
|
-
/* 분명히 내 범위가 아닌 엘리먼트로 포커스가 옮겨졌다면, ox-popup-list는 닫혀야 한다. */
|
|
57
|
-
// @ts-ignore for debug
|
|
58
|
-
!window.POPUP_DEBUG && this.close();
|
|
59
|
-
}
|
|
51
|
+
if (!this.contains(to)) {
|
|
52
|
+
/* 분명히 내 범위가 아닌 엘리먼트로 포커스가 옮겨졌다면, ox-popup-list는 닫혀야 한다. */
|
|
53
|
+
// @ts-ignore for debug
|
|
54
|
+
!window.POPUP_DEBUG && this.close();
|
|
60
55
|
}
|
|
61
56
|
}.bind(this);
|
|
62
57
|
this._onclick = function (e) {
|
|
@@ -64,59 +59,105 @@ let PopupList = class PopupList extends Popup {
|
|
|
64
59
|
e.stopPropagation();
|
|
65
60
|
const option = (_a = e.target) === null || _a === void 0 ? void 0 : _a.closest('[option]');
|
|
66
61
|
if (option) {
|
|
67
|
-
this.setActive(option);
|
|
68
|
-
this.select(
|
|
62
|
+
this.setActive(option, true);
|
|
63
|
+
this.select();
|
|
69
64
|
}
|
|
70
65
|
}.bind(this);
|
|
71
66
|
}
|
|
72
67
|
render() {
|
|
73
|
-
return html `
|
|
68
|
+
return html `
|
|
69
|
+
<slot
|
|
70
|
+
@change=${(e) => {
|
|
71
|
+
e.stopPropagation();
|
|
72
|
+
}}
|
|
73
|
+
>
|
|
74
|
+
</slot>
|
|
75
|
+
`;
|
|
74
76
|
}
|
|
75
77
|
updated(changes) {
|
|
76
78
|
if (changes.has('activeIndex')) {
|
|
77
|
-
this.setActive(this.activeIndex);
|
|
79
|
+
this.activeIndex !== undefined && this.setActive(this.activeIndex);
|
|
78
80
|
}
|
|
79
81
|
}
|
|
80
|
-
select(
|
|
81
|
-
|
|
82
|
-
|
|
82
|
+
async select() {
|
|
83
|
+
await this.updateComplete;
|
|
84
|
+
const options = Array.from(this.querySelectorAll(':scope > [option]'));
|
|
85
|
+
const selected = options
|
|
86
|
+
.filter(option => option.hasAttribute('value') && option.hasAttribute(this.attrSelected || 'selected'))
|
|
87
|
+
.map(option => option.getAttribute('value'));
|
|
88
|
+
this.dispatchEvent(new CustomEvent('select', {
|
|
89
|
+
bubbles: true,
|
|
90
|
+
composed: true,
|
|
91
|
+
detail: this.multiple ? selected : selected[0]
|
|
92
|
+
}));
|
|
93
|
+
const option = options[this.activeIndex];
|
|
94
|
+
if (!option.hasAttribute('alive-on-select') && !this.hasAttribute('multiple')) {
|
|
83
95
|
this.close();
|
|
84
96
|
}
|
|
85
97
|
}
|
|
86
|
-
setActive(active) {
|
|
87
|
-
const options = Array.from(this.querySelectorAll('
|
|
88
|
-
|
|
98
|
+
setActive(active, withSelect) {
|
|
99
|
+
const options = Array.from(this.querySelectorAll('[option]'));
|
|
100
|
+
if (active instanceof Element) {
|
|
101
|
+
const index = options.findIndex(option => option === active);
|
|
102
|
+
this.setActive(index === -1 ? 0 : index, withSelect);
|
|
103
|
+
return;
|
|
104
|
+
}
|
|
105
|
+
options.forEach(async (option, index) => {
|
|
89
106
|
if (typeof active === 'number' && index === (active + options.length) % options.length) {
|
|
90
107
|
option.setAttribute('active', '');
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
108
|
+
if (withSelect && !this.attrSelected) {
|
|
109
|
+
this.multiple ? option.toggleAttribute('selected') : option.setAttribute('selected', '');
|
|
110
|
+
}
|
|
111
|
+
guaranteeFocus(option);
|
|
95
112
|
this.activeIndex = index;
|
|
96
113
|
}
|
|
97
114
|
else {
|
|
98
115
|
option.removeAttribute('active');
|
|
116
|
+
!this.attrSelected && !this.multiple && withSelect && option.removeAttribute('selected');
|
|
99
117
|
}
|
|
100
118
|
});
|
|
101
119
|
}
|
|
120
|
+
open(params) {
|
|
121
|
+
super.open(params);
|
|
122
|
+
if (this.activeIndex === undefined) {
|
|
123
|
+
this.activeIndex = 0;
|
|
124
|
+
}
|
|
125
|
+
else {
|
|
126
|
+
this.setActive(this.activeIndex);
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
close() {
|
|
130
|
+
if (this.hasAttribute('active')) {
|
|
131
|
+
this.dispatchEvent(new CustomEvent('close', {
|
|
132
|
+
bubbles: true,
|
|
133
|
+
composed: true
|
|
134
|
+
}));
|
|
135
|
+
}
|
|
136
|
+
super.close();
|
|
137
|
+
}
|
|
102
138
|
/**
|
|
103
|
-
* Open
|
|
139
|
+
* Open OxPopup
|
|
104
140
|
*
|
|
105
141
|
* @param {PopupOpenOptions}
|
|
106
142
|
*/
|
|
107
|
-
static open({ template, top, left, parent }) {
|
|
143
|
+
static open({ template, top, left, right, bottom, parent }) {
|
|
108
144
|
const owner = parent || document.body;
|
|
109
145
|
const target = document.createElement('ox-popup-list');
|
|
110
146
|
render(template, target);
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
147
|
+
if (left !== undefined)
|
|
148
|
+
target.style.left = `${left}px`;
|
|
149
|
+
if (top !== undefined)
|
|
150
|
+
target.style.top = `${top}px`;
|
|
151
|
+
if (right !== undefined)
|
|
152
|
+
target.style.right = `${right}px`;
|
|
153
|
+
if (bottom !== undefined)
|
|
154
|
+
target.style.bottom = `${bottom}px`;
|
|
114
155
|
target._parent = owner;
|
|
115
156
|
owner.appendChild(target);
|
|
116
157
|
}
|
|
117
158
|
};
|
|
118
|
-
|
|
119
|
-
...
|
|
159
|
+
OxPopupList.styles = [
|
|
160
|
+
...OxPopup.styles,
|
|
120
161
|
css `
|
|
121
162
|
:host {
|
|
122
163
|
display: none;
|
|
@@ -125,8 +166,7 @@ PopupList.styles = [
|
|
|
125
166
|
background-color: var(--theme-white-color, #fff);
|
|
126
167
|
z-index: 100;
|
|
127
168
|
box-shadow: 2px 3px 10px 5px rgba(0, 0, 0, 0.15);
|
|
128
|
-
padding:
|
|
129
|
-
|
|
169
|
+
padding: 4px 0;
|
|
130
170
|
color: var(--theme-primary-text-color, #3c3938);
|
|
131
171
|
font: normal 14px 'Roboto', sans-serif;
|
|
132
172
|
}
|
|
@@ -139,22 +179,29 @@ PopupList.styles = [
|
|
|
139
179
|
outline: none;
|
|
140
180
|
}
|
|
141
181
|
|
|
182
|
+
::slotted([option]) {
|
|
183
|
+
white-space: nowrap;
|
|
184
|
+
border-left: 3px solid transparent;
|
|
185
|
+
}
|
|
186
|
+
|
|
142
187
|
::slotted(*) {
|
|
143
|
-
|
|
144
|
-
|
|
188
|
+
padding: 5px 10px;
|
|
189
|
+
border-bottom: 1px solid rgba(0, 0, 0, 0.05);
|
|
190
|
+
cursor: pointer;
|
|
145
191
|
}
|
|
146
192
|
|
|
147
193
|
::slotted(*:focus) {
|
|
148
194
|
outline: none;
|
|
149
195
|
}
|
|
150
196
|
|
|
151
|
-
::slotted([option])
|
|
152
|
-
|
|
197
|
+
::slotted([option][active]),
|
|
198
|
+
::slotted([option]:hover) {
|
|
199
|
+
background-color: #f6f6f6;
|
|
153
200
|
}
|
|
154
201
|
|
|
155
|
-
::slotted([option][
|
|
156
|
-
|
|
157
|
-
|
|
202
|
+
::slotted([option][selected]) {
|
|
203
|
+
border-left: 3px solid var(--primary-color, #38a25b);
|
|
204
|
+
font-weight: bold;
|
|
158
205
|
}
|
|
159
206
|
|
|
160
207
|
::slotted([separator]) {
|
|
@@ -169,10 +216,16 @@ PopupList.styles = [
|
|
|
169
216
|
`
|
|
170
217
|
];
|
|
171
218
|
__decorate([
|
|
172
|
-
property({ type:
|
|
173
|
-
],
|
|
174
|
-
|
|
219
|
+
property({ type: Boolean })
|
|
220
|
+
], OxPopupList.prototype, "multiple", void 0);
|
|
221
|
+
__decorate([
|
|
222
|
+
property({ type: String, attribute: 'attr-selected' })
|
|
223
|
+
], OxPopupList.prototype, "attrSelected", void 0);
|
|
224
|
+
__decorate([
|
|
225
|
+
state()
|
|
226
|
+
], OxPopupList.prototype, "activeIndex", void 0);
|
|
227
|
+
OxPopupList = __decorate([
|
|
175
228
|
customElement('ox-popup-list')
|
|
176
|
-
],
|
|
177
|
-
export {
|
|
229
|
+
], OxPopupList);
|
|
230
|
+
export { OxPopupList };
|
|
178
231
|
//# sourceMappingURL=ox-popup-list.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"ox-popup-list.js","sourceRoot":"","sources":["../../src/ox-popup-list.ts"],"names":[],"mappings":"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","sourcesContent":["import { PropertyValues, css, html } from 'lit'\nimport { customElement, property } from 'lit/decorators.js'\n\nimport { Popup } from './ox-popup'\nimport { render } from 'lit-html'\n\nfunction focusClosest(element: HTMLElement) {\n /* Find the closest focusable element. */\n const closest = element.closest(\n 'button, [href], input, select, textarea, [tabindex]:not([tabindex=\"-1\"])'\n ) as HTMLElement\n\n closest?.focus()\n\n return closest\n}\n\n@customElement('ox-popup-list')\nexport class PopupList extends Popup {\n static styles = [\n ...Popup.styles,\n css`\n :host {\n display: none;\n flex-direction: column;\n align-items: stretch;\n background-color: var(--theme-white-color, #fff);\n z-index: 100;\n box-shadow: 2px 3px 10px 5px rgba(0, 0, 0, 0.15);\n padding: 5px;\n\n color: var(--theme-primary-text-color, #3c3938);\n font: normal 14px 'Roboto', sans-serif;\n }\n\n :host([active]) {\n display: flex;\n }\n\n :host(*:focus) {\n outline: none;\n }\n\n ::slotted(*) {\n margin: 1px 0;\n padding: 4px 2px 3px 2px;\n }\n\n ::slotted(*:focus) {\n outline: none;\n }\n\n ::slotted([option]) {\n border-radius: 5px;\n }\n\n ::slotted([option][active]) {\n background-color: rgba(20, 154, 155, 0.1);\n cursor: pointer;\n }\n\n ::slotted([separator]) {\n height: 1px;\n width: 100%;\n padding: 0;\n background-color: rgba(0, 0, 0, 0.15);\n }\n ::slotted([option] > input) {\n border: var(--border-dark-color, 1px solid rgba(0, 0, 0, 0.3));\n }\n `\n ]\n\n @property({ type: Number }) activeIndex: number = 0\n\n render() {\n return html` <slot> </slot> `\n }\n\n protected _onkeydown: (e: KeyboardEvent) => void = function (this: PopupList, e: KeyboardEvent) {\n e.stopPropagation()\n\n switch (e.key) {\n case 'Esc': // for IE/Edge\n case 'Escape':\n this.close()\n break\n\n case 'Left': // for IE/Edge\n case 'ArrowLeft':\n case 'Up': // for IE/Edge\n case 'ArrowUp':\n this.activeIndex--\n break\n\n case 'Right': // for IE/Edge\n case 'ArrowRight':\n case 'Down': // for IE/Edge\n case 'ArrowDown':\n this.activeIndex++\n break\n\n case 'Enter':\n e.stopPropagation()\n var option = (e.target as HTMLElement)?.closest('[option]')\n if (option) {\n this.select(option)\n }\n break\n }\n }.bind(this)\n\n protected _onfocusout: (e: FocusEvent) => void = function (this: PopupList, e: FocusEvent) {\n const target = e.target as HTMLElement\n const to = e.relatedTarget as HTMLElement\n const from = target.closest('ox-popup-list')\n\n if (!to && from !== this) {\n e.stopPropagation()\n\n /* \"하위의 ox-popup-list 엘리먼트가 포커스를 잃었지만, 그 포커스를 받은 엘리먼트가 없다.\"는 의미는 그 서브메뉴가 클로즈된 것을 의미한다. */\n this.setActive(this.activeIndex)\n } else {\n if (!this.contains(to)) {\n /* 분명히 내 범위가 아닌 엘리먼트로 포커스가 옮겨졌다면, ox-popup-list는 닫혀야 한다. */\n // @ts-ignore for debug\n !window.POPUP_DEBUG && this.close()\n }\n }\n }.bind(this)\n\n protected _onclick: (e: MouseEvent) => void = function (this: PopupList, e: MouseEvent) {\n e.stopPropagation()\n\n const option = (e.target as HTMLElement)?.closest('[option]')\n if (option) {\n this.setActive(option)\n this.select(option)\n }\n }.bind(this)\n\n updated(changes: PropertyValues<this>) {\n if (changes.has('activeIndex')) {\n this.setActive(this.activeIndex)\n }\n }\n\n select(option: Element) {\n option.dispatchEvent(new CustomEvent('selected', { bubbles: true, composed: true, detail: this }))\n\n if (!option.hasAttribute('alive-on-select') && !this.hasAttribute('alive-on-select')) {\n this.close()\n }\n }\n\n setActive(active: number | Element | null) {\n const options = Array.from(this.querySelectorAll(':scope > [option]'))\n\n options.map(async (option, index) => {\n if (typeof active === 'number' && index === (active + options.length) % options.length) {\n option.setAttribute('active', '')\n focusClosest(option as HTMLElement)\n\n this.activeIndex = index\n } else if (active === option) {\n this.activeIndex = index\n } else {\n option.removeAttribute('active')\n }\n })\n }\n\n /**\n * Open Popup\n *\n * @param {PopupOpenOptions}\n */\n static open({\n template,\n top,\n left,\n parent\n }: {\n template: unknown\n top: number\n left: number\n parent?: Element | null\n }) {\n const owner = parent || document.body\n const target = document.createElement('ox-popup-list') as PopupList\n render(template, target)\n\n target.style.left = `${left}px`\n target.style.top = `${top}px`\n\n target.setAttribute('active', '')\n\n target._parent = owner\n\n owner.appendChild(target)\n }\n}\n"]}
|
|
1
|
+
{"version":3,"file":"ox-popup-list.js","sourceRoot":"","sources":["../../src/ox-popup-list.ts"],"names":[],"mappings":"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","sourcesContent":["import { css, html, PropertyValues } from 'lit'\nimport { render } from 'lit-html'\nimport { customElement, property, state } from 'lit/decorators.js'\n\nimport { OxPopup } from './ox-popup'\n\nfunction guaranteeFocus(element: HTMLElement) {\n // 1. 옵션 엘리먼트의 하위 첫번째 focusible 엘리먼트에 focus 기회를 준다.\n const focusible = element.querySelector('button, [href], input, select, textarea, [tabindex]:not([tabindex=\"-1\"])')\n\n if (focusible) {\n ;(focusible as HTMLElement).focus()\n return\n }\n\n // 2. 자신을 포함해서 가장 가까운 부모에게 focus 기회를 준다.\n const closest = element.closest(\n 'button, [href], input, select, textarea, [tabindex]:not([tabindex=\"-1\"])'\n ) as HTMLElement\n\n closest?.focus()\n}\n\n@customElement('ox-popup-list')\nexport class OxPopupList extends OxPopup {\n static styles = [\n ...OxPopup.styles,\n css`\n :host {\n display: none;\n flex-direction: column;\n align-items: stretch;\n background-color: var(--theme-white-color, #fff);\n z-index: 100;\n box-shadow: 2px 3px 10px 5px rgba(0, 0, 0, 0.15);\n padding: 4px 0;\n color: var(--theme-primary-text-color, #3c3938);\n font: normal 14px 'Roboto', sans-serif;\n }\n\n :host([active]) {\n display: flex;\n }\n\n :host(*:focus) {\n outline: none;\n }\n\n ::slotted([option]) {\n white-space: nowrap;\n border-left: 3px solid transparent;\n }\n\n ::slotted(*) {\n padding: 5px 10px;\n border-bottom: 1px solid rgba(0, 0, 0, 0.05);\n cursor: pointer;\n }\n\n ::slotted(*:focus) {\n outline: none;\n }\n\n ::slotted([option][active]),\n ::slotted([option]:hover) {\n background-color: #f6f6f6;\n }\n\n ::slotted([option][selected]) {\n border-left: 3px solid var(--primary-color, #38a25b);\n font-weight: bold;\n }\n\n ::slotted([separator]) {\n height: 1px;\n width: 100%;\n padding: 0;\n background-color: rgba(0, 0, 0, 0.15);\n }\n ::slotted([option] > input) {\n border: var(--border-dark-color, 1px solid rgba(0, 0, 0, 0.3));\n }\n `\n ]\n\n @property({ type: Boolean }) multiple: boolean = false\n @property({ type: String, attribute: 'attr-selected' }) attrSelected?: string\n\n @state() activeIndex?: number\n\n render() {\n return html`\n <slot\n @change=${(e: Event) => {\n e.stopPropagation()\n }}\n >\n </slot>\n `\n }\n\n protected _onkeydown: (e: KeyboardEvent) => void = function (this: OxPopupList, e: KeyboardEvent) {\n e.stopPropagation()\n\n switch (e.key) {\n case 'Esc': // for IE/Edge\n case 'Escape':\n this.close()\n break\n\n case 'Left': // for IE/Edge\n case 'ArrowLeft':\n case 'Up': // for IE/Edge\n case 'ArrowUp':\n this.activeIndex!--\n break\n\n case 'Right': // for IE/Edge\n case 'ArrowRight':\n case 'Down': // for IE/Edge\n case 'ArrowDown':\n this.activeIndex!++\n break\n\n case 'Enter':\n case ' ':\n case 'Spacebar': // for old firefox\n this.setActive(this.activeIndex!, true)\n this.select()\n break\n }\n }.bind(this)\n\n protected _onfocusout: (e: FocusEvent) => void = function (this: OxPopupList, e: FocusEvent) {\n const to = e.relatedTarget as HTMLElement\n\n if (!this.contains(to)) {\n /* 분명히 내 범위가 아닌 엘리먼트로 포커스가 옮겨졌다면, ox-popup-list는 닫혀야 한다. */\n // @ts-ignore for debug\n !window.POPUP_DEBUG && this.close()\n }\n }.bind(this)\n\n protected _onclick: (e: MouseEvent) => void = function (this: OxPopupList, e: MouseEvent) {\n e.stopPropagation()\n\n const option = (e.target as HTMLElement)?.closest('[option]')\n if (option) {\n this.setActive(option, true)\n this.select()\n }\n }.bind(this)\n\n updated(changes: PropertyValues<this>) {\n if (changes.has('activeIndex')) {\n this.activeIndex !== undefined && this.setActive(this.activeIndex)\n }\n }\n\n async select() {\n await this.updateComplete\n\n const options = Array.from(this.querySelectorAll(':scope > [option]'))\n\n const selected = options\n .filter(option => option.hasAttribute('value') && option.hasAttribute(this.attrSelected || 'selected'))\n .map(option => option.getAttribute('value'))\n\n this.dispatchEvent(\n new CustomEvent('select', {\n bubbles: true,\n composed: true,\n detail: this.multiple ? selected : selected[0]\n })\n )\n\n const option = options[this.activeIndex!]\n if (!option.hasAttribute('alive-on-select') && !this.hasAttribute('multiple')) {\n this.close()\n }\n }\n\n setActive(active: number | Element | null, withSelect?: boolean) {\n const options = Array.from(this.querySelectorAll('[option]'))\n\n if (active instanceof Element) {\n const index = options.findIndex(option => option === active)\n this.setActive(index === -1 ? 0 : index, withSelect)\n return\n }\n\n options.forEach(async (option, index) => {\n if (typeof active === 'number' && index === (active + options.length) % options.length) {\n option.setAttribute('active', '')\n if (withSelect && !this.attrSelected) {\n this.multiple ? option.toggleAttribute('selected') : option.setAttribute('selected', '')\n }\n\n guaranteeFocus(option as HTMLElement)\n\n this.activeIndex = index\n } else {\n option.removeAttribute('active')\n !this.attrSelected && !this.multiple && withSelect && option.removeAttribute('selected')\n }\n })\n }\n\n override open(params: { left?: number; top?: number; right?: number; bottom?: number; silent?: boolean }) {\n super.open(params)\n\n if (this.activeIndex === undefined) {\n this.activeIndex = 0\n } else {\n this.setActive(this.activeIndex)\n }\n }\n\n override close() {\n if (this.hasAttribute('active')) {\n this.dispatchEvent(\n new CustomEvent('close', {\n bubbles: true,\n composed: true\n })\n )\n }\n\n super.close()\n }\n\n /**\n * Open OxPopup\n *\n * @param {PopupOpenOptions}\n */\n static open({\n template,\n top,\n left,\n right,\n bottom,\n parent\n }: {\n template: unknown\n top?: number\n left?: number\n right?: number\n bottom?: number\n parent?: Element | null\n }) {\n const owner = parent || document.body\n const target = document.createElement('ox-popup-list') as OxPopupList\n render(template, target)\n\n if (left !== undefined) target.style.left = `${left}px`\n if (top !== undefined) target.style.top = `${top}px`\n if (right !== undefined) target.style.right = `${right}px`\n if (bottom !== undefined) target.style.bottom = `${bottom}px`\n\n target._parent = owner\n\n owner.appendChild(target)\n }\n}\n"]}
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { PropertyValues } from 'lit';
|
|
2
|
-
import {
|
|
3
|
-
export declare class
|
|
2
|
+
import { OxPopup } from './ox-popup';
|
|
3
|
+
export declare class OxPopupMenu extends OxPopup {
|
|
4
4
|
static styles: import("lit").CSSResult[];
|
|
5
5
|
activeIndex: number;
|
|
6
6
|
render(): import("lit-html").TemplateResult<1>;
|