semdiff-output 0.4.2

HTML, JSON, and summary report outputs for semdiff.
Documentation
<!doctype html>
<html lang="ja">
  <head>
    <meta charset="utf-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1" />
    <title>semdiff report</title>
    <style>
      :root {
        color-scheme: light dark;
        --status-neutral-bg: #f9fafb;
        --status-neutral-surface: #ffffff;
        --status-neutral-border: #e5e7eb;
        --status-neutral-text: #4b5563;
        --status-neutral-text-muted: #6b7280;
        --status-neutral-text-subtle: #9ca3af;
        --status-neutral-link: #2563eb;
        --status-neutral-shadow: #0000000A;
        --status-unchanged-bg: #ecfeff;
        --status-unchanged-text: #155e75;
        --status-unchanged-border: #a5f3fc;
        --status-modified-bg: #fffbeb;
        --status-modified-text: #92400e;
        --status-modified-border: #fde68a;
        --status-added-bg: #ecfdf5;
        --status-added-text: #065f46;
        --status-added-border: #a7f3d0;
        --status-deleted-bg: #fef2f2;
        --status-deleted-text: #991b1b;
        --status-deleted-border: #fecaca;
        --diff-accent-1: #22c55e;
        --diff-accent-2: #0ea5e9;
        --diff-accent-3: #ef4444;
      }
      @media (prefers-color-scheme: dark) {
        :root {
          --status-neutral-bg: #0f172a;
          --status-neutral-surface: #182338;
          --status-neutral-border: #2b374c;
          --status-neutral-text: #cbd5e1;
          --status-neutral-text-muted: #94a3b8;
          --status-neutral-text-subtle: #64748b;
          --status-neutral-link: #60a5fa;
          --status-neutral-shadow: #00000080;
          --status-unchanged-bg: #083344;
          --status-unchanged-text: #67e8f9;
          --status-unchanged-border: #164e63;
          --status-modified-bg: #451a03;
          --status-modified-text: #fbbf24;
          --status-modified-border: #78350f;
          --status-added-bg: #052e16;
          --status-added-text: #6ee7b7;
          --status-added-border: #065f46;
          --status-deleted-bg: #450a0a;
          --status-deleted-text: #fca5a5;
          --status-deleted-border: #7f1d1d;
          --diff-accent-1: #22c55e;
          --diff-accent-2: #38bdf8;
          --diff-accent-3: #f87171;
        }
      }
      body {
        font-family: "Segoe UI", system-ui, -apple-system, sans-serif;
        margin: 0;
        padding: 1.5rem;
        background: var(--status-neutral-bg);
        color: var(--status-neutral-text);
      }
      .tint {
        display: inline-block;
        overflow: hidden;
        max-width: 100%;
      }
      .tint__img {
        width: 100%;
        height: auto;
        display: block;
        filter: drop-shadow(9999px 0 0 var(--c));
        transform: translateX(-9999px);
      }
      .tint--waveform {
        --c: var(--diff-accent-1);
      }
      .tint--spectrogram {
        --c: var(--diff-accent-2);
      }
      .tint--spectrogram-diff {
        --c: var(--diff-accent-3);
      }
      .tint--image-diff {
        --c: var(--diff-accent-3);
      }
      h1 {
        margin: 0 0 0.75rem;
      }
      .summary {
        display: flex;
        gap: 0.75rem;
        flex-wrap: wrap;
        margin-bottom: 1.25rem;
      }
      .summary-button {
        background: var(--status-neutral-surface);
        border: 1px solid var(--status-neutral-border);
        border-radius: 0.5rem;
        padding: 0.65rem 0.9rem;
        min-width: 8.5rem;
        box-shadow: 0 0.0625rem 0.125rem var(--status-neutral-shadow);
        display: inline-flex;
        align-items: center;
        justify-content: space-between;
        gap: 0.5rem;
        cursor: pointer;
        color: inherit;
        font: inherit;
      }
      .summary-button[aria-pressed="true"] {
        outline: 0.125rem solid var(--status-neutral-link);
        outline-offset: 0.125rem;
      }
      .summary-label {
        text-transform: capitalize;
      }
      .summary-count {
        font-weight: 700;
      }
      .summary-button:focus-visible {
        outline: 0.125rem solid var(--status-neutral-link);
        outline-offset: 0.125rem;
      }
      .entry-groups {
        display: grid;
        grid-template-columns: repeat(auto-fill, minmax(15rem, 1fr));
        row-gap: 1.5rem;
      }
      .entry-group {
        display: grid;
        gap: 0.75rem;
        grid-column: 1 / -1;
        grid-template-columns: repeat(auto-fill, minmax(15rem, 1fr));
      }
      .entry-group-header {
        display: flex;
        align-items: center;
        gap: 0.5rem;
        grid-column: 1 / -1;
      }
      .entry-group-title {
        margin: 0;
        font-size: 1.05rem;
        text-transform: capitalize;
      }
      .entry-group.is-hidden {
        display: none;
      }
      .entries {
        display: grid;
        grid-template-columns: repeat(auto-fill, minmax(15rem, 1fr));
        gap: 1rem;
        grid-column: 1 / -1;
      }
      @supports (grid-template-columns: subgrid) {
        .entry-group {
          grid-template-columns: subgrid;
        }
        .entries {
          grid-template-columns: subgrid;
        }
      }
      .entry {
        background: var(--status-neutral-surface);
        border: 1px solid var(--status-neutral-border);
        border-radius: 0.625rem;
        padding: 1rem;
        padding-top: 3.25rem;
        box-shadow: 0 0.0625rem 0.125rem var(--status-neutral-shadow);
        aspect-ratio: 1 / 1;
        display: flex;
        flex-direction: column;
        gap: 0.5rem;
        position: relative;
      }
      .entry-badges {
        position: absolute;
        top: -0.0625rem;
        left: -0.0625rem;
        display: flex;
        gap: 0.375rem;
        padding: 0.375rem 0.625rem;
        background: var(--status-neutral-surface);
        border-top: 1px solid var(--status-neutral-border);
        border-left: 1px solid var(--status-neutral-border);
        border-right: 1px solid var(--status-neutral-border);
        border-bottom: 1px solid var(--status-neutral-border);
        border-top-left-radius: 0.625rem;
        border-bottom-right-radius: 0.625rem;
      }
      .entry-link {
        color: inherit;
        text-decoration: none;
      }
      .entry-link[href]:hover {
        box-shadow: 0 0.125rem 0.375rem var(--status-neutral-shadow);
        outline: 0.125rem solid var(--status-neutral-link);
        outline-offset: 0;
        text-decoration: none;
      }
      .entry-link[href]:focus-visible {
        outline: 0.125rem solid var(--status-neutral-link);
        outline-offset: 0.125rem;
      }
      .entry-header {
        display: flex;
        justify-content: space-between;
        gap: 0.75rem;
        margin-bottom: 0.5rem;
      }
      .entry-name {
        font-weight: 600;
        word-break: break-word;
      }
      .entry-link[href]:hover .entry-name {
        text-decoration: underline;
      }
      .badge {
        padding: 0.125rem 0.5rem;
        border-radius: 999em;
        font-size: 0.75rem;
        font-weight: 600;
        border: 1px solid transparent;
      }
      .badge.unchanged {
        background: var(--status-unchanged-bg);
        color: var(--status-unchanged-text);
        border-color: var(--status-unchanged-border);
      }
      .badge.modified {
        background: var(--status-modified-bg);
        color: var(--status-modified-text);
        border-color: var(--status-modified-border);
      }
      .badge.added {
        background: var(--status-added-bg);
        color: var(--status-added-text);
        border-color: var(--status-added-border);
      }
      .badge.deleted {
        background: var(--status-deleted-bg);
        color: var(--status-deleted-text);
        border-color: var(--status-deleted-border);
      }
      .preview {
        font-family: "Segoe UI", system-ui, -apple-system, sans-serif;
        font-size: 1rem;
        color: var(--status-neutral-text-muted);
        flex: 1;
        overflow: hidden;
        display: flex;
        flex-direction: column;
        gap: 0.5rem;
      }
      .preview-box {
        background: var(--status-neutral-bg);
        border: 1px solid var(--status-neutral-border);
        border-radius: 0.5rem;
        padding: 0.5rem;
        flex: 1;
        display: flex;
        flex-direction: column;
        justify-content: center;
        min-height: 0;
      }
      .preview-media {
        width: 100%;
        height: 100%;
        border-radius: 0.375rem;
        overflow: hidden;
        background: var(--status-neutral-bg);
        display: flex;
        align-items: stretch;
        justify-content: flex-start;
        min-height: 0;
      }
      a { color: var(--status-neutral-link); text-decoration: none; }
      a:hover { text-decoration: underline; }
    </style>
  </head>
  <body>
    <h1>semdiff Report</h1>
    <div class="summary">
      <button class="summary-button" type="button" data-status-filter="all" aria-pressed="true">
        <span class="summary-label">all</span>
        <span class="summary-count">{{ total }}</span>
      </button>
      <button class="summary-button" type="button" data-status-filter="modified" aria-pressed="false">
        <span class="badge modified summary-label">modified</span>
        <span class="summary-count">{{ modified }}</span>
      </button>
      <button class="summary-button" type="button" data-status-filter="deleted" aria-pressed="false">
        <span class="badge deleted summary-label">deleted</span>
        <span class="summary-count">{{ deleted }}</span>
      </button>
      <button class="summary-button" type="button" data-status-filter="added" aria-pressed="false">
        <span class="badge added summary-label">added</span>
        <span class="summary-count">{{ added }}</span>
      </button>
      <button class="summary-button" type="button" data-status-filter="unchanged" aria-pressed="false">
        <span class="badge unchanged summary-label">unchanged</span>
        <span class="summary-count">{{ unchanged }}</span>
      </button>
    </div>
    <div class="entry-groups">
      {% for group in entry_groups %}
      {% if group.entries.len() > 0 %}
      <section class="entry-group" data-status-group="{{ group.status_class }}">
        <div class="entry-group-header">
          <h2 class="entry-group-title" id="status-{{ group.status_class }}">{{ group.status_label }}</h2>
          <span class="badge {{ group.status_class }}">{{ group.entries.len() }}</span>
        </div>
        <div class="entries">
          {% for entry in group.entries %}
          <a class="entry entry-link" href="{{ entry.detail_link }}" data-status-entry="{{ entry.status_class }}">
            <div class="entry-badges">
              <span class="badge {{ entry.status_class }}">{{ entry.status_label }}</span>
              <span class="badge">{{ entry.compares }}</span>
            </div>
            <div class="entry-header">
              <div class="entry-name">{{ entry.name }}</div>
            </div>
            <div class="preview">
              <div class="preview-box">
                <div class="preview-media">{{ entry.preview_html | safe }}</div>
              </div>
            </div>
          </a>
          {% endfor %}
        </div>
      </section>
      {% endif %}
      {% endfor %}
    </div>
    <script>
      (() => {
        const buttons = document.querySelectorAll('[data-status-filter]');
        const groups = document.querySelectorAll('[data-status-group]');
        if (buttons.length === 0 || groups.length === 0) {
          return;
        }
        const applyFilter = (status) => {
          buttons.forEach((button) => {
            const active = button.dataset.statusFilter === status;
            button.setAttribute('aria-pressed', active ? 'true' : 'false');
          });
          groups.forEach((group) => {
            const visible = status === 'all' || group.dataset.statusGroup === status;
            group.classList.toggle('is-hidden', !visible);
          });
        };
        buttons.forEach((button) => {
          button.addEventListener('click', () => applyFilter(button.dataset.statusFilter));
        });
        applyFilter('all');
      })();
    </script>
  </body>
</html>