frontpack 0.1.0
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.
- checksums.yaml +7 -0
- data/MIT-LICENSE +20 -0
- data/README.md +51 -0
- data/Rakefile +5 -0
- data/app/assets/config/frontpack_manifest.js +1 -0
- data/app/assets/images/logo.svg +141 -0
- data/app/assets/javascripts/frontpack/auto-complete.js +173 -0
- data/app/assets/javascripts/frontpack/base-webcomponent.js +160 -0
- data/app/assets/javascripts/frontpack/expandable-list.js +99 -0
- data/app/assets/javascripts/frontpack/image-preview.js +160 -0
- data/app/assets/javascripts/frontpack/image-uploader.js +34 -0
- data/app/assets/javascripts/frontpack/localized-time.js +17 -0
- data/app/assets/javascripts/frontpack/modal-dialog.js +206 -0
- data/app/assets/javascripts/frontpack/shortcuts.js +63 -0
- data/app/assets/javascripts/frontpack/tab-control.js +77 -0
- data/app/assets/javascripts/frontpack/text-box.js +70 -0
- data/app/assets/javascripts/frontpack/toggle-button.js +73 -0
- data/app/assets/stylesheets/frontpack/000-reset.scss +15 -0
- data/app/assets/stylesheets/frontpack/010-config.scss +148 -0
- data/app/assets/stylesheets/frontpack/020-mixins.scss +218 -0
- data/app/assets/stylesheets/frontpack/030-consts.scss +13 -0
- data/app/assets/stylesheets/frontpack/031-margins.scss +5 -0
- data/app/assets/stylesheets/frontpack/100-layout.scss +113 -0
- data/app/assets/stylesheets/frontpack/101-container.scss +131 -0
- data/app/assets/stylesheets/frontpack/200-design.scss +75 -0
- data/app/assets/stylesheets/frontpack/201-box.scss +6 -0
- data/app/assets/stylesheets/frontpack/202-borders.scss +5 -0
- data/app/assets/stylesheets/frontpack/300-typography.scss +128 -0
- data/app/assets/stylesheets/frontpack/800-components.scss +85 -0
- data/app/assets/stylesheets/frontpack/801-card.scss +53 -0
- data/app/assets/stylesheets/frontpack/802-buttons.scss +58 -0
- data/app/assets/stylesheets/frontpack/803-form-controls.scss +161 -0
- data/app/assets/stylesheets/frontpack/804-table.scss +60 -0
- data/app/assets/stylesheets/frontpack/805-switch-toggle.scss +43 -0
- data/app/assets/stylesheets/frontpack/frontpack.scss +18 -0
- data/app/controllers/concerns/frontpack/searchable.rb +33 -0
- data/app/controllers/frontpack/autocomplete_controller.rb +33 -0
- data/app/helpers/frontpack/application_helper.rb +7 -0
- data/app/helpers/frontpack/form_builder.rb +130 -0
- data/app/models/concerns/frontpack/auto_completable.rb +35 -0
- data/config/importmap.rb +1 -0
- data/config/routes.rb +3 -0
- data/lib/frontpack/button_to_patch.rb +64 -0
- data/lib/frontpack/engine.rb +28 -0
- data/lib/frontpack/extended_model_translations.rb +31 -0
- data/lib/frontpack/form_builder_options.rb +5 -0
- data/lib/frontpack/version.rb +5 -0
- data/lib/frontpack.rb +9 -0
- data/lib/generators/frontpack/install_generator.rb +37 -0
- data/lib/generators/frontpack/locale_generator.rb +11 -0
- data/lib/generators/templates/initializers/customize_form_with_errors.rb +31 -0
- data/lib/generators/templates/locales/frontpack.en.yml +43 -0
- data/lib/generators/templates/views/layouts/_form-errors.html.slim +3 -0
- data/lib/generators/templates/views/layouts/_navigation.html.slim +1 -0
- data/lib/generators/templates/views/layouts/application.html.slim +37 -0
- data/lib/generators/templates/views/layouts/errors.html.slim +19 -0
- data/lib/generators/templates/views/layouts/mailer.html.slim +6 -0
- data/lib/generators/templates/views/layouts/mailer.text.slim +1 -0
- data/lib/tasks/frontpack_tasks.rake +6 -0
- data/lib/templates/rails/file_utils/searchable.rb +30 -0
- data/lib/templates/rails/scaffold_controller/controller.rb.tt +93 -0
- data/lib/templates/slim/scaffold/_form.html.slim.tt +10 -0
- data/lib/templates/slim/scaffold/edit.html.slim.tt +10 -0
- data/lib/templates/slim/scaffold/index.html.slim.tt +20 -0
- data/lib/templates/slim/scaffold/new.html.slim.tt +8 -0
- data/lib/templates/slim/scaffold/partial.html.slim.tt +6 -0
- data/lib/templates/slim/scaffold/show.html.slim.tt +12 -0
- data/lib/templates/test_unit/model/fixtures.yml.tt +18 -0
- data/lib/templates/test_unit/model/unit_test.rb.tt +41 -0
- metadata +143 -0
@@ -0,0 +1,160 @@
|
|
1
|
+
class ImagePreview extends HTMLElement {
|
2
|
+
constructor () {
|
3
|
+
super()
|
4
|
+
const shadowRoot = this.attachShadow({ mode: 'open' })
|
5
|
+
shadowRoot.innerHTML = `
|
6
|
+
<style>
|
7
|
+
:host(*){
|
8
|
+
background-size: contain;
|
9
|
+
background-repeat: no-repeat;
|
10
|
+
background-position: center;
|
11
|
+
}
|
12
|
+
|
13
|
+
:host([status='empty']):before {
|
14
|
+
content: attr(text);
|
15
|
+
}
|
16
|
+
</style>
|
17
|
+
`
|
18
|
+
if (this.hasAttribute('target-input')) {
|
19
|
+
this.input = document.getElementById(this.getAttribute('target-input'))
|
20
|
+
} else {
|
21
|
+
console.log('Input file not defined')
|
22
|
+
return
|
23
|
+
}
|
24
|
+
|
25
|
+
this.setAttribute('status', 'empty')
|
26
|
+
this.source = null
|
27
|
+
}
|
28
|
+
|
29
|
+
set text (value) {
|
30
|
+
this.setAttribute('text', value)
|
31
|
+
}
|
32
|
+
|
33
|
+
get text () {
|
34
|
+
return this.getAttribute('text')
|
35
|
+
}
|
36
|
+
|
37
|
+
set status (value) {
|
38
|
+
this.setAttribute('status', value)
|
39
|
+
this._dispatch('statuschange', value)
|
40
|
+
}
|
41
|
+
|
42
|
+
get status () {
|
43
|
+
return this.getAttribute('status')
|
44
|
+
}
|
45
|
+
|
46
|
+
onClick () {
|
47
|
+
this.input.click()
|
48
|
+
}
|
49
|
+
|
50
|
+
onChange () {
|
51
|
+
this.file = this.input.files[0]
|
52
|
+
this.source = 'fileUpload'
|
53
|
+
this.render()
|
54
|
+
}
|
55
|
+
|
56
|
+
onDragOver (event) {
|
57
|
+
event.preventDefault()
|
58
|
+
this.status = 'dragging'
|
59
|
+
}
|
60
|
+
|
61
|
+
onDragLeave () {
|
62
|
+
if (!this.file) {
|
63
|
+
this.setAttribute('status', 'empty')
|
64
|
+
} else {
|
65
|
+
this.setAttribute('status', 'ready')
|
66
|
+
}
|
67
|
+
}
|
68
|
+
|
69
|
+
onDrop (event) {
|
70
|
+
event.preventDefault()
|
71
|
+
this.file = event.dataTransfer.files[0]
|
72
|
+
this.source = 'dragAndDrop'
|
73
|
+
this.render()
|
74
|
+
}
|
75
|
+
|
76
|
+
onPaste (event) {
|
77
|
+
try {
|
78
|
+
this.file = event.clipboardData.items[1].getAsFile()
|
79
|
+
this.source = 'clipboard'
|
80
|
+
this.render()
|
81
|
+
} catch (e) {
|
82
|
+
this._dispatch('error', e.message)
|
83
|
+
}
|
84
|
+
}
|
85
|
+
|
86
|
+
render () {
|
87
|
+
const validExtensions = ['image/jpeg', 'image/jpg', 'image/png', 'image/gif']
|
88
|
+
if (!this.file || !validExtensions.includes(this.file.type)) {
|
89
|
+
this._dispatch('error', 'Invalid image file.')
|
90
|
+
this._clear()
|
91
|
+
return
|
92
|
+
}
|
93
|
+
|
94
|
+
try {
|
95
|
+
this.status = 'loading'
|
96
|
+
const fileReader = new FileReader()
|
97
|
+
fileReader.addEventListener('load', this.onReaderReady.bind(this))
|
98
|
+
fileReader.addEventListener('error', this.onReaderError.bind(this))
|
99
|
+
fileReader.readAsDataURL(this.file)
|
100
|
+
this.input.files = this.copyFile()
|
101
|
+
} catch (e) {
|
102
|
+
this._clear()
|
103
|
+
this._dispatch('error', e.message)
|
104
|
+
}
|
105
|
+
}
|
106
|
+
|
107
|
+
copyFile () {
|
108
|
+
const dt = new DataTransfer()
|
109
|
+
dt.items.add(new File([this.file.size, this.file.type], this.file.name))
|
110
|
+
return dt.files
|
111
|
+
}
|
112
|
+
|
113
|
+
onReaderReady (event) {
|
114
|
+
const fileURL = event.target.result
|
115
|
+
this.style.backgroundImage = `url(${fileURL})`
|
116
|
+
this.base64Data = fileURL
|
117
|
+
this.setAttribute('status', 'ready')
|
118
|
+
this.status = 'ready'
|
119
|
+
this._dispatch('success', { source: this.source })
|
120
|
+
}
|
121
|
+
|
122
|
+
onReaderError (event) {
|
123
|
+
this._dispatch('error', event.target.error)
|
124
|
+
this.status = 'error'
|
125
|
+
}
|
126
|
+
|
127
|
+
_dispatch (eventName, details) {
|
128
|
+
const event = new CustomEvent(eventName, {
|
129
|
+
cancelable: true,
|
130
|
+
bubbles: true,
|
131
|
+
detail: details
|
132
|
+
})
|
133
|
+
this.dispatchEvent(event)
|
134
|
+
}
|
135
|
+
|
136
|
+
_clear () {
|
137
|
+
this.file = null
|
138
|
+
this.style.backgroundImage = ''
|
139
|
+
this.setAttribute('status', 'empty')
|
140
|
+
this.input.files = []
|
141
|
+
}
|
142
|
+
|
143
|
+
clear () {
|
144
|
+
this._clear()
|
145
|
+
this._dispatch('clear')
|
146
|
+
}
|
147
|
+
|
148
|
+
connectedCallback () {
|
149
|
+
this.addEventListener('click', this.onClick.bind(this))
|
150
|
+
this.addEventListener('dragover', this.onDragOver.bind(this))
|
151
|
+
this.addEventListener('drop', this.onDrop.bind(this))
|
152
|
+
this.addEventListener('dragleave', this.onDragLeave.bind(this))
|
153
|
+
this.input.addEventListener('change', this.onChange.bind(this))
|
154
|
+
if (this.getAttribute('disable-clipboard') !== 'true') {
|
155
|
+
window.addEventListener('paste', this.onPaste.bind(this))
|
156
|
+
}
|
157
|
+
}
|
158
|
+
}
|
159
|
+
|
160
|
+
customElements.define('image-preview', ImagePreview)
|
@@ -0,0 +1,34 @@
|
|
1
|
+
class ImageUploader {
|
2
|
+
onUpdatePreview(event) {
|
3
|
+
this.preview.src = event.target.result;
|
4
|
+
}
|
5
|
+
|
6
|
+
onChange() {
|
7
|
+
if (this.input.files && this.input.files[0]) {
|
8
|
+
let reader = new FileReader();
|
9
|
+
reader.addEventListener('load', this.onUpdatePreview.bind(this));
|
10
|
+
reader.readAsDataURL(this.input.files[0]);
|
11
|
+
}
|
12
|
+
}
|
13
|
+
|
14
|
+
constructor(element) {
|
15
|
+
this.preview = element.querySelector('img');
|
16
|
+
this.input = element.querySelector('input');
|
17
|
+
this.input.addEventListener('change', this.onChange.bind(this));
|
18
|
+
|
19
|
+
window.setTimeout(_ => {
|
20
|
+
let previewContainer = element.querySelector('.preview');
|
21
|
+
let rect = previewContainer.getBoundingClientRect();
|
22
|
+
this.preview.style.width = rect.width + 'px';
|
23
|
+
this.preview.style.height = rect.height + 'px';
|
24
|
+
this.preview.src = ' ';
|
25
|
+
}, 1);
|
26
|
+
}
|
27
|
+
}
|
28
|
+
|
29
|
+
document.addEventListener('turbolinks:load', _ => {
|
30
|
+
let uploaders = document.querySelectorAll('.image-uploader');
|
31
|
+
for (let uploader of uploaders) {
|
32
|
+
new ImageUploader(uploader);
|
33
|
+
}
|
34
|
+
});
|
@@ -0,0 +1,17 @@
|
|
1
|
+
class LocalizedTime extends HTMLElement {
|
2
|
+
constructor() {
|
3
|
+
super();
|
4
|
+
const shadowRoot = this.attachShadow({ mode: 'open' });
|
5
|
+
shadowRoot.innerHTML = `<slot></slot>`;
|
6
|
+
}
|
7
|
+
|
8
|
+
connectedCallback() {
|
9
|
+
this.shadowRoot.innerHTML = this.time.toLocaleString();
|
10
|
+
}
|
11
|
+
|
12
|
+
get time() {
|
13
|
+
let date_time = new Date(this.getAttribute('time'));
|
14
|
+
return date_time;
|
15
|
+
}
|
16
|
+
}
|
17
|
+
customElements.define('localized-time', LocalizedTime);
|
@@ -0,0 +1,206 @@
|
|
1
|
+
class ModalDialog extends HTMLElement {
|
2
|
+
constructor() {
|
3
|
+
super();
|
4
|
+
this.attachShadow({ mode: 'open' } ).innerHTML = `
|
5
|
+
<style>
|
6
|
+
* {
|
7
|
+
box-sizing: border-box;
|
8
|
+
}
|
9
|
+
:host {
|
10
|
+
position: absolute;
|
11
|
+
display: none;
|
12
|
+
flex-flow: column nowrap;
|
13
|
+
align-items: center;
|
14
|
+
justify-content: center;
|
15
|
+
left: 0;
|
16
|
+
top: 0;
|
17
|
+
width: 100%;
|
18
|
+
height: 100%;
|
19
|
+
z-index: 200;
|
20
|
+
background: rgba(128, 128, 128, 0.2);
|
21
|
+
}
|
22
|
+
#dialog {
|
23
|
+
display: grid;
|
24
|
+
grid-template-columns: 100%;
|
25
|
+
grid-template-rows: min-content 1fr min-content;
|
26
|
+
background: #f8f9fa;
|
27
|
+
box-shadow: 1px 1px 5px gray;
|
28
|
+
}
|
29
|
+
#header {
|
30
|
+
display: grid;
|
31
|
+
grid-template-columns: 1fr min-content;
|
32
|
+
grid-template-areas: "title buttons";
|
33
|
+
background: rgba(0, 0, 0, 0.1);
|
34
|
+
border-bottom: rgba(0, 0, 0, 0.2) solid 1px;
|
35
|
+
padding: 0 0.5em;
|
36
|
+
}
|
37
|
+
#title {
|
38
|
+
grid-area: title;
|
39
|
+
margin: 0;
|
40
|
+
padding: 0.25em 0;
|
41
|
+
}
|
42
|
+
#buttons {
|
43
|
+
display: inline-grid;
|
44
|
+
grid-auto-columns: 2em;
|
45
|
+
grid-template-rows: auto;
|
46
|
+
}
|
47
|
+
#title-buttons button {
|
48
|
+
font-family: "Material Icons";
|
49
|
+
font-size: 1.3em;
|
50
|
+
padding: 0;
|
51
|
+
margin: 0;
|
52
|
+
background: transparent;
|
53
|
+
border: none;
|
54
|
+
cursor: pointer;
|
55
|
+
height: 100%;
|
56
|
+
}
|
57
|
+
#title-buttons button:hover {
|
58
|
+
color: red;
|
59
|
+
}
|
60
|
+
#content {
|
61
|
+
position: relative;
|
62
|
+
padding: 0.5em;
|
63
|
+
}
|
64
|
+
#footer {
|
65
|
+
display: grid;
|
66
|
+
grid-template-columns: min-content 1fr min-content;
|
67
|
+
grid-template-rows: min-content;
|
68
|
+
grid-template-areas: "left center right";
|
69
|
+
padding: 0.5em;
|
70
|
+
}
|
71
|
+
.center-buttons {
|
72
|
+
display: flex;
|
73
|
+
flex-flow: row nowrap;
|
74
|
+
justify-content: center;
|
75
|
+
text-align: center;
|
76
|
+
}
|
77
|
+
.standard-button {
|
78
|
+
border: rgba(0, 0, 0, 0.2) 1px solid;
|
79
|
+
background: rgba(0, 0, 0, 0.1);
|
80
|
+
padding: 0.5em 1em;
|
81
|
+
}
|
82
|
+
iframe {
|
83
|
+
width: 100%;
|
84
|
+
height: 100%;
|
85
|
+
border: none;
|
86
|
+
}
|
87
|
+
</style>
|
88
|
+
<div id="dialog">
|
89
|
+
<div id="header">
|
90
|
+
<h3 id="title"></h3>
|
91
|
+
<span id="title-buttons">
|
92
|
+
<button type="button" id="close-button" data-resolve="false" data-data="cancel">close</button>
|
93
|
+
</span>
|
94
|
+
</div>
|
95
|
+
<div id="content">
|
96
|
+
<slot></slot>
|
97
|
+
</div>
|
98
|
+
<div id="footer">
|
99
|
+
<div class="left-buttons"><slot name="left"></slot></div>
|
100
|
+
<div class="center-buttons"><slot name="center"></slot></div>
|
101
|
+
<div class="right-buttons"><slot name="right"></slot></div>
|
102
|
+
</div>
|
103
|
+
</div>
|
104
|
+
`;
|
105
|
+
}
|
106
|
+
|
107
|
+
connectedCallback() {
|
108
|
+
this._title = this.shadowRoot.getElementById('title');
|
109
|
+
this._content = this.shadowRoot.getElementById('content');
|
110
|
+
this._buttons = new Map([
|
111
|
+
['left', this.shadowRoot.getElementById('left')],
|
112
|
+
['center', this.shadowRoot.getElementById('center')],
|
113
|
+
['right', this.shadowRoot.getElementById('right')]
|
114
|
+
]);
|
115
|
+
this._footer = this.shadowRoot.getElementById('footer');
|
116
|
+
this._footer.addEventListener('click', this.onButtonClick.bind(this));
|
117
|
+
this._close_button = this.shadowRoot.getElementById('close-button');
|
118
|
+
this._close_button.addEventListener('click', this.onButtonClick.bind(this));
|
119
|
+
this._title.innerHTML = this.caption;
|
120
|
+
}
|
121
|
+
|
122
|
+
get caption() {
|
123
|
+
return this.getAttribute('caption');
|
124
|
+
}
|
125
|
+
|
126
|
+
set caption(value) {
|
127
|
+
this.setAttribute('caption', value);
|
128
|
+
this._title.innerHTML = value;
|
129
|
+
}
|
130
|
+
|
131
|
+
get closeButton() {
|
132
|
+
return this._close_button.style.display = 'inline';
|
133
|
+
}
|
134
|
+
|
135
|
+
set closeButton(value) {
|
136
|
+
this._close_button.style.display = value ? 'inline' : 'none';
|
137
|
+
}
|
138
|
+
|
139
|
+
set buttons(options) {
|
140
|
+
['left', 'center', 'right'].forEach((position) => {
|
141
|
+
this._buttons.get(position).innerHTML = '';
|
142
|
+
});
|
143
|
+
for (let button of options) {
|
144
|
+
let newButton = document.createElement('button');
|
145
|
+
newButton.innerHTML = button.caption;
|
146
|
+
newButton.dataset.data = button.data;
|
147
|
+
newButton.dataset.resolve = button.resolve;
|
148
|
+
newButton.className = 'standard-button';
|
149
|
+
this._buttons.get(button.position).appendChild(newButton);
|
150
|
+
}
|
151
|
+
}
|
152
|
+
|
153
|
+
onButtonClick(event) {
|
154
|
+
let button = event.target;
|
155
|
+
if (button.tagName.toLowerCase() !== 'button') {
|
156
|
+
return;
|
157
|
+
}
|
158
|
+
|
159
|
+
this.close(button.dataset.resolve === 'true', button.dataset.data);
|
160
|
+
}
|
161
|
+
|
162
|
+
show(options = {}) {
|
163
|
+
return new Promise((resolve, reject) => {
|
164
|
+
if (options.hideCloseButton) {
|
165
|
+
this.closeButton = false;
|
166
|
+
}
|
167
|
+
if (options.width) {
|
168
|
+
this._content.style.width = options.width;
|
169
|
+
}
|
170
|
+
if (options.height) {
|
171
|
+
this._content.style.height = options.height;
|
172
|
+
}
|
173
|
+
this.resolve = resolve;
|
174
|
+
this.reject = reject;
|
175
|
+
this.style.display = 'inline-flex';
|
176
|
+
});
|
177
|
+
}
|
178
|
+
|
179
|
+
dialogBox(caption, text, buttons, options = {}) {
|
180
|
+
this.caption = caption;
|
181
|
+
this._content.innerHTML = `<p>${text}</p>`;
|
182
|
+
this.buttons = buttons;
|
183
|
+
return this.show(options);
|
184
|
+
}
|
185
|
+
|
186
|
+
messageBox(caption, text, options = {}) {
|
187
|
+
return this.dialogBox(caption, text, [{ caption: '✔', data: 'ok', resolve: true, position: 'center' }], options);
|
188
|
+
}
|
189
|
+
|
190
|
+
confirmationBox(caption, text, options = {}) {
|
191
|
+
return this.dialogBox(caption, text, [
|
192
|
+
{ caption: '✔', data: 'yes', resolve: true, position: 'left' },
|
193
|
+
{ caption: '❌', data: 'no', resolve: true, position: 'right' }
|
194
|
+
], options);
|
195
|
+
}
|
196
|
+
|
197
|
+
close(resolve, data = '') {
|
198
|
+
this.style.display = 'none';
|
199
|
+
if (resolve) {
|
200
|
+
this.resolve(data);
|
201
|
+
} else {
|
202
|
+
this.reject(data);
|
203
|
+
}
|
204
|
+
}
|
205
|
+
}
|
206
|
+
customElements.define('modal-dialog', ModalDialog);
|
@@ -0,0 +1,63 @@
|
|
1
|
+
'use strict';
|
2
|
+
|
3
|
+
window.Shortcut = {
|
4
|
+
_registeredActions: new Map(),
|
5
|
+
_registeredShortcuts: new Map(),
|
6
|
+
_actionNameIndex: 0,
|
7
|
+
|
8
|
+
_parseShortcut(shortcut) {
|
9
|
+
if (!/^(?:(ctrl|alt|shift)\+){0,3}[a-z0-9]+$/i.test(shortcut)) {
|
10
|
+
throw new Error('Invalid shortcut combination');
|
11
|
+
}
|
12
|
+
},
|
13
|
+
|
14
|
+
onKeyUp(event) {
|
15
|
+
let { ctrlKey, shiftKey, altKey, code } = event;
|
16
|
+
let tokens = [];
|
17
|
+
if (ctrlKey) { tokens.push('ctrl'); }
|
18
|
+
if (altKey) { tokens.push('alt'); }
|
19
|
+
if (shiftKey) { tokens.push('shift'); }
|
20
|
+
tokens.push(code.toLowerCase().replace(/(key|digit)/, ''));
|
21
|
+
let shortcut = tokens.join('+').toLowerCase();
|
22
|
+
|
23
|
+
if (this._registeredShortcuts.has(shortcut)) {
|
24
|
+
event.preventDefault();
|
25
|
+
event.stopPropagation();
|
26
|
+
this.execute(this._registeredShortcuts.get(shortcut));
|
27
|
+
}
|
28
|
+
},
|
29
|
+
|
30
|
+
nextActionName() {
|
31
|
+
this._actionNameIndex++;
|
32
|
+
return `Action${this._actionNameIndex}`;
|
33
|
+
},
|
34
|
+
|
35
|
+
execute(name) {
|
36
|
+
this._registeredActions.get(name)();
|
37
|
+
},
|
38
|
+
|
39
|
+
/**
|
40
|
+
* Register a named action with an optional shortcut
|
41
|
+
* @param {String} name Action name
|
42
|
+
* @param {CallableFunction} callback Action to be executed
|
43
|
+
* @param {String} shortcut Any shortcut to listen globally
|
44
|
+
*/
|
45
|
+
registerAction(name, callback, shortcut = null) {
|
46
|
+
this._registeredActions.set(name, callback);
|
47
|
+
if (shortcut) {
|
48
|
+
this._parseShortcut(shortcut.toLowerCase());
|
49
|
+
this._registeredShortcuts.set(shortcut.toLowerCase(), name);
|
50
|
+
}
|
51
|
+
},
|
52
|
+
|
53
|
+
start() {
|
54
|
+
document.addEventListener('keydown', this.onKeyUp.bind(this));
|
55
|
+
let elements = document.querySelectorAll('[shortcut]');
|
56
|
+
for (let element of elements) {
|
57
|
+
let actionName = this.nextActionName();
|
58
|
+
this.registerAction(actionName, element.click.bind(element), element.getAttribute('shortcut'));
|
59
|
+
}
|
60
|
+
}
|
61
|
+
};
|
62
|
+
|
63
|
+
export default Shortcut;
|
@@ -0,0 +1,77 @@
|
|
1
|
+
class TabNav extends HTMLElement {
|
2
|
+
constructor() {
|
3
|
+
super();
|
4
|
+
const shadowRoot = this.attachShadow({ mode: 'open' });
|
5
|
+
shadowRoot.innerHTML = `<slot></slot>`;
|
6
|
+
this.currentTab = null;
|
7
|
+
}
|
8
|
+
|
9
|
+
connectedCallback() {
|
10
|
+
this.addEventListener('tabshow', this.onTabClick.bind(this));
|
11
|
+
}
|
12
|
+
|
13
|
+
onTabClick(event) {
|
14
|
+
if (this.currentTab) {
|
15
|
+
this.currentTab.active = false;
|
16
|
+
}
|
17
|
+
this.currentTab = event.detail.tab;
|
18
|
+
this.currentTab.active = true;
|
19
|
+
}
|
20
|
+
}
|
21
|
+
customElements.define('tab-nav', TabNav);
|
22
|
+
|
23
|
+
class TabButton extends HTMLElement {
|
24
|
+
constructor() {
|
25
|
+
super();
|
26
|
+
const shadowRoot = this.attachShadow({ mode: 'open' });
|
27
|
+
shadowRoot.innerHTML = `<slot></slot>`;
|
28
|
+
}
|
29
|
+
|
30
|
+
connectedCallback() {
|
31
|
+
this.addEventListener('click', this.onClick.bind(this));
|
32
|
+
this._target = document.getElementById(this.target);
|
33
|
+
if (this.hasAttribute('active')) {
|
34
|
+
this.active = true;
|
35
|
+
this.dispatchClickEvent();
|
36
|
+
} else {
|
37
|
+
this.active = false;
|
38
|
+
}
|
39
|
+
}
|
40
|
+
|
41
|
+
get target() {
|
42
|
+
return this.getAttribute('target');
|
43
|
+
}
|
44
|
+
|
45
|
+
set target(value) {
|
46
|
+
this.setAttribute('target', value);
|
47
|
+
this._target = document.getElementById(value);
|
48
|
+
}
|
49
|
+
|
50
|
+
get active() {
|
51
|
+
return this.hasAttribute('active');
|
52
|
+
}
|
53
|
+
|
54
|
+
set active(value) {
|
55
|
+
if (value) {
|
56
|
+
this.setAttribute('active', 'true');
|
57
|
+
this._target.style.zIndex = '10';
|
58
|
+
} else {
|
59
|
+
this.removeAttribute('active');
|
60
|
+
this._target.style.zIndex = '0';
|
61
|
+
}
|
62
|
+
}
|
63
|
+
|
64
|
+
onClick() {
|
65
|
+
if (this.active) {
|
66
|
+
return;
|
67
|
+
}
|
68
|
+
|
69
|
+
this.dispatchClickEvent();
|
70
|
+
}
|
71
|
+
|
72
|
+
dispatchClickEvent() {
|
73
|
+
let event = new CustomEvent('tabshow', { cancelable: true, bubbles: true, detail: { tab: this } });
|
74
|
+
this.dispatchEvent(event);
|
75
|
+
}
|
76
|
+
}
|
77
|
+
customElements.define('tab-button', TabButton);
|
@@ -0,0 +1,70 @@
|
|
1
|
+
class TextBox extends HTMLElement {
|
2
|
+
constructor() {
|
3
|
+
super();
|
4
|
+
const shadowRoot = this.attachShadow({ mode: 'open' });
|
5
|
+
shadowRoot.innerHTML = `
|
6
|
+
<style>
|
7
|
+
:host {
|
8
|
+
display: none;
|
9
|
+
grid-template-columns: min-content 1fr min-content;
|
10
|
+
grid-template-rows: auto;
|
11
|
+
align-items: center;
|
12
|
+
align-content: center;
|
13
|
+
grid-gap: .5rem;
|
14
|
+
padding: .5rem;
|
15
|
+
}
|
16
|
+
|
17
|
+
i, button {
|
18
|
+
border: none;
|
19
|
+
background: none;
|
20
|
+
font-family: "Material Icons";
|
21
|
+
font-style: normal;
|
22
|
+
font-size: 1.2rem;
|
23
|
+
color: inherit;
|
24
|
+
}
|
25
|
+
</style>
|
26
|
+
<i></i>
|
27
|
+
<div><slot></slot></div>
|
28
|
+
<button type="button">close</button>
|
29
|
+
`;
|
30
|
+
}
|
31
|
+
|
32
|
+
connectedCallback() {
|
33
|
+
this._icon = this.shadowRoot.querySelector('i');
|
34
|
+
this._icon.innerText = this.icon;
|
35
|
+
this._closeButton = this.shadowRoot.querySelector('button');
|
36
|
+
this._closeButton.addEventListener('click', this.close.bind(this));
|
37
|
+
this.persistent = this.hasAttribute('persistent');
|
38
|
+
|
39
|
+
if (this.persistent && this.hasAttribute('id')) {
|
40
|
+
let key = `${document.location.pathname.substring(1)}#${this.id}`;
|
41
|
+
if (localStorage.getItem(key) === 'true') {
|
42
|
+
return;
|
43
|
+
}
|
44
|
+
}
|
45
|
+
this.open();
|
46
|
+
}
|
47
|
+
|
48
|
+
get icon() {
|
49
|
+
return this.getAttribute('icon');
|
50
|
+
}
|
51
|
+
|
52
|
+
set icon(value) {
|
53
|
+
this.setAttribute('icon', value);
|
54
|
+
this._icon.innerText = value;
|
55
|
+
}
|
56
|
+
|
57
|
+
open() {
|
58
|
+
this.style.display = 'grid';
|
59
|
+
}
|
60
|
+
|
61
|
+
close() {
|
62
|
+
this.style.display = 'none';
|
63
|
+
|
64
|
+
if (this.persistent && this.hasAttribute('id')) {
|
65
|
+
let key = `${document.location.pathname.substring(1)}#${this.id}`;
|
66
|
+
localStorage.setItem(key, 'true');
|
67
|
+
}
|
68
|
+
}
|
69
|
+
}
|
70
|
+
customElements.define('text-box', TextBox);
|
@@ -0,0 +1,73 @@
|
|
1
|
+
class ToggleButton extends HTMLElement {
|
2
|
+
constructor() {
|
3
|
+
super();
|
4
|
+
const shadowRoot = this.attachShadow({mode: 'open'});
|
5
|
+
shadowRoot.innerHTML = '<slot></slot>';
|
6
|
+
}
|
7
|
+
|
8
|
+
connectedCallback() {
|
9
|
+
this.mode = this.getAttribute('mode');
|
10
|
+
|
11
|
+
if (this.mode === 'copy') {
|
12
|
+
this.addEventListener('click', this.copy.bind(this));
|
13
|
+
} else {
|
14
|
+
this.addEventListener('click', this.toggle.bind(this));
|
15
|
+
}
|
16
|
+
}
|
17
|
+
|
18
|
+
target() {
|
19
|
+
if (!this._target) {
|
20
|
+
this._target = document.getElementById(this.getAttribute('target'));
|
21
|
+
}
|
22
|
+
|
23
|
+
return this._target;
|
24
|
+
}
|
25
|
+
|
26
|
+
render() {
|
27
|
+
if (this.state) {
|
28
|
+
this.target().classList.remove(this.hiddenClass)
|
29
|
+
this.target().classList.add(this.visibleClass)
|
30
|
+
} else {
|
31
|
+
this.target().classList.remove(this.visibleClass)
|
32
|
+
this.target().classList.add(this.hiddenClass)
|
33
|
+
}
|
34
|
+
}
|
35
|
+
|
36
|
+
copy () {
|
37
|
+
const target = this.target()
|
38
|
+
|
39
|
+
if (typeof target.value !== 'undefined') {
|
40
|
+
navigator.clipboard.writeText(target.value)
|
41
|
+
} else {
|
42
|
+
navigator.clipboard.writeText(target.innerText)
|
43
|
+
}
|
44
|
+
}
|
45
|
+
|
46
|
+
toggle() {
|
47
|
+
this.state = !this.state;
|
48
|
+
}
|
49
|
+
|
50
|
+
get state() {
|
51
|
+
return this.hasAttribute('on');
|
52
|
+
}
|
53
|
+
|
54
|
+
set state(value) {
|
55
|
+
if (value) {
|
56
|
+
this.setAttribute('on', value);
|
57
|
+
} else {
|
58
|
+
this.removeAttribute('on');
|
59
|
+
}
|
60
|
+
|
61
|
+
this.render();
|
62
|
+
}
|
63
|
+
|
64
|
+
get visibleClass() {
|
65
|
+
return this.getAttribute('visible-class');
|
66
|
+
}
|
67
|
+
|
68
|
+
get hiddenClass() {
|
69
|
+
return this.getAttribute('hidden-class');
|
70
|
+
}
|
71
|
+
}
|
72
|
+
|
73
|
+
customElements.define('toggle-button', ToggleButton);
|