<!DOCTYPE html>
<html lang="en">

<head>
  <meta charset="utf-8" />
  <meta name="viewport" content="width=device-width, initial-scale=1" />
  <title>Interactive Hyperplane Classifier — p5.js</title>
  <style>
    html,
    body {
      height: 100%;
      margin: 0;
      user-select: none;
      background: #0b0e14;
      color: #e6e6e6;
      font-family: system-ui, -apple-system, Segoe UI, Roboto, Arial, sans-serif;
    }

    #ui {
      position: fixed;
      inset: 12px auto auto 12px;
      background: rgba(255, 255, 255, 0.06);
      backdrop-filter: blur(4px);
      border: 1px solid rgba(255, 255, 255, 0.12);
      border-radius: 12px;
      padding: 10px 12px;
      line-height: 1.4;
    }

    kbd {
      background: rgba(255, 255, 255, 0.2);
      border: 2px solid rgba(255, 255, 255, 0.4);
      border-radius: 6px;
      padding: 2px 6px;
      font-family: ui-monospace, SFMono-Regular, Menlo, Consolas, monospace;
      font-size: 14px;
    }

    a {
      color: #8dd0ff;
      text-decoration: none;
    }
  </style>
  <script src="https://cdn.jsdelivr.net/npm/p5@1.9.4/lib/p5.min.js"></script>
</head>

