<!DOCTYPE html>
<html lang="ko">
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>Virtual Pointer Rating Task</title>
    <style>
      :root { font-family: system-ui, sans-serif; }
      body { display: flex; flex-direction: column; align-items: center; gap: 20px; padding: 40px 16px; }
      h1 { font-size: 1.6rem; margin: 0; text-align: center; }
      #instructions { text-align: center; max-width: 560px; line-height: 1.4; }
      #target-display { font-size: 1.2rem; margin-bottom: 6px; }
      #start-btn { padding: 10px 20px; border: none; border-radius: 8px; background: #00aa55; color: #fff; font-size: 1rem; cursor: pointer; transition: background 0.2s, transform 0.2s; }
      #start-btn:hover { background: #007acc; transform: scale(1.03); }
      #star-wrap { user-select: none; touch-action: none; display: flex; gap: 10px; cursor: pointer; --preview-color: gold; --confirm-color: orange; }
      .star { position: relative; font-size: 60px; line-height: 1; color: #c7c7c7; }
      .star::before { content: '★'; position: absolute; left: 0; top: 0; width: var(--fill,0%); overflow: hidden; color: var(--overlay,var(--preview-color)); transition: width 0.06s ease; }
      #toast { position: fixed; top: 16px; left: 50%; transform: translateX(-50%); padding: 10px 18px; border-radius: 6px; color: #fff; font-weight: 600; display: none; z-index: 9999; background: #333; }
      table { border-collapse: collapse; width: 100%; max-width: 660px; }
      th,td { border: 1px solid #ccc; padding: 4px 6px; font-size: 0.8rem; text-align: center; }
      #download { padding: 8px 16px; border: none; border-radius: 6px; background: #0066ff; color: #fff; font-size: 0.9rem; cursor: pointer; }
      #download:hover { filter: brightness(1.1); }
      #pointer {
        position: fixed;
        top: 50%;
        left: 50%;
        transform: translate(-50%, -50%) scale(1, -1);
        pointer-events: none;
        z-index: 9999;
        width: 32px;
        height: 32px;
      }
    </style>
  </head>
  <body>
    <h1>⭐ Virtual Pointer Rating Task (0.5★ units)</h1>
    <div id="instructions">First, freely manipulate the star rating in <strong>Practice Mode</strong>.<br>Once you're comfortable, press the <em>Start Experiment</em> button.<br>In this experiment, you'll need to enter the given <strong>Target Rating</strong> as quickly and accurately as possible.</div>
    <button id="start-btn">Start Experiment</button>
    <div id="target-display">Practice Mode: Feel free to manipulate your star rating!</div>
    <div id="star-wrap"></div>
    <button id="download">Download CSV</button>

    <h2>Logs</h2>
    <table id="log-table">
      <thead><tr><th>Trial</th><th>Mode</th><th>Target</th><th>FirstAttempt</th><th>Attempts</th><th>Overshoot</th><th>T1st(ms)</th><th>Tcorrect(ms)</th></tr></thead>
      <tbody></tbody>
    </table>
    <div id="toast"></div>
    <img id="pointer" alt="pointer" />

    <script>
      (function () {
        const TOTAL_STARS = 5;
        const STEP = 0.5;
        const DELAY_NEXT = 400;

        const wrap = document.getElementById('star-wrap');
        const startBtn = document.getElementById('start-btn');
        const toast = document.getElementById('toast');
        const targetEl = document.getElementById('target-display');
        const manualDL = document.getElementById('download');
        const tbody = document.querySelector('#log-table tbody');
        const pointer = document.getElementById("pointer");

        for (let i = 1; i <= TOTAL_STARS; i++) {
          const s = document.createElement('span');
          s.className = 'star';
          s.dataset.star = i;
          s.textContent = '☆';
          wrap.appendChild(s);
        }
        const stars = Array.from(document.querySelectorAll('.star'));

        let mode = 'practice';
        let targets = [];
        let trialIdx = -1;
        let currentTarget = null;
        let stimulusTime = null;
        let firstAttemptRating = null;
        let timeToFirstAttempt = null;
        let attemptCount = 0;
        let practiceCounter = 0;
        const logsMain = [];

        let pointerX = window.innerWidth / 2;
        let pointerY = window.innerHeight / 2;
        let lastTouch = null;
        let lastElement = null;
        let currentHoveredRating = null;

        function getPixelsPerMM() {
          const div = document.createElement("div");
          div.style.width = "100mm";
          div.style.position = "absolute";
          div.style.visibility = "hidden";
          document.body.appendChild(div);
          const pxPer100mm = div.getBoundingClientRect().width;
          document.body.removeChild(div);
          return pxPer100mm / 100;
        }

        const pixelsPerMM = getPixelsPerMM();
        const physicalMM = 8.47; // 32px at 96dpi ≈ 8.47mm
        const sizePx = pixelsPerMM * physicalMM;

        pointer.src = "data:image/png;base64,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";
        pointer.style.width = `${sizePx}px`;
        pointer.style.height = `${sizePx}px`;

        const fmt = r => (r % 1 === 0 ? r.toFixed(0) + '★' : Math.floor(r) + '★½');
        const shuffle = arr => arr.sort(() => Math.random() - 0.5);

        function renderStars(rating, colorVar) {
          stars.forEach((s, idx) => {
            const n = idx + 1;
            let fill = 0;
            if (rating >= n) fill = 100;
            else if (rating >= n - 0.5) fill = 50;
            s.style.setProperty('--fill', fill + '%');
            s.style.setProperty('--overlay', colorVar);
          });
        }

        function eventToRating() {
          const pointerRect = pointer.getBoundingClientRect();
          const x = pointerRect.left;  // Based on the top left of the image
          const y = pointerRect.top;   // Based on the top of the image

          for (let i = 0; i < stars.length; i++) {
            const rect = stars[i].getBoundingClientRect();
            if (x >= rect.left && x <= rect.right && y >= rect.top && y <= rect.bottom) {
              const local = x - rect.left;
              return i + (local <= rect.width / 2 ? 0.5 : 1);
            }
          }

          for (let i = 1; i < stars.length; i++) {
            const prev = stars[i - 1].getBoundingClientRect();
            const curr = stars[i].getBoundingClientRect();
            if (x > prev.right && x < curr.left && y >= prev.top && y <= prev.bottom) {
              return i + 0.0;
            }
          }

          const first = stars[0].getBoundingClientRect();
          const last = stars[stars.length - 1].getBoundingClientRect();
          if (x < first.left || y < first.top || y > first.bottom) return 0.5;
          if (x > last.right || y < last.top || y > last.bottom) return TOTAL_STARS;

          return 0.5;
        }

        function updateHoverLoop() {
          if (mode !== 'finished') {
            const r = eventToRating();
            currentHoveredRating = r;  // Remember the currently hovered rating
            renderStars(r, getComputedStyle(wrap).getPropertyValue('--preview-color'));

            const el = document.elementFromPoint(pointerX, pointerY);
            if (el !== lastElement) {
              if (lastElement) lastElement.dispatchEvent(new Event("mouseout"));
              if (el) el.dispatchEvent(new Event("mouseover"));
              lastElement = el;
            }
            if (el) el.dispatchEvent(new Event("mousemove"));
          }
          requestAnimationFrame(updateHoverLoop);
        }
        requestAnimationFrame(updateHoverLoop);

        function isPointerOverStarOrGap() {
          const x = pointer.getBoundingClientRect().left;
          const y = pointer.getBoundingClientRect().top;

          for (let i = 0; i < stars.length; i++) {
            const rect = stars[i].getBoundingClientRect();
            if (x >= rect.left && x <= rect.right && y >= rect.top && y <= rect.bottom) {
              return true;  // Above the stars
            }
          }
          for (let i = 1; i < stars.length; i++) {
            const prev = stars[i - 1].getBoundingClientRect();
            const curr = stars[i].getBoundingClientRect();
            if (
              x > prev.right && x < curr.left &&
              y >= prev.top && y <= prev.bottom
            ) {
              return true;  // Spacing between stars
            }
          }
          return false;
        }

        function addRow(l) {
          const tr = document.createElement('tr');
          tr.innerHTML = `<td>${l.trial}</td><td>${l.mode}</td><td>${l.target}</td><td>${l.firstAttempt}</td><td>${l.attempts}</td><td>${l.overshoot}</td><td>${l.t1st}</td><td>${l.tCorrect}</td>`;
          tbody.prepend(tr);
        }

        function downloadCSV(auto = false) {
          if (!logsMain.length) {
            if (!auto) showToast('There is no experimental data.', false);
            return;
          }

          let csv = 'trial,target,firstAttempt,attempts,overshoot,t1st_ms,tCorrect_ms\n';
          logsMain.forEach(l => {
            csv += `${l.trial},${l.target},${l.firstAttempt},${l.attempts},${l.overshoot},${l.t1st},${l.tCorrect}\n`;
          });
          const blob = new Blob([csv], { type: 'text/csv;charset=utf-8;' });
          const url = URL.createObjectURL(blob);
          const a = document.createElement('a');
          a.href = url;
          a.download = 'rating_task_logs.csv';
          a.style.display = 'none';
          document.body.appendChild(a);
          a.click();
          document.body.removeChild(a);
          URL.revokeObjectURL(url);
        }
        manualDL.addEventListener('click', () => downloadCSV(false));

        function showToast(msg, ok) {
          toast.textContent = msg;
          toast.style.background = ok ? '#19c37d' : '#d91e1e';
          toast.style.display = 'block';
          setTimeout(() => toast.style.display = 'none', 1500);
        }

        function beginMain() {
          mode = 'main';
          startBtn.style.display = 'none';
          document.getElementById('instructions').textContent = 'In progress…';
          targets = shuffle(Array.from({ length: TOTAL_STARS * 2 }, (_, i) => (i + 1) * STEP));
          trialIdx = -1;
          nextTrial();
        }

        function nextTrial() {
          trialIdx++;
          if (trialIdx >= targets.length) return finish();
          currentTarget = targets[trialIdx];
          targetEl.textContent = `Target Rating: ${fmt(currentTarget)}`;
          stimulusTime = performance.now();
          firstAttemptRating = null;
          timeToFirstAttempt = null;
          attemptCount = 0;
          renderStars(0, 'var(--preview-color)');
        }

        function finish() {
          mode = 'finished';
          targetEl.textContent = 'The experiment is complete! Thank you.';
          renderStars(0, 'var(--preview-color)');
          downloadCSV(true);
        }

        function clamp(val, min, max) {
          return Math.max(min, Math.min(max, val));
        }

        document.addEventListener("touchstart", (e) => {
          if (e.touches.length === 1) {
            // Start moving the pointer
            lastTouch = { x: e.touches[0].clientX, y: e.touches[0].clientY };
          } else if (e.touches.length === 2) {
            const el = document.elementFromPoint(pointerX, pointerY);
            const rating = currentHoveredRating;

            if (!el || rating == null) return;

            if ((el.classList.contains('star') || isPointerOverStarOrGap()) && rating) {
              // Evaluation is executed even when hovered
              if (mode === 'practice') {
                practiceCounter++;
                addRow({
                  trial: 'P' + practiceCounter,
                  mode: 'practice',
                  target: '-',
                  firstAttempt: rating,
                  attempts: 1,
                  overshoot: '-',
                  t1st: '-',
                  tCorrect: '-'
                });
                renderStars(rating, getComputedStyle(wrap).getPropertyValue('--confirm-color'));
              } else if (mode === 'main') {
                attemptCount++;
                if (firstAttemptRating === null) {
                  firstAttemptRating = rating;
                  timeToFirstAttempt = Math.round(performance.now() - stimulusTime);
                }
                if (rating === currentTarget) {
                  const tCorrect = Math.round(performance.now() - stimulusTime);
                  const overshoot = Math.abs(firstAttemptRating - currentTarget);
                  renderStars(rating, getComputedStyle(wrap).getPropertyValue('--confirm-color'));
                  showToast('That\'s correct!', true);
                  const log = {
                    trial: trialIdx + 1,
                    mode: 'main',
                    target: currentTarget,
                    firstAttempt: firstAttemptRating,
                    attempts: attemptCount,
                    overshoot,
                    t1st: timeToFirstAttempt,
                    tCorrect
                  };
                  logsMain.push(log);
                  addRow(log);
                  setTimeout(nextTrial, DELAY_NEXT);
                } else {
                  renderStars(rating, getComputedStyle(wrap).getPropertyValue('--preview-color'));
                  showToast('Try again!', false);
                }
              }
            } else {
              el.click?.();
            }
          }
        }, { passive: false });

        document.addEventListener("touchmove", (e) => {
          if (e.touches.length !== 1) return;
          e.preventDefault();

          const touch = e.touches[0];
          const dx = touch.clientX - lastTouch.x;
          const dy = touch.clientY - lastTouch.y;

          pointerX = clamp(pointerX + dx, 0, window.innerWidth);
          pointerY = clamp(pointerY + dy, 0, window.innerHeight);

          pointer.style.left = `${pointerX}px`;
          pointer.style.top = `${pointerY}px`;

          const el = document.elementFromPoint(pointerX, pointerY);
          lastElement = el;

          lastTouch = { x: touch.clientX, y: touch.clientY };
        }, { passive: false });

        document.addEventListener("touchend", (e) => {
          // Click is not processed (Clicking is possible only with two-finger tap)
        });

        startBtn.addEventListener('click', beginMain);
        renderStars(0, 'var(--preview-color)');
      })();
    </script>
  </body>
</html>
