<!DOCTYPE html> 
<html lang="en">
<head>
  <meta charset="utf-8" />
  <title>Medium-BPP Kodak-comparison</title>
  <meta name="viewport" content="width=device-width, initial-scale=1" />
  <link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap.min.css" crossorigin="anonymous">
  <style>
    body { background:#fafafa; }
    .container { max-width: 1800px; }
    .page-title { margin: 24px 0 8px; font-weight: 600; }
    .subtitle { margin-top: 0; color:#666; }

    .card { border: 1px solid #eaeaea; border-radius: 10px; margin-bottom: 18px; background:#fff; }
    .card-head { padding: 10px 12px; border-bottom:1px dashed #eee; }
    .img-index { color:#999; font-weight:600; margin-right: 8px; }
    .img-name { font-weight:600; }

    .method-title{
      font-size: 14px;
      font-weight: 600;
      margin: 6px 0 4px;
      white-space: normal;
      overflow-wrap: anywhere;
      word-break: break-word;
    }
    .img-wrap { position: relative; overflow: hidden; background:#f9f9f9; border:1px solid #eee; border-radius:8px; padding: 0; cursor: zoom-in; }
    .img-wrap img { display:block; image-rendering: pixelated; width: 100%; height: auto; }
    .row-gap { margin-left: -6px; margin-right: -6px; }
    .row-gap > [class*='col-'] { padding-left: 6px; padding-right: 6px; }

    .toolbar { margin: 8px 0 16px; }
    .legend { font-size: 13px; color:#666; }
    .pager { margin: 10px 0 22px; }
    .pager a, .pager strong { margin-right: 6px; }
    .caption-text { font-size: 12px; color:#444; margin-top: 6px; min-height: 18px; }

    .per-img-caption{
      font-size: 12px;
      color: #444;
      margin-top: 6px;
      min-height: 18px;
    }

    /* Lightbox */
    .lightbox { display: none; position: fixed; z-index: 9999; left: 0; top: 0; width: 100%; height: 100%; background: rgba(0,0,0,0.9); justify-content: center; align-items: center; }
    .lightbox img {
      max-width: 90vw;
      max-height: 90vh;
      width: auto;
      height: auto;
      cursor: zoom-out;
      display: block;
      margin: 0 auto;
    }
  </style>
</head>
<body>
  <div class="container">
    <h1 class="page-title" id="main-title">Kodak Image Comparison — Medium BPP regime (~ 0.185 bpp)</h1>
    <h4 class="subtitle" id="h4-name">Loading…</h4>

    <div class="toolbar row">
      <div class="col-sm-12">
        <form class="form-inline" id="controls">
          <div class="form-group">
            <label for="perPage">Per page</label>
            <input type="number" min="1" class="form-control input-sm" id="perPage" style="width: 80px;" />
          </div>
          <div class="form-group">
            <label for="page">Page</label>
            <input type="number" min="1" class="form-control input-sm" id="page" style="width: 80px;" />
          </div>
          <button type="submit" class="btn btn-primary btn-sm">Go</button>
        </form>
      </div>
    </div>

    <div id="grid"></div>
    <div class="pager" id="pager"></div>
  </div>

  <div class="lightbox" id="lightbox">
    <div style="text-align:center; color:#fff;">
      <img src="" alt="">
      <div id="lightbox-label" style="margin-top:12px; font-size:16px;"></div>
    </div>
  </div>

  <script src="bpp_data.js"></script>

  <script>
    // If image_list.json exists, automatically build imageList
    fetch('image_list.json').then(r=>r.ok?r.json():null).then(json=>{
      if(json){ window.CONFIG = window.CONFIG || {}; window.CONFIG.imageList = json; }
      init();
    }).catch(()=>{ init(); });

    var lightbox = document.getElementById('lightbox');
    var lightboxImg = lightbox.querySelector('img');
    var LB_LIST = [];
    var LB_INDEX = -1;
    function lbUpdate() {
      if (LB_INDEX >= 0 && LB_INDEX < LB_LIST.length) {
        var node = LB_LIST[LB_INDEX];
        lightboxImg.src = node.src;
        var labelBox = document.getElementById('lightbox-label');
        labelBox.textContent = (node.getAttribute('data-label') || '') + ' — ' + (node.getAttribute('data-fname') || '');
      }
    }
    function lbOpenAt(idx) {
      LB_INDEX = idx;
      lbUpdate();
      lightbox.style.display = 'flex';
    }
    function lbNext() { if (LB_LIST.length) { LB_INDEX = (LB_INDEX + 1) % LB_LIST.length; lbUpdate(); } }
    function lbPrev() { if (LB_LIST.length) { LB_INDEX = (LB_INDEX - 1 + LB_LIST.length) % LB_LIST.length; lbUpdate(); } }

    document.addEventListener('keydown', function(e){
      if (lightbox.style.display === 'flex') {
        if (e.key === 'ArrowRight') { e.preventDefault(); lbNext(); }
        else if (e.key === 'ArrowLeft') { e.preventDefault(); lbPrev(); }
        else if (e.key === 'Escape') { lightbox.style.display = 'none'; }
      }
    });

    function init(){
      var DEFAULTS = {
        name: 'Kodak',
        baseDir: 'images',
        methods: [
        { folder: 'originals',        label: 'Original' },
        { folder: 'C3-WD',        label: 'C3/WDs' },
          { folder: 'HiFiC',   label: 'HiFiC' },
          { folder: 'Wavelet-WD',   label: 'ReReIC' },

        ],
        imageList: [
          'kodim01.png','kodim09.png','kodim12.png','kodim14.png',
          'kodim16.png','kodim18.png','kodim20.png',
          'kodim23.png','kodim24.png',
        ],
        captions: {},
        metrics: {}
      };

      var CFG = (window.CONFIG ? Object.assign({}, DEFAULTS, window.CONFIG) : DEFAULTS);

      var url = new URL(window.location.href);
      function gp(name, d){ var v=url.searchParams.get(name); return v?parseInt(v,10):d; }
      var page = gp('page', 1);
      var perPage = gp('perPage', CFG.imageList.length);

      document.getElementById('perPage').value = perPage;
      document.getElementById('page').value = page;
      document.getElementById('controls').addEventListener('submit', function(e){
        e.preventDefault();
        url.searchParams.set('perPage', document.getElementById('perPage').value);
        url.searchParams.set('page', document.getElementById('page').value);
        window.location.href = url.toString();
      });

      document.getElementById('h4-name').textContent = 'Methods: Original | MLIC++ (0.175) | VTM (0.182) | HiFiC (0.183) | C3/WDs (0.183) | Re2IC (0.183)';

      function render(){
        var grid = document.getElementById('grid'); grid.innerHTML='';
        var start = (page - 1) * perPage; var end = Math.min(start + perPage, CFG.imageList.length);

        CFG.imageList.slice(start, end).forEach(function(fname, idx){
          var card = document.createElement('div'); card.className='card';
          var head = document.createElement('div'); head.className='card-head';
          head.innerHTML = '<span class="img-index">Image '+(start+idx+1)+':</span><span class="img-name">'+fname.replace(/\.[^.]+$/, '')+'</span>';
          card.appendChild(head);

          var row = null;
          CFG.methods.forEach(function(m, i){
            if (i % 2 === 0) {
              row = document.createElement('div'); row.className = 'row row-gap';
              card.appendChild(row);
            }
            row.appendChild(makeCol(m, fname));
          });

          var captionText = document.createElement('div');
          captionText.className='caption-text';
          captionText.textContent = CFG.captions[fname] || '';
          card.appendChild(captionText);

          grid.appendChild(card);
        });

        LB_LIST = Array.prototype.slice.call(document.querySelectorAll('.img-wrap img'));
        LB_LIST.forEach(function(node, i){ node.setAttribute('data-lb-index', i); });

        renderPager();
      }

      function makeCol(method, fname){
        var col = document.createElement('div');
        col.className = 'col-xs-12 col-sm-6 col-md-6 col-lg-6';

        var title = document.createElement('div');
        title.className = 'method-title';
        title.textContent = method.label;
        col.appendChild(title);

        var wrap = document.createElement('div');
        wrap.className = 'img-wrap';

        if (method.folder) {
          var img = document.createElement('img');
          img.src = [CFG.baseDir, method.folder, fname].join('/');
          img.setAttribute('data-label', method.label);
          img.setAttribute('data-fname', fname);
          wrap.appendChild(img);
          col.appendChild(wrap);

          wrap.addEventListener('click', function(){
            var idx = parseInt(img.getAttribute('data-lb-index'), 10);
            lbOpenAt(idx);
          });
        } else {
          wrap.style.minHeight = "150px";
          wrap.style.background = "#f5f5f5";
          wrap.textContent = "";
          col.appendChild(wrap);
        }

        var txt = (CFG.metrics && CFG.metrics[method.folder] && CFG.metrics[method.folder][fname]) || '';
        var meta = document.createElement('div');
        meta.className = 'per-img-caption';
        meta.textContent = txt || '\u00A0';
        col.appendChild(meta);

        return col;
      }

      function renderPager(){
        var pager = document.getElementById('pager'); pager.innerHTML='';
        var numPages = Math.ceil(CFG.imageList.length / perPage); if (numPages<=1) return;
        for (var p=1;p<=numPages;p++){
          if (p===page){
            var strong=document.createElement('strong'); strong.textContent=p; pager.appendChild(strong);
          } else {
            var a=document.createElement('a'); var u=new URL(window.location.href);
            u.searchParams.set('page', p); u.searchParams.set('perPage', perPage);
            a.href=u.toString(); a.textContent=p; pager.appendChild(a);
          }
          if (p<numPages) pager.appendChild(document.createTextNode(' '));
        }
      }

      document.getElementById('lightbox').addEventListener('click', function(){ this.style.display='none'; });

      render();
    }
  </script>
</body>
</html>
