forked from royalfig/fluid-typography-calculator
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.js
161 lines (143 loc) · 4.63 KB
/
index.js
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
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
const sample = document.querySelector(".sample");
const viewportRadioBtns = document.querySelectorAll('input[name="viewport"]');
const inputs = document.querySelectorAll(".input");
const viewport = document.querySelector("#viewport");
const viewportTitle = document.querySelector("#viewport-title");
const calcInput = document.querySelector("#calc");
const slider = document.querySelector("#slider");
const computed = document.querySelector("#computed");
const viewportIcon = document.querySelectorAll(".viewport-icon");
slider.addEventListener("input", updateWidth);
const inputsArray = () =>
Array.from(inputs).map((el) => {
const parsed = parseEl(el);
el.addEventListener("blur", updateEls);
return parsed;
});
function unitConversion(obj) {
const { number, unit } = obj;
if (unit === "rem") {
const rem = number + "rem";
const px = number * 16 + "px";
return { rem, px };
}
const px = number + "px";
const rem = number / 16 + "rem";
return { rem, px };
}
function parseValue(str) {
const num = +str.match(/[\d.]+/) && +str.match(/[\d.]+/)[0];
const unit = str.match(/[^.\d]+/) && str.match(/[^.\d]+/)[0].toLowerCase();
return { num, unit };
}
function errorHandler(el) {
const element = getEl(el);
const helper = element.closest(".field").nextElementSibling;
if (!element.validity.valid) {
element.classList.add("is-danger");
helper.textContent = "Inputs needs to be a number followed by px or rem.";
helper.classList.add("is-danger");
return true;
} else {
element.classList.remove("is-danger");
helper.textContent = "";
return false;
}
}
function updateEls(el) {
const error = errorHandler(el);
if (error) {
console.log(error);
}
parseEl(el);
calculateClamp();
}
function getEl(el) {
return el.target ? el.target : el;
}
function parseEl(el) {
const element = getEl(el);
const id = element.id;
const value = element.value;
const number = parseValue(value).num;
const unit = parseValue(value).unit;
const px = unitConversion({ number, unit }).px;
const rem = unitConversion({ number, unit }).rem;
const parsed = { el, id, value, number, unit, px, rem };
updateConversion(parsed);
return parsed;
}
function updateConversion(obj) {
const targetEl = document.querySelector(`[data-unit=${obj.id}]`);
if (targetEl) {
const text = obj.unit !== "rem" ? obj.rem : obj.px;
targetEl.textContent = text;
}
}
function convertToRem(obj) {
return obj.unit === "rem" ? obj.number : obj.number / 16;
}
function calculateClamp() {
console.log("ran");
const updatedValues = inputsArray();
const minFs = convertToRem(updatedValues.find((obj) => obj.id === "min-fs"));
const maxFs = convertToRem(updatedValues.find((obj) => obj.id === "max-fs"));
const minVw = convertToRem(updatedValues.find((obj) => obj.id === "min-vw"));
const maxVw = convertToRem(updatedValues.find((obj) => obj.id === "max-vw"));
const factor = (1 / (maxVw - minVw)) * (maxFs - minFs);
const calcRem = minFs - minVw * factor;
const calcVw = 100 * factor;
const calc = `${calcRem}rem + ${calcVw}vw`;
const min = Math.min(minFs, maxFs);
const max = Math.max(minFs, maxFs);
const clamp = `font-size: ${min}rem;
font-size: clamp(${min}rem, ${calc}, ${max}rem);`;
const css = clamp;
calcInput.textContent = css;
// Enter the simulation
const simulatedVwNum = slider.value;
viewport.textContent = `${simulatedVwNum}px`;
viewportTitle.textContent = `${simulatedVwNum}px`;
const simulatedCalcVw = simulatedVwNum * factor;
const simulatedCalc = `calc(${calcRem}em + ${simulatedCalcVw}px)`;
const simulatedClamp = `font-size: ${min}em;
font-size: clamp(${min}em, ${simulatedCalc}, ${max}em);`;
const simulatedFontSize = +(calcRem * 16 + simulatedCalcVw).toFixed(4);
const clampedSimulatedFontSize = calcInternalClamp(
simulatedFontSize,
min * 16,
max * 16
);
computed.textContent = `${clampedSimulatedFontSize}px`;
sample.setAttribute("style", simulatedClamp);
}
function determineViewport(num) {
switch (true) {
case num <= 768:
return "mobile";
case num <= 1023:
return "tablet";
case num <= 1407:
return "laptop";
default:
return "desktop";
}
}
function updateWidth(e = slider) {
const val = e.target?.value || e.value;
const viewport = determineViewport(val);
viewportIcon.forEach((el) => {
if (el.classList.contains(viewport)) {
el.classList.remove("is-hidden");
} else {
el.classList.add("is-hidden");
}
});
calculateClamp();
}
function calcInternalClamp(val, min, max) {
return val <= max && val >= min ? val : val > max ? max : min;
}
inputsArray();
calculateClamp();
updateWidth();