@massimo-cassandro/minimo 0.1.3 → 0.1.5
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/package.json +4 -6
- package/src/components/auto-datatable/auto-datatable.css +355 -0
- package/src/components/auto-datatable/auto-datatable.js +154 -0
- package/src/components/init.js +7 -0
- package/src/components/modal-alert/flash-alerts.js +32 -0
- package/src/components/modal-alert/modal-alert.css +47 -0
- package/src/components/modal-alert/modal-alert.js +85 -0
- package/src/components/modal-content/modal-content.js +152 -0
- package/src/components/modal-content/modal-content.module.css +142 -0
- package/src/components/recaptcha/recaptcha.css +74 -0
- package/src/components/sf-macro/sf-macro.css +104 -0
- package/src/components/sf-macro/sf-macro.js +44 -0
- package/src/components/vanilla-cookie-consent/cookie-consent.css +20 -0
- package/src/components/vanilla-cookie-consent/cookie-consent.js +76 -0
- package/src/components/vanilla-cookie-consent/render-cookie-list.js +32 -0
- package/src/components/vanilla-cookie-consent/src/cookie-list.js +85 -0
- package/src/components/vanilla-cookie-consent/src/it-translation.js +38 -0
- package/src/components/vanilla-cookie-consent/src/run-analytics.js +54 -0
- package/src/components/vanilla-cookie-consent/src/run-recaptcha.js +71 -0
- package/src/css/alerts.css +95 -0
- package/src/css/anchors.css +40 -0
- package/src/css/buttons/btn-close.css +41 -0
- package/src/css/buttons/buttons.css +188 -0
- package/src/css/buttons/status-buttons.css +107 -0
- package/src/css/container.css +15 -0
- package/src/css/data-formats.css +30 -0
- package/src/css/details.css +19 -0
- package/src/css/dialog-content.css +72 -0
- package/src/css/flex.css +30 -0
- package/src/css/forms/form-edit-info.css +17 -0
- package/src/css/forms/forms.css +315 -0
- package/src/css/grid.css +63 -0
- package/src/css/headings.css +34 -0
- package/src/css/icons.css +114 -0
- package/src/css/inner-nav.css +11 -0
- package/src/css/layout-base.css +11 -0
- package/src/css/overlay.css +11 -0
- package/src/css/reset.css +59 -0
- package/src/css/spinners/TODO spinner-circle.css +54 -0
- package/src/css/spinners/TODO spinner-three-dots.css +14 -0
- package/src/css/spinners/spinner-circle-basic.css +15 -0
- package/src/css/spinners/spinner-trailing-dots.css +51 -0
- package/src/css/spinners/spinner-wrapper.css +20 -0
- package/src/css/svg.css +3 -0
- package/src/css/table.css +66 -0
- package/src/css/text.css +114 -0
- package/src/css/utilities.css +128 -0
- package/src/custom-media-default.css +16 -0
- package/src/custom-properties-default.css +278 -0
- package/src/index-template.css +49 -0
- package/src/index.js +0 -0
- package/src/js/alert-autoclose.js +9 -0
- package/src/js/create-blurhash-canvas.js +47 -0
- package/src/js/dialog-content.js +82 -0
- package/src/js/dismiss-alerts.js +6 -0
- package/src/js/forms.js +36 -0
- package/src/js/img-viewer-dom-builder.js +196 -0
- package/src/js/inner-nav.js +11 -0
- package/src/js/overlay.js +12 -0
- package/src/js/print-icon.js +51 -0
- package/src/js/spinner.js +30 -0
|
@@ -0,0 +1,196 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* minimo: img-viewer-dom-builder
|
|
3
|
+
* Produce un array da utilizzare con domBuilder per generare un tag `picture`
|
|
4
|
+
*
|
|
5
|
+
* @returns Array
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
// TODO implementare getImgInfo (da ada)
|
|
9
|
+
// TODO detect se amabiente local per uso avif
|
|
10
|
+
|
|
11
|
+
/*
|
|
12
|
+
img_viewer_dom_builder({
|
|
13
|
+
viewer_url: 'iviewer',
|
|
14
|
+
sources: [
|
|
15
|
+
{
|
|
16
|
+
src: 'xxx',
|
|
17
|
+
crop: null, // 'x,y,w,h',
|
|
18
|
+
w: 100,
|
|
19
|
+
h: 100,
|
|
20
|
+
imgWidths: [100, 200, 300],
|
|
21
|
+
mq: null, // '(max-width: xxx)'
|
|
22
|
+
},
|
|
23
|
+
...
|
|
24
|
+
],
|
|
25
|
+
imgClassName: null,
|
|
26
|
+
// condition: true | false,
|
|
27
|
+
add2XWidths: false, // true,
|
|
28
|
+
alt_text: '',
|
|
29
|
+
useAvif: false | true,
|
|
30
|
+
// loadingMode: null | 'lazy' | 'eager'
|
|
31
|
+
})
|
|
32
|
+
*/
|
|
33
|
+
export function img_viewer_dom_builder({
|
|
34
|
+
/** url del viewer */
|
|
35
|
+
viewer_url,
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
array dei sorgenti
|
|
39
|
+
deve contenere almeno un oggetto configurato come `source_default`
|
|
40
|
+
l'ultimo oggetto della lista è quello sulla base el quale si costruirà il tag img
|
|
41
|
+
In mancanza di uno degli elementi richiesti (fatta eccezione per `mq` e `crop`),
|
|
42
|
+
il sorgente viene ignorato
|
|
43
|
+
*/
|
|
44
|
+
sources = [], // array dei sorgenti
|
|
45
|
+
|
|
46
|
+
/** se false, il markup non viene generato */
|
|
47
|
+
condition = true,
|
|
48
|
+
|
|
49
|
+
/** se true, alle dimensioni fornite, viene aggiunta la versione 2x */
|
|
50
|
+
add2XWidths = false,
|
|
51
|
+
|
|
52
|
+
/** testo alt */
|
|
53
|
+
alt_text = null,
|
|
54
|
+
|
|
55
|
+
/** classe del tag img */
|
|
56
|
+
imgClassName = null,
|
|
57
|
+
|
|
58
|
+
/** se true viene usato anche il formato avif, oltre a webp e jpg */
|
|
59
|
+
useAvif = false,
|
|
60
|
+
|
|
61
|
+
/** valore dell'attributo loading: eager o lazy */
|
|
62
|
+
loadingMode = null
|
|
63
|
+
}) {
|
|
64
|
+
|
|
65
|
+
if(!viewer_url || !condition) {
|
|
66
|
+
return null;
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
const originalSourcesLength = (sources?? []).length,
|
|
70
|
+
source_default = {
|
|
71
|
+
id: null, // id per immagini da db
|
|
72
|
+
src: null, // percorso dalla root dell'immagine sorgente
|
|
73
|
+
crop: null, // eventuali coordinate per il crop dell'immagine sorgente
|
|
74
|
+
w: null, // larghezza e altezza dell'immagine da restituire
|
|
75
|
+
h: null,
|
|
76
|
+
imgWidths: [], // dimensioni dell'immagine
|
|
77
|
+
mq: null // media query, necessaria solo per gli oggetti prima dell'ultimo
|
|
78
|
+
};
|
|
79
|
+
|
|
80
|
+
sources = sources.map(item => {
|
|
81
|
+
item = {...source_default, ...item?? {}};
|
|
82
|
+
|
|
83
|
+
if((!item.src && !item.id) || !item.w || !item.h || !item.imgWidths?.length) {
|
|
84
|
+
return null;
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
return item;
|
|
88
|
+
|
|
89
|
+
}).filter(result => result != null);
|
|
90
|
+
|
|
91
|
+
if(!sources.length || sources.length !== originalSourcesLength) {
|
|
92
|
+
// eslint-disable-next-line no-console
|
|
93
|
+
console.error('`sources` array error');
|
|
94
|
+
return [];
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
// array per gli elementi <source>
|
|
98
|
+
const sourceTagsArray = [],
|
|
99
|
+
srcset_builder = (format, srcsetsArray) => {
|
|
100
|
+
return srcsetsArray.map(src => `${src[0]}${format? `&f=${format}` : ''} ${src[1]}`).join(',');
|
|
101
|
+
},
|
|
102
|
+
formats = (useAvif? ['avif','webp'] : ['webp']);
|
|
103
|
+
|
|
104
|
+
// sono definite fuori dal ciclo forEach perché saranno utilizzate anche dal tag img finale
|
|
105
|
+
let srcsetsArray, sizes;
|
|
106
|
+
|
|
107
|
+
// costruzione array `sourceTagsArray`
|
|
108
|
+
sources.forEach((item, sourceIdx) => {
|
|
109
|
+
|
|
110
|
+
item.imgWidths = item.imgWidths.map(w => add2XWidths? [w, w * 2] : w).flat().toSorted((a,b) => b - a);
|
|
111
|
+
|
|
112
|
+
const isLastSource = sourceIdx === sources.length - 1,
|
|
113
|
+
baseSrc = viewer_url + (item.src? `?src=${encodeURIComponent(item.src)}&` : `/${item.id}?`);
|
|
114
|
+
|
|
115
|
+
srcsetsArray = item.imgWidths.map(w =>{
|
|
116
|
+
const bbHeight = w / (item.w / item.h);
|
|
117
|
+
return [`${baseSrc}bb=${w}x${bbHeight}`, `${w}w`];
|
|
118
|
+
});
|
|
119
|
+
|
|
120
|
+
sizes = item.imgWidths.map((w,idx) => {
|
|
121
|
+
|
|
122
|
+
// la prima dimensione va ignorata (sarà utilizzata nell'iterazione successiva)
|
|
123
|
+
if(idx === 0) {
|
|
124
|
+
return null;
|
|
125
|
+
|
|
126
|
+
} else {
|
|
127
|
+
// elemento `sizes` diverso dal primo e dall'ultimo
|
|
128
|
+
const middle_size_mq = `(${w}px < width <= ${item.imgWidths.at(idx-1)}px) ${item.imgWidths.at(idx-1)}px`;
|
|
129
|
+
|
|
130
|
+
// primo elemento dell'attributo sizes
|
|
131
|
+
if(idx === 1) {
|
|
132
|
+
return `(width > ${w}px) ${item.imgWidths.at(0)}px`;
|
|
133
|
+
|
|
134
|
+
// ultimo
|
|
135
|
+
} else if(idx === item.imgWidths.length - 1) {
|
|
136
|
+
return [
|
|
137
|
+
middle_size_mq,
|
|
138
|
+
`${w}px`
|
|
139
|
+
];
|
|
140
|
+
|
|
141
|
+
// tutti gli altri
|
|
142
|
+
} else {
|
|
143
|
+
return middle_size_mq;
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
}).filter(s => s != null).flat().join(','); // rimuove elemento null, flat e crea stringa sizes
|
|
147
|
+
|
|
148
|
+
// il formato `null` è riferito a jpeg, che non è necessario indicare esplicitamente
|
|
149
|
+
// ed è utilizzato solo nel caso di presenza di media query per creare il tag source per il formato jpeg,
|
|
150
|
+
// non va inoltre considerato nell'ultimo elemento di source
|
|
151
|
+
// visto che, in questo caso, il formato jpeg viene gestito nel tag `img`
|
|
152
|
+
[...formats, ...(item.mq && !isLastSource? [null] : [])].forEach(format => {
|
|
153
|
+
sourceTagsArray.push({
|
|
154
|
+
tag: 'source',
|
|
155
|
+
attrs: {
|
|
156
|
+
type: format? `image/${format}` : null,
|
|
157
|
+
srcset: srcset_builder(format, srcsetsArray),
|
|
158
|
+
sizes: sizes,
|
|
159
|
+
media: item.mq,
|
|
160
|
+
|
|
161
|
+
// width e height sono necessarie solo in caso di presenza della media query
|
|
162
|
+
// presupponendo che in questi casi l'immagine abbia proporzioni diverse
|
|
163
|
+
// da quelle dell'immagine del tag `img`.
|
|
164
|
+
// In assenza di media query, vengono "ereditate" quelle indicate in `img`
|
|
165
|
+
width: item.mq? item.w : null,
|
|
166
|
+
height: item.mq? item.h : null,
|
|
167
|
+
}
|
|
168
|
+
});
|
|
169
|
+
});
|
|
170
|
+
|
|
171
|
+
});
|
|
172
|
+
|
|
173
|
+
const img_source_obj = sources.at(-1);
|
|
174
|
+
|
|
175
|
+
return {
|
|
176
|
+
tag: 'picture',
|
|
177
|
+
condition: condition,
|
|
178
|
+
children: [
|
|
179
|
+
...sourceTagsArray,
|
|
180
|
+
{
|
|
181
|
+
tag: 'img',
|
|
182
|
+
className: imgClassName,
|
|
183
|
+
attrs: {
|
|
184
|
+
alt: alt_text,
|
|
185
|
+
width: img_source_obj.w,
|
|
186
|
+
height: img_source_obj.h,
|
|
187
|
+
srcset: srcset_builder(null, srcsetsArray),
|
|
188
|
+
sizes: sizes,
|
|
189
|
+
src: srcsetsArray.at(-1)[0],
|
|
190
|
+
loading: loadingMode,
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
]
|
|
194
|
+
};
|
|
195
|
+
|
|
196
|
+
}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
// navigazione scheda / elenco
|
|
2
|
+
let nav_primary = document.querySelector('.inner-nav-main');
|
|
3
|
+
if (nav_primary) {
|
|
4
|
+
// nav_primary.classList.add('text-right', 'd-print-none');
|
|
5
|
+
let nav = nav_primary.innerHTML;
|
|
6
|
+
|
|
7
|
+
document.querySelectorAll('.inner-nav').forEach(item => {
|
|
8
|
+
item.innerHTML = nav;
|
|
9
|
+
// item.classList.add('text-right', 'd-print-none');
|
|
10
|
+
});
|
|
11
|
+
}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
// import './overlay.css'; incorporato nel css principale
|
|
2
|
+
|
|
3
|
+
export function overlay(context = document.body, scroll_lock = false) {
|
|
4
|
+
context.insertAdjacentHTML('beforeend', '<div class="overlay"/>');
|
|
5
|
+
if(scroll_lock) {
|
|
6
|
+
document.body.classList.add('overflow-hidden');
|
|
7
|
+
}
|
|
8
|
+
}
|
|
9
|
+
export function removeOverlay(context = document.body) {
|
|
10
|
+
context.querySelector(':scope > .overlay')?.remove();
|
|
11
|
+
document.body.classList.remove('overflow-hidden');
|
|
12
|
+
}
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import { domBuilder } from '@massimo-cassandro/js-utilities/index.js';
|
|
2
|
+
|
|
3
|
+
const icon_file = document.body.dataset.icons;
|
|
4
|
+
|
|
5
|
+
export function printIcon({iconId, className='icon', title= null, output = 'html'}) {
|
|
6
|
+
|
|
7
|
+
if(!['html', 'element', 'builderArray'].includes(output)) {
|
|
8
|
+
// eslint-disable-next-line no-console
|
|
9
|
+
console.error(`'printIcon': parametro output non corretto (${output}). `+
|
|
10
|
+
'Deve essere `html`, `element` o `builderArray`');
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
// accetta anche un solo argomento stringa (= iconId)
|
|
14
|
+
if(arguments.length === 1 && typeof arguments[0] === 'string') {
|
|
15
|
+
iconId = arguments[0];
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
|
|
19
|
+
const iconArray = [
|
|
20
|
+
{
|
|
21
|
+
tag: 'svg',
|
|
22
|
+
className: className,
|
|
23
|
+
children: [
|
|
24
|
+
{
|
|
25
|
+
tag: 'title',
|
|
26
|
+
content: title,
|
|
27
|
+
condition: title != null
|
|
28
|
+
},
|
|
29
|
+
{
|
|
30
|
+
tag: 'use',
|
|
31
|
+
attrs: {
|
|
32
|
+
href: `${icon_file}#${iconId}`
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
]
|
|
36
|
+
}
|
|
37
|
+
];
|
|
38
|
+
|
|
39
|
+
if(output === 'builderArray') {
|
|
40
|
+
return iconArray;
|
|
41
|
+
} else {
|
|
42
|
+
|
|
43
|
+
const iconElement = domBuilder(iconArray);
|
|
44
|
+
if(output === 'element') {
|
|
45
|
+
return iconElement;
|
|
46
|
+
} else {
|
|
47
|
+
return iconElement.outerHTML;
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
}
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
// import './spinner.css'; // incorporato nel css principale
|
|
2
|
+
|
|
3
|
+
/*
|
|
4
|
+
uso
|
|
5
|
+
|
|
6
|
+
import { spinner } from '@minimo/spinner.js';
|
|
7
|
+
|
|
8
|
+
spinner({container: xxxx, withOverlay: false|true });
|
|
9
|
+
|
|
10
|
+
*/
|
|
11
|
+
|
|
12
|
+
// TODO rifare con domBuilder
|
|
13
|
+
|
|
14
|
+
export function spinner({container, withOverlay = false }) {
|
|
15
|
+
const spinnerMarkup =
|
|
16
|
+
`<div class="${withOverlay? 'spinner-overlay' : 'spinner-wrapper'}"><div class="spinner">Loading...</div></div>`;
|
|
17
|
+
|
|
18
|
+
|
|
19
|
+
if(container != null) {
|
|
20
|
+
container.insertAdjacentHTML('beforeend', spinnerMarkup);
|
|
21
|
+
} else {
|
|
22
|
+
return spinnerMarkup;
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
export function removeSpinner(container) {
|
|
27
|
+
container.querySelectorAll('.spinner-overlay, .spinner-wrapper')
|
|
28
|
+
.forEach(item => item.remove());
|
|
29
|
+
|
|
30
|
+
}
|