<body>
  <div id="ui">
    <div><strong>Interactive Hyperplane Classifier</strong></div>
    <div id="stats"></div>
    <div style="width: 900px">Left‑click: add point • Drag: move point • <kbd>1</kbd>/<kbd>2</kbd> active class •
      <kbd>F</kbd> fit
      (logistic) • <kbd>P</kbd> perceptron • <kbd>L</kbd> live fit • <kbd>U</kbd> toggle UI • <kbd>M</kbd> toggle
      margins • <kbd>R</kbd>
      random • <kbd>C</kbd> clear
    </div>
  </div>
  <script>
    const W = 900, H = 620;
    const RADIUS = 12;
    let pts = [];
    let numRandomPoints = 20;
    let activeClass = 1;
    let w = [0, 0, 0];
    let lr = 0.1; // learning rate: smaller=small updates, slow training, stable. large=vice versa
    // initial given was 0.2
    let liveFit = false;
    let dragging = null;
    let showMargins = false;
    let showUI = true;
    const statsUI = document.getElementById('ui');
    const statsEl = document.getElementById('stats');

    function setup() {
      const c = createCanvas(W, H);
      c.parent(document.body);
      pixelDensity(1);
      randomPoints(numRandomPoints);
      noStroke();
      textFont('ui-monospace, SFMono-Regular, Menlo, Consolas, monospace');
    }

    function draw() {
      background(20);

      // uncomment to see the training at the start (liveFit will override)
      // for (let i = 0; i < 2; i++) stepLogReg();

      drawDecisionField();
      drawDecisionBoundary();
      if (showMargins) drawMargins();
      drawPoints();

      if (liveFit && pts.length > 0) {
        for (let i = 0; i < 2; i++) stepLogReg();
      }

      const s = computeStats();
      statsEl.innerHTML = `Class: <b>${activeClass === 1 ? 'blue (1)' : 'orange (0)'}</b> • ` +
        `w=[${w.map(v => v.toFixed(2)).join(', ')}] • ` +
        `acc=${(s.acc * 100).toFixed(1)}% • N=${pts.length}`;
    }

    function mousePressed() {
      for (let i = pts.length - 1; i >= 0; i--) {
        const p = pts[i];
        if (dist(mouseX, mouseY, p.x, p.y) <= RADIUS + 2) { dragging = i; return; }
      }
      if (mouseX >= 0 && mouseX <= width && mouseY >= 0 && mouseY <= height) {
        pts.push({ x: mouseX, y: mouseY, y01: activeClass, drag: false });
      }
    }
    function mouseDragged() { if (dragging != null) { pts[dragging].x = mouseX; pts[dragging].y = mouseY; } }
    function mouseReleased() { dragging = null; }

    function keyPressed() {
      if (key === '1') { activeClass = 0; }
      if (key === '2') { activeClass = 1; }
      if (key === 'F' || key === 'f') { for (let i = 0; i < 1500; i++) stepLogReg(); }
      if (key === 'P' || key === 'p') { perceptronTrain(4000); }
      if (key === 'L' || key === 'l') { liveFit = !liveFit; }
      if (key === 'U' || key === 'u') { showUI = !showUI; }
      if (key === 'M' || key === 'm') { showMargins = !showMargins; }
      if (key === 'R' || key === 'r') { randomPoints(numRandomPoints); }
      if (key === 'C' || key === 'c') { pts = []; randomizeWeights(); }
      (showUI) ? statsUI.style.display = "block" : statsUI.style.display = "none";
    }

    function normX(px) { return map(px, 0, width, -1, 1); }
    function normY(py) { return map(py, height, 0, -1, 1); }
    function denormX(nx) { return map(nx, -1, 1, 0, width); }
    function denormY(ny) { return map(ny, -1, 1, height, 0); }

    function sigmoid(z) { return 1 / (1 + Math.exp(-z)); }

    function stepLogReg() {
      if (pts.length === 0) return;
      const p = random(pts);
      const x = [normX(p.x), normY(p.y), 1];
      const y = p.y01;
      const z = w[0] * x[0] + w[1] * x[1] + w[2] * x[2];
      const yhat = sigmoid(z);
      const err = (y - yhat);
      w[0] += lr * err * x[0];
      w[1] += lr * err * x[1];
      w[2] += lr * err * x[2];
    }

    function perceptronTrain(steps = 3000) {
      if (pts.length === 0) return;
      for (let s = 0; s < steps; s++) {
        const p = pts[s % pts.length];
        const x = [normX(p.x), normY(p.y), 1];
        const y = p.y01 ? 1 : -1;
        const z = w[0] * x[0] + w[1] * x[1] + w[2] * x[2];
        const yhat = z >= 0 ? 1 : -1;
        if (y * yhat <= 0) {
          w[0] += lr * y * x[0];
          w[1] += lr * y * x[1];
          w[2] += lr * y * x[2];
        }
      }
    }

    function predict01(px, py) {
      const x = [normX(px), normY(py), 1];
      const z = w[0] * x[0] + w[1] * x[1] + w[2] * x[2];
      return sigmoid(z) >= 0.5 ? 1 : 0;
    }

    function computeStats() {
      if (pts.length === 0) return { acc: 1 };
      let correct = 0;
      for (const p of pts) { if (predict01(p.x, p.y) === p.y01) correct++; }
      return { acc: correct / pts.length };
    }

    function randomizeWeights() { w = [random(-1, 1), random(-1, 1), random(-0.3, 0.3)]; }

    function drawPoints() {
      for (const p of pts) {
        const isBlue = p.y01 === 1;
        stroke(0, 0, 0, 200); strokeWeight(2); fill(isBlue ? color(90, 160, 255) : color(255, 150, 70));
        circle(p.x, p.y, RADIUS * 2);
      }
    }

    function drawDecisionBoundary() {
      const wx = w[0], wy = w[1], b = w[2];
      if (Math.abs(wy) > 1e-6) {
        const xL = -1, xR = 1;
        const yL = -(wx * xL + b) / wy;
        const yR = -(wx * xR + b) / wy;
        stroke(235, 235, 235, 180); strokeWeight(3); noFill();
        line(denormX(xL), denormY(yL), denormX(xR), denormY(yR));
      } else {
        const xN = -b / (wx || 1e-6);
        stroke(235, 235, 235, 180); strokeWeight(3);
        line(denormX(xN), 0, denormX(xN), height);
      }
    }

    function drawMargins() {
      const wx = w[0], wy = w[1], b = w[2];
      const norm = Math.sqrt(wx * wx + wy * wy) || 1e-6;
      for (const p of pts) {
        const x0 = normX(p.x), y0 = normY(p.y);
        const signed = (wx * x0 + wy * y0 + b) / norm;
        const xp = x0 - (wx * signed) / norm;
        const yp = y0 - (wy * signed) / norm;
        const Xp = denormX(xp), Yp = denormY(yp);
        stroke(p.y01 === 1 ? color(90, 160, 255, 170) : color(255, 150, 70, 170));
        strokeWeight(1.5);
        line(p.x, p.y, Xp, Yp);
        noStroke(); fill(235, 235, 235, 200); textSize(14); textAlign(LEFT, CENTER);
        text(Math.abs(signed).toFixed(2), p.x + 14, p.y);
      }
    }

    function drawDecisionField() {
      noStroke();
      const step = 10;
      for (let y = 0; y < height; y += step) {
        for (let x = 0; x < width; x += step) {
          const p = predict01(x + step / 2, y + step / 2);
          const col = p ? color(30, 70, 120, 40) : color(120, 70, 30, 40);
          fill(col); rect(x, y, step, step);
        }
      }
    }

    function randomPoints(n = 20) {
      pts = []; randomizeWeights();
      const blobs = [{ x: width * 0.33, y: height * 0.5, c: 0 }, { x: width * 0.66, y: height * 0.5, c: 1 }];
      for (let i = 0; i < n; i++) {
        const b = random(blobs);
        const px = b.x + randomGaussian() * width * 0.15;
        const py = b.y + randomGaussian() * height * 0.1;
        const p = { x: constrain(px, 10, width - 10), y: constrain(py, 10, height - 10), y01: b.c, drag: false };
        pts.push(p);
      }
      for (let k = 0; k < 500; k++) stepLogReg(); // initial training of the logReg line so it stabilizes a bit
    }
  </script>
</body>

</html>