-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathdxb-slider.js
More file actions
104 lines (85 loc) · 3.21 KB
/
dxb-slider.js
File metadata and controls
104 lines (85 loc) · 3.21 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
// dxb-slider.js
(function() {
function createSliderStructure(rangeInput) {
// Create container and wrapper
const container = document.createElement('div');
container.className = 'dxb-slider-container';
const wrapper = document.createElement('div');
wrapper.className = 'dxb-slider-wrapper';
const track = document.createElement('div');
track.className = 'dxb-slider-track';
// Add slider class
rangeInput.classList.add('dxb-slider');
// Restructure DOM
rangeInput.parentNode.insertBefore(container, rangeInput);
container.appendChild(wrapper);
wrapper.appendChild(track);
track.appendChild(rangeInput);
return wrapper;
}
function initDXBSliders() {
document.querySelectorAll('[data-dxb-slider]:not([data-dxb-initialized])').forEach(rangeInput => {
const wrapper = createSliderStructure(rangeInput);
// Create number input programmatically
const numberInput = document.createElement('input');
numberInput.type = 'number';
numberInput.className = 'dxb-slider-value';
numberInput.setAttribute('tabindex', '-1');
numberInput.setAttribute('pattern', '[0-9]*');
numberInput.setAttribute('step', rangeInput.step);
const step = parseFloat(rangeInput.step);
if (step && step % 1 !== 0) {
numberInput.setAttribute('inputmode', 'decimal');
} else {
numberInput.setAttribute('inputmode', 'numeric');
}
wrapper.appendChild(numberInput);
function updateValue() {
const val = rangeInput.value;
const min = rangeInput.min;
const max = rangeInput.max;
const percent = (val - min) / (max - min) * 100;
rangeInput.style.setProperty('--value-percent', `${percent}%`);
numberInput.value = val;
numberInput.min = min;
numberInput.max = max;
rangeInput.setAttribute('aria-valuenow', val);
}
rangeInput.addEventListener('input', updateValue);
numberInput.addEventListener('input', () => {
rangeInput.value = numberInput.value;
updateValue();
rangeInput.dispatchEvent(new Event('input', { bubbles: true }));
});
numberInput.addEventListener('change', () => {
rangeInput.dispatchEvent(new Event('change', { bubbles: true }));
});
// Set initial ARIA attributes
rangeInput.setAttribute('aria-valuemin', rangeInput.min);
rangeInput.setAttribute('aria-valuemax', rangeInput.max);
updateValue();
// Mark as initialized
rangeInput.setAttribute('data-dxb-initialized', 'true');
});
}
initDXBSliders();
const observer = new MutationObserver(mutations => {
let shouldInit = false;
for (const mutation of mutations) {
if (mutation.type === 'childList') {
for (const node of mutation.addedNodes) {
if (node.nodeType === Node.ELEMENT_NODE &&
(node.matches('[data-dxb-slider]') || node.querySelector('[data-dxb-slider]'))) {
shouldInit = true;
break;
}
}
if (shouldInit) break;
}
}
if (shouldInit) {
initDXBSliders();
}
});
observer.observe(document.body, { childList: true, subtree: true });
})();