261 lines
8.3 KiB
HTML
261 lines
8.3 KiB
HTML
|
<!--
|
||
|
Calculator widget.
|
||
|
This file should contain all required
|
||
|
CSS, HTML, and JS for it.
|
||
|
-->
|
||
|
|
||
|
<style>
|
||
|
#calc-text {
|
||
|
background: var(--whoogle-dark-page-bg);
|
||
|
padding: 8px;
|
||
|
border-radius: 8px;
|
||
|
text-align: right;
|
||
|
font-family: monospace;
|
||
|
font-size: 16px;
|
||
|
color: var(--whoogle-dark-text);
|
||
|
}
|
||
|
#prev-equation {
|
||
|
text-align: right;
|
||
|
}
|
||
|
.error-border {
|
||
|
border: 1px solid red;
|
||
|
}
|
||
|
|
||
|
#calc-btns {
|
||
|
display: grid;
|
||
|
grid-template-columns: repeat(6, 1fr);
|
||
|
grid-template-rows: repeat(5, 1fr);
|
||
|
gap: 5px;
|
||
|
}
|
||
|
#calc-btns button {
|
||
|
background: #313141;
|
||
|
color: var(--whoogle-dark-text);
|
||
|
border: none;
|
||
|
border-radius: 8px;
|
||
|
padding: 8px;
|
||
|
cursor: pointer;
|
||
|
}
|
||
|
#calc-btns button:hover {
|
||
|
background: #414151;
|
||
|
}
|
||
|
#calc-btns .common {
|
||
|
background: #51516a;
|
||
|
}
|
||
|
#calc-btns .common:hover {
|
||
|
background: #61617a;
|
||
|
}
|
||
|
#calc-btn-0 { grid-row: 5; grid-column: 3; }
|
||
|
#calc-btn-1 { grid-row: 4; grid-column: 3; }
|
||
|
#calc-btn-2 { grid-row: 4; grid-column: 4; }
|
||
|
#calc-btn-3 { grid-row: 4; grid-column: 5; }
|
||
|
#calc-btn-4 { grid-row: 3; grid-column: 3; }
|
||
|
#calc-btn-5 { grid-row: 3; grid-column: 4; }
|
||
|
#calc-btn-6 { grid-row: 3; grid-column: 5; }
|
||
|
#calc-btn-7 { grid-row: 2; grid-column: 3; }
|
||
|
#calc-btn-8 { grid-row: 2; grid-column: 4; }
|
||
|
#calc-btn-9 { grid-row: 2; grid-column: 5; }
|
||
|
#calc-btn-EQ { grid-row: 5; grid-column: 5; }
|
||
|
#calc-btn-PT { grid-row: 5; grid-column: 4; }
|
||
|
#calc-btn-BCK { grid-row: 5; grid-column: 6; }
|
||
|
#calc-btn-ADD { grid-row: 4; grid-column: 6; }
|
||
|
#calc-btn-SUB { grid-row: 3; grid-column: 6; }
|
||
|
#calc-btn-MLT { grid-row: 2; grid-column: 6; }
|
||
|
#calc-btn-DIV { grid-row: 1; grid-column: 6; }
|
||
|
#calc-btn-CLR { grid-row: 1; grid-column: 5; }
|
||
|
#calc-btn-PRC{ grid-row: 1; grid-column: 4; }
|
||
|
#calc-btn-RP { grid-row: 1; grid-column: 3; }
|
||
|
#calc-btn-LP { grid-row: 1; grid-column: 2; }
|
||
|
#calc-btn-ABS { grid-row: 1; grid-column: 1; }
|
||
|
#calc-btn-SIN { grid-row: 2; grid-column: 2; }
|
||
|
#calc-btn-COS { grid-row: 3; grid-column: 2; }
|
||
|
#calc-btn-TAN { grid-row: 4; grid-column: 2; }
|
||
|
#calc-btn-SQR { grid-row: 5; grid-column: 2; }
|
||
|
#calc-btn-EXP { grid-row: 2; grid-column: 1; }
|
||
|
#calc-btn-E { grid-row: 3; grid-column: 1; }
|
||
|
#calc-btn-PI { grid-row: 4; grid-column: 1; }
|
||
|
#calc-btn-LOG { grid-row: 5; grid-column: 1; }
|
||
|
</style>
|
||
|
<p id="prev-equation"></p>
|
||
|
<div id="calculator-widget">
|
||
|
<p id="calc-text">0</p>
|
||
|
<div id="calc-btns">
|
||
|
<button id="calc-btn-0" class="common">0</button>
|
||
|
<button id="calc-btn-1" class="common">1</button>
|
||
|
<button id="calc-btn-2" class="common">2</button>
|
||
|
<button id="calc-btn-3" class="common">3</button>
|
||
|
<button id="calc-btn-4" class="common">4</button>
|
||
|
<button id="calc-btn-5" class="common">5</button>
|
||
|
<button id="calc-btn-6" class="common">6</button>
|
||
|
<button id="calc-btn-7" class="common">7</button>
|
||
|
<button id="calc-btn-8" class="common">8</button>
|
||
|
<button id="calc-btn-9" class="common">9</button>
|
||
|
<button id="calc-btn-EQ" class="common">=</button>
|
||
|
<button id="calc-btn-PT" class="common">.</button>
|
||
|
<button id="calc-btn-BCK">⬅</button>
|
||
|
<button id="calc-btn-ADD">+</button>
|
||
|
<button id="calc-btn-SUB">-</button>
|
||
|
<button id="calc-btn-MLT">x</button>
|
||
|
<button id="calc-btn-DIV">/</button>
|
||
|
<button id="calc-btn-CLR">C</button>
|
||
|
<button id="calc-btn-PRC">%</button>
|
||
|
<button id="calc-btn-RP">)</button>
|
||
|
<button id="calc-btn-LP">(</button>
|
||
|
<button id="calc-btn-ABS">|x|</button>
|
||
|
<button id="calc-btn-SIN">sin</button>
|
||
|
<button id="calc-btn-COS">cos</button>
|
||
|
<button id="calc-btn-TAN">tan</button>
|
||
|
<button id="calc-btn-SQR">√</button>
|
||
|
<button id="calc-btn-EXP">^</button>
|
||
|
<button id="calc-btn-E">ℇ</button>
|
||
|
<button id="calc-btn-PI">π</button>
|
||
|
<button id="calc-btn-LOG">log</button>
|
||
|
</div>
|
||
|
</div>
|
||
|
<script>
|
||
|
// JS does not have this by default.
|
||
|
// from https://www.freecodecamp.org/news/how-to-factorialize-a-number-in-javascript-9263c89a4b38/
|
||
|
function factorial(num) {
|
||
|
if (num < 0)
|
||
|
return -1;
|
||
|
else if (num === 0)
|
||
|
return 1;
|
||
|
else {
|
||
|
return (num * factorial(num - 1));
|
||
|
}
|
||
|
}
|
||
|
// returns true if the user is currently focused on the calculator widget
|
||
|
function usingCalculator() {
|
||
|
let activeElement = document.activeElement;
|
||
|
while (true) {
|
||
|
if (!activeElement) return false;
|
||
|
if (activeElement.id === "calculator-wrapper") return true;
|
||
|
activeElement = activeElement.parentElement;
|
||
|
}
|
||
|
}
|
||
|
const $ = q => document.querySelectorAll(q);
|
||
|
// key bindings for commonly used buttons
|
||
|
const keybindings = {
|
||
|
"0": "0",
|
||
|
"1": "1",
|
||
|
"2": "2",
|
||
|
"3": "3",
|
||
|
"4": "4",
|
||
|
"5": "5",
|
||
|
"6": "6",
|
||
|
"7": "7",
|
||
|
"8": "8",
|
||
|
"9": "9",
|
||
|
"Enter": "EQ",
|
||
|
".": "PT",
|
||
|
"+": "ADD",
|
||
|
"-": "SUB",
|
||
|
"*": "MLT",
|
||
|
"/": "DIV",
|
||
|
"%": "PRC",
|
||
|
"c": "CLR",
|
||
|
"(": "LP",
|
||
|
")": "RP",
|
||
|
"Backspace": "BCK",
|
||
|
}
|
||
|
window.addEventListener("keydown", event => {
|
||
|
if (!usingCalculator()) return;
|
||
|
if (event.key === "Enter" && document.activeElement.id !== "search-bar")
|
||
|
event.preventDefault();
|
||
|
if (keybindings[event.key])
|
||
|
document.getElementById("calc-btn-" + keybindings[event.key]).click();
|
||
|
})
|
||
|
// calculates the string
|
||
|
const calc = () => {
|
||
|
var mathtext = document.getElementById("calc-text");
|
||
|
var statement = mathtext.innerHTML
|
||
|
// remove empty ()
|
||
|
.replace("()", "")
|
||
|
// special constants
|
||
|
.replace("π", "(Math.PI)")
|
||
|
.replace("ℇ", "(Math.E)")
|
||
|
// turns 3(1+2) into 3*(1+2) (for example)
|
||
|
.replace(/(?<=[0-9\)])(?<=[^+\-x*\/%^])\(/, "x(")
|
||
|
// same except reversed
|
||
|
.replace(/\)(?=[0-9\(])(?=[^+\-x*\/%^])/, ")x")
|
||
|
// replace human friendly x with JS *
|
||
|
.replace("x", "*")
|
||
|
// trig & misc functions
|
||
|
.replace("sin", "Math.sin")
|
||
|
.replace("cos", "Math.cos")
|
||
|
.replace("tan", "Math.tan")
|
||
|
.replace("√", "Math.sqrt")
|
||
|
.replace("^", "**")
|
||
|
.replace("abs", "Math.abs")
|
||
|
.replace("log", "Math.log")
|
||
|
;
|
||
|
// add any missing )s to the end
|
||
|
while(true) if (
|
||
|
(statement.match(/\(/g) || []).length >
|
||
|
(statement.match(/\)/g) || []).length
|
||
|
) statement += ")"; else break;
|
||
|
// evaluate the expression.
|
||
|
console.log("calculating [" + statement + "]");
|
||
|
try {
|
||
|
var result = eval(statement);
|
||
|
document.getElementById("prev-equation").innerHTML = mathtext.innerHTML + " = ";
|
||
|
mathtext.innerHTML = result;
|
||
|
mathtext.classList.remove("error-border");
|
||
|
} catch (e) {
|
||
|
mathtext.classList.add("error-border");
|
||
|
console.error(e);
|
||
|
}
|
||
|
}
|
||
|
const updateCalc = (e) => {
|
||
|
// character(s) recieved from button
|
||
|
var c = event.target.innerHTML;
|
||
|
var mathtext = document.getElementById("calc-text");
|
||
|
if (mathtext.innerHTML === "0") mathtext.innerHTML = "";
|
||
|
// special cases
|
||
|
switch (c) {
|
||
|
case "C":
|
||
|
// Clear
|
||
|
mathtext.innerHTML = "0";
|
||
|
break;
|
||
|
case "⬅":
|
||
|
// Delete
|
||
|
mathtext.innerHTML = mathtext.innerHTML.slice(0, -1);
|
||
|
if (mathtext.innerHTML.length === 0) {
|
||
|
mathtext.innerHTML = "0";
|
||
|
}
|
||
|
break;
|
||
|
case "=":
|
||
|
calc()
|
||
|
break;
|
||
|
case "sin":
|
||
|
case "cos":
|
||
|
case "tan":
|
||
|
case "log":
|
||
|
case "√":
|
||
|
mathtext.innerHTML += `${c}(`;
|
||
|
break;
|
||
|
case "|x|":
|
||
|
mathtext.innerHTML += "abs("
|
||
|
break;
|
||
|
case "+":
|
||
|
case "-":
|
||
|
case "x":
|
||
|
case "/":
|
||
|
case "%":
|
||
|
case "^":
|
||
|
if (mathtext.innerHTML.length === 0) mathtext.innerHTML = "0";
|
||
|
// prevent typing 2 operators in a row
|
||
|
if (mathtext.innerHTML.match(/[+\-x\/%^] $/))
|
||
|
mathtext.innerHTML = mathtext.innerHTML.slice(0, -3);
|
||
|
mathtext.innerHTML += ` ${c} `;
|
||
|
break;
|
||
|
default:
|
||
|
mathtext.innerHTML += c;
|
||
|
}
|
||
|
}
|
||
|
for (let i of $("#calc-btns button")) {
|
||
|
i.addEventListener('click', event => {
|
||
|
updateCalc(event);
|
||
|
})
|
||
|
}
|
||
|
</script>
|