parent
9e8dd00f10
commit
c3554037b0
@ -0,0 +1,50 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="UTF-8" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||
<title>Bin2Dec Practice App</title>
|
||||
<link rel="preconnect" href="https://fonts.googleapis.com" />
|
||||
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin />
|
||||
<link href="https://fonts.googleapis.com/css2?family=Space+Grotesk:wght@400;500;700&display=swap" rel="stylesheet" />
|
||||
<link rel="stylesheet" href="style.css" />
|
||||
</head>
|
||||
<body>
|
||||
<main class="app-shell">
|
||||
<section class="card" aria-labelledby="title">
|
||||
<p class="eyebrow">CommitVerse Practice</p>
|
||||
<h1 id="title">Binary to Decimal Converter</h1>
|
||||
<p class="subtitle">Enter a binary number and get its base-10 value.</p>
|
||||
|
||||
<form id="converter-form" novalidate>
|
||||
<label for="binary-input">Binary Input</label>
|
||||
<input
|
||||
id="binary-input"
|
||||
name="binary"
|
||||
type="text"
|
||||
inputmode="numeric"
|
||||
maxlength="8"
|
||||
placeholder="Example: 10110101"
|
||||
aria-describedby="input-hint error-text"
|
||||
required
|
||||
/>
|
||||
<p id="input-hint" class="hint">Up to 8 digits (0 or 1 only)</p>
|
||||
|
||||
<label class="toggle-row" for="allow-variable-length">
|
||||
<input id="allow-variable-length" type="checkbox" />
|
||||
<span>Bonus mode: allow variable length input</span>
|
||||
</label>
|
||||
|
||||
<button type="submit">Convert</button>
|
||||
</form>
|
||||
|
||||
<p id="error-text" class="error" role="alert" aria-live="polite"></p>
|
||||
|
||||
<label for="decimal-output">Decimal Output</label>
|
||||
<output id="decimal-output" class="output" aria-live="polite">-</output>
|
||||
</section>
|
||||
</main>
|
||||
|
||||
<script src="script.js"></script>
|
||||
</body>
|
||||
</html>
|
||||
@ -0,0 +1,69 @@
|
||||
const form = document.getElementById("converter-form");
|
||||
const binaryInput = document.getElementById("binary-input");
|
||||
const allowVariableLength = document.getElementById("allow-variable-length");
|
||||
const errorText = document.getElementById("error-text");
|
||||
const decimalOutput = document.getElementById("decimal-output");
|
||||
|
||||
function setError(message) {
|
||||
errorText.textContent = message;
|
||||
}
|
||||
|
||||
function sanitizeInput(value) {
|
||||
return value.trim();
|
||||
}
|
||||
|
||||
function isBinaryString(value) {
|
||||
return /^[01]+$/.test(value);
|
||||
}
|
||||
|
||||
function binaryToDecimal(binaryValue) {
|
||||
let decimalValue = 0;
|
||||
let positionFromRight = 0;
|
||||
|
||||
for (let i = binaryValue.length - 1; i >= 0; i -= 1) {
|
||||
if (binaryValue[i] === "1") {
|
||||
decimalValue += Math.pow(2, positionFromRight);
|
||||
}
|
||||
positionFromRight += 1;
|
||||
}
|
||||
|
||||
return decimalValue;
|
||||
}
|
||||
|
||||
function validateInput(binaryValue, allowLongInput) {
|
||||
if (!binaryValue) {
|
||||
return "Please enter a binary number.";
|
||||
}
|
||||
|
||||
if (!isBinaryString(binaryValue)) {
|
||||
return "Only 0 and 1 are allowed.";
|
||||
}
|
||||
|
||||
if (!allowLongInput && binaryValue.length > 8) {
|
||||
return "Use up to 8 digits, or enable bonus mode.";
|
||||
}
|
||||
|
||||
return "";
|
||||
}
|
||||
|
||||
allowVariableLength.addEventListener("change", () => {
|
||||
binaryInput.maxLength = allowVariableLength.checked ? 64 : 8;
|
||||
setError("");
|
||||
});
|
||||
|
||||
form.addEventListener("submit", (event) => {
|
||||
event.preventDefault();
|
||||
setError("");
|
||||
|
||||
const binaryValue = sanitizeInput(binaryInput.value);
|
||||
const validationMessage = validateInput(binaryValue, allowVariableLength.checked);
|
||||
|
||||
if (validationMessage) {
|
||||
decimalOutput.textContent = "-";
|
||||
setError(validationMessage);
|
||||
return;
|
||||
}
|
||||
|
||||
const result = binaryToDecimal(binaryValue);
|
||||
decimalOutput.textContent = String(result);
|
||||
});
|
||||
@ -0,0 +1,157 @@
|
||||
:root {
|
||||
--bg-a: #fbf3ea;
|
||||
--bg-b: #d6efe2;
|
||||
--card-bg: rgba(255, 255, 255, 0.82);
|
||||
--text-main: #1f2937;
|
||||
--text-muted: #5b6878;
|
||||
--accent: #006d77;
|
||||
--accent-strong: #00545c;
|
||||
--danger: #bf1f2f;
|
||||
--ring: rgba(0, 109, 119, 0.28);
|
||||
}
|
||||
|
||||
* {
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
body {
|
||||
margin: 0;
|
||||
min-height: 100vh;
|
||||
font-family: "Space Grotesk", sans-serif;
|
||||
color: var(--text-main);
|
||||
background:
|
||||
radial-gradient(circle at 12% 20%, rgba(250, 173, 20, 0.2), transparent 42%),
|
||||
radial-gradient(circle at 82% 15%, rgba(0, 109, 119, 0.23), transparent 36%),
|
||||
linear-gradient(145deg, var(--bg-a), var(--bg-b));
|
||||
display: grid;
|
||||
place-items: center;
|
||||
padding: 24px;
|
||||
}
|
||||
|
||||
.app-shell {
|
||||
width: min(640px, 100%);
|
||||
}
|
||||
|
||||
.card {
|
||||
border-radius: 24px;
|
||||
padding: 28px;
|
||||
background: var(--card-bg);
|
||||
border: 1px solid rgba(255, 255, 255, 0.65);
|
||||
backdrop-filter: blur(8px);
|
||||
box-shadow: 0 24px 60px rgba(18, 38, 63, 0.18);
|
||||
animation: rise-in 420ms ease-out;
|
||||
}
|
||||
|
||||
.eyebrow {
|
||||
margin: 0;
|
||||
font-weight: 700;
|
||||
text-transform: uppercase;
|
||||
letter-spacing: 0.08em;
|
||||
color: var(--accent);
|
||||
font-size: 0.76rem;
|
||||
}
|
||||
|
||||
h1 {
|
||||
margin: 10px 0 6px;
|
||||
font-size: clamp(1.5rem, 3.2vw, 2rem);
|
||||
}
|
||||
|
||||
.subtitle {
|
||||
margin: 0 0 18px;
|
||||
color: var(--text-muted);
|
||||
}
|
||||
|
||||
form {
|
||||
display: grid;
|
||||
gap: 10px;
|
||||
}
|
||||
|
||||
label {
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
input[type="text"] {
|
||||
width: 100%;
|
||||
border: 1px solid rgba(31, 41, 55, 0.26);
|
||||
border-radius: 12px;
|
||||
padding: 11px 12px;
|
||||
font: inherit;
|
||||
transition: border-color 160ms ease, box-shadow 160ms ease;
|
||||
}
|
||||
|
||||
input[type="text"]:focus {
|
||||
outline: none;
|
||||
border-color: var(--accent);
|
||||
box-shadow: 0 0 0 4px var(--ring);
|
||||
}
|
||||
|
||||
.hint {
|
||||
margin: 0 0 6px;
|
||||
color: var(--text-muted);
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.toggle-row {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
font-size: 0.92rem;
|
||||
font-weight: 500;
|
||||
color: var(--text-muted);
|
||||
margin: 2px 0 8px;
|
||||
}
|
||||
|
||||
button {
|
||||
border: none;
|
||||
border-radius: 12px;
|
||||
padding: 12px;
|
||||
font: inherit;
|
||||
font-weight: 700;
|
||||
background: var(--accent);
|
||||
color: #fff;
|
||||
cursor: pointer;
|
||||
transition: transform 140ms ease, background-color 140ms ease;
|
||||
}
|
||||
|
||||
button:hover {
|
||||
background: var(--accent-strong);
|
||||
transform: translateY(-1px);
|
||||
}
|
||||
|
||||
.error {
|
||||
min-height: 1.4em;
|
||||
margin: 14px 0 8px;
|
||||
color: var(--danger);
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.output {
|
||||
display: block;
|
||||
width: 100%;
|
||||
margin-top: 8px;
|
||||
border: 1px dashed rgba(31, 41, 55, 0.35);
|
||||
background: rgba(255, 255, 255, 0.7);
|
||||
border-radius: 12px;
|
||||
padding: 12px;
|
||||
font-size: 1.15rem;
|
||||
font-weight: 700;
|
||||
min-height: 50px;
|
||||
}
|
||||
|
||||
@keyframes rise-in {
|
||||
from {
|
||||
opacity: 0;
|
||||
transform: translateY(14px) scale(0.985);
|
||||
}
|
||||
to {
|
||||
opacity: 1;
|
||||
transform: translateY(0) scale(1);
|
||||
}
|
||||
}
|
||||
|
||||
@media (max-width: 640px) {
|
||||
.card {
|
||||
padding: 22px;
|
||||
border-radius: 18px;
|
||||
}
|
||||
}
|
||||
Loading…
Reference in new issue