Add renderThumbnails() function that creates page buttons with SVG
thumbnails fetched from /api/page/{i}/thumbnail, with lazy loading via
Intersection Observer for performance on large documents.
Changes:
- app.js: Add renderThumbnails() with click navigation and lazy loading
- style.css: Increase sidebar width to 250px, thumbnail-img to 200px
Acceptance criteria:
- Sidebar shows page buttons with thumbnail images
- Click navigates main view and updates URL fragment
- Lazy loading for 100-page documents (<3s load)
- Active page highlighting via .active class
- Cross-browser compatible (standard APIs)
See notes/pdftract-2z88j.md for verification details.
65 lines
5.6 KiB
CSS
65 lines
5.6 KiB
CSS
*{box-sizing:border-box;margin:0;padding:0}
|
|
body{font-family:system-ui,-apple-system,sans-serif;font-size:14px;line-height:1.5;background:#f5f5f5;color:#333;height:100vh;overflow:hidden}
|
|
.app{display:flex;height:100vh}
|
|
.sidebar{width:250px;background:#fff;border-right:1px solid #ddd;display:flex;flex-direction:column}
|
|
.sidebar-header{padding:12px;border-bottom:1px solid #ddd;font-weight:600;background:#f9f9f9}
|
|
.thumbnails{flex:1;overflow-y:auto;padding:8px;display:flex;flex-direction:column;gap:8px}
|
|
.thumbnail{padding:8px;background:#f9f9f9;border:1px solid #ddd;border-radius:4px;cursor:pointer;transition:background .15s}
|
|
.thumbnail:hover{background:#e8f4ff}
|
|
.thumbnail.active{background:#0078d4;color:#fff;border-color:#005a9e}
|
|
.thumbnail-img{width:200px;height:auto;background:#fff;border:1px solid #eee;margin-bottom:4px}
|
|
.thumbnail-number{font-size:12px;font-weight:500}
|
|
.main{flex:1;display:flex;flex-direction:column;overflow:hidden}
|
|
.toolbar{padding:8px 12px;background:#fff;border-bottom:1px solid #ddd;display:flex;gap:8px;align-items:center;flex-wrap:wrap}
|
|
.btn{padding:6px 12px;background:#fff;border:1px solid #ddd;border-radius:4px;cursor:pointer;font-size:13px;font-family:inherit}
|
|
.btn:hover{background:#f0f0f0}
|
|
.btn:active{background:#e0e0e0}
|
|
.btn:disabled{opacity:.5;cursor:not-allowed}
|
|
.search-input{flex:1;max-width:300px;padding:6px 10px;border:1px solid #ddd;border-radius:4px;font-size:13px;font-family:inherit}
|
|
.search-input:focus{outline:none;border-color:#0078d4;box-shadow:0 0 0 2px rgba(0,120,212,.2)}
|
|
.match-count{font-size:12px;color:#666;white-space:nowrap;padding:0 8px}
|
|
.toggles{display:flex;gap:4px;flex-wrap:wrap}
|
|
.layer-toggle{padding:4px 8px;background:#fff;border:1px solid #ddd;border-radius:3px;cursor:pointer;font-size:11px;font-family:inherit;white-space:nowrap}
|
|
.layer-toggle:hover{background:#f0f0f0}
|
|
.layer-toggle.active{background:#0078d4;color:#fff;border-color:#005a9e}
|
|
.canvas-container{flex:1;overflow:auto;background:#e0e0e0;display:flex;justify-content:center;align-items:flex-start;padding:20px;position:relative}
|
|
#page-svg{background:#fff;box-shadow:0 2px 8px rgba(0,0,0,.1)}
|
|
.panel{width:280px;background:#fff;border-left:1px solid #ddd;display:flex;flex-direction:column}
|
|
.panel-header{padding:12px;border-bottom:1px solid #ddd;font-weight:600;background:#f9f9f9}
|
|
.json-tree{flex:1;overflow:auto;padding:12px;font-size:12px;font-family:ui-monospace,monospace}
|
|
.json-tree details{margin-left:12px;margin-bottom:2px}
|
|
.json-tree summary{cursor:pointer;font-size:12px;padding:2px 4px;border-radius:2px;outline:none;user-select:none}
|
|
.json-tree summary:hover{background:#f0f0f0}
|
|
.json-leaf{padding:2px 4px;margin-left:16px;font-size:12px}
|
|
.json-key{color:#8f8}
|
|
.json-value{color:#8cf}
|
|
.span-entry{padding:4px 8px;margin:2px 0;border-radius:3px;font-size:12px;cursor:pointer;transition:background .15s}
|
|
.span-entry:hover{background:#f5f5f5}
|
|
.span-entry.highlighted{background:#ffff3b;animation:json-highlight 2s ease-out}
|
|
.span-index{color:#666;font-size:11px;margin-right:4px}
|
|
.span-text{font-weight:500;color:#333}
|
|
.span-meta{color:#888;font-size:11px;margin-left:6px}
|
|
.block-entry{padding:4px 8px;margin:2px 0;font-size:12px;color:#666}
|
|
@keyframes json-highlight{0%{background:#ffff00}100%{background:#ffff3b}}
|
|
.loading{position:absolute;top:50%;left:50%;transform:translate(-50%,-50%);font-size:16px;color:#666}
|
|
.tooltip{position:absolute;background:rgba(255,255,255,.95);border:1px solid #ccc;padding:6px 10px;font-family:ui-monospace,SFMono-Regular,SF Mono,Menlo,Consolas,monospace;font-size:12px;pointer-events:none;z-index:1000;max-width:400px;white-space:pre;line-height:1.4;transition:opacity 0s}
|
|
.layer-spans,.layer-blocks,.layer-columns,.layer-reading-order,.layer-confidence-heatmap,.layer-ocr,.layer-ocr_regions,.layer-mcid,.layer-anchors,.layer-diff{display:none}
|
|
html[data-layers~="spans"] .layer-spans,html[data-layers~="blocks"] .layer-blocks,html[data-layers~="columns"] .layer-columns,html[data-layers~="reading-order"] .layer-reading-order,html[data-layers~="confidence-heatmap"] .layer-confidence-heatmap,html[data-layers~="ocr"] .layer-ocr,html[data-layers~="ocr_regions"] .layer-ocr_regions,html[data-layers~="mcid"] .layer-mcid,html[data-layers~="anchors"] .layer-anchors,html[data-layers~="diff"] .layer-diff{display:block}
|
|
.tooltip-key{color:#8f8}
|
|
.tooltip-value{color:#8cf}
|
|
.tooltip-number{color:#f8c}
|
|
.search-highlight{background:#ffeb3b;outline:2px solid #ff9800}
|
|
.search-match{outline:2px solid #ff9800;outline-offset:2px}
|
|
.search-match.active{outline:3px solid #ff6f00;outline-offset:3px;outline-style:double}
|
|
.search-match-found{animation:highlight-pulse 1s ease-out}
|
|
@keyframes highlight-pulse{0%{background:#ff9800}100%{background:#ffeb3b}}
|
|
.compare-container{display:flex;gap:20px;justify-content:center;align-items:flex-start;width:100%;height:100%}
|
|
.compare-side{flex:1;max-width:50%;display:flex;flex-direction:column;align-items:center;background:#fff;box-shadow:0 2px 8px rgba(0,0,0,.1);border-radius:4px;overflow:hidden}
|
|
.compare-label{padding:8px 16px;background:#f9f9f9;border-bottom:1px solid #ddd;font-weight:600;font-size:13px;width:100%;text-align:center}
|
|
.svg-wrapper{flex:1;overflow:auto;position:relative;min-height:400px}
|
|
.comparison-controls{display:flex;gap:12px;align-items:center;padding-left:12px;border-left:1px solid #ddd;margin-left:8px}
|
|
.sync-toggle{display:flex;align-items:center;gap:6px;font-size:12px;cursor:pointer;user-select:none}
|
|
.sync-toggle input{cursor:pointer}
|
|
.diff-removed{fill:none;stroke:#dc3545;stroke-width:2;stroke-dasharray:4,2;opacity:.7}
|
|
.diff-added{fill:none;stroke:#28a745;stroke-width:2;stroke-dasharray:4,2;opacity:.7}
|
|
.diff-changed{fill:none;stroke:#ffc107;stroke-width:2;stroke-dasharray:4,2;opacity:.7}
|