A vibe coded tangled fork which supports pijul.
at master 301 lines 12 kB view raw
1{{ define "repo/fragments/diff" }} 2 <style> 3 #filesToggle:checked ~ div label[for="filesToggle"] .show-text { display: none; } 4 #filesToggle:checked ~ div label[for="filesToggle"] .hide-text { display: inline; } 5 #filesToggle:not(:checked) ~ div label[for="filesToggle"] .hide-text { display: none; } 6 #filesToggle:checked ~ div div#files { width: fit-content; max-width: 15vw; } 7 #filesToggle:not(:checked) ~ div div#files { width: 0; display: none; margin-right: 0; } 8 #filesToggle:not(:checked) ~ div div#resize-files { display: none; } 9 </style> 10 11 <div id="diff-area"> 12 {{ template "diffTopbar" . }} 13 {{ block "diffLayout" . }} {{ end }} 14 {{ template "fragments/resizable" }} 15 {{ template "activeFileHighlight" }} 16 {{ template "fragments/line-quote-button" }} 17 </div> 18{{ end }} 19 20{{ define "diffTopbar" }} 21 {{ $diff := index . 0 }} 22 {{ $opts := index . 1 }} 23 {{ $root := "" }} 24 {{ if gt (len .) 2 }} 25 {{ $root = index . 2 }} 26 {{ end }} 27 28 {{ block "filesCheckbox" $ }} {{ end }} 29 {{ block "subsCheckbox" $ }} {{ end }} 30 31 <!-- top bar --> 32 <div class="sticky top-0 z-30 bg-slate-100 dark:bg-gray-900 flex items-center gap-2 col-span-full h-12 p-2 {{ if $root }}mt-4{{ end }}"> 33 <!-- left panel toggle --> 34 {{ template "filesToggle" . }} 35 36 <!-- stats --> 37 {{ $stat := $diff.Stats }} 38 {{ $count := len $diff.ChangedFiles }} 39 {{ template "repo/fragments/diffStatPill" $stat }} 40 <span class="text-xs text-gray-600 dark:text-gray-400 hidden md:inline-flex">{{ $count }} changed file{{ if ne $count 1 }}s{{ end }}</span> 41 42 {{ if $root }} 43 {{ if $root.IsInterdiff }} 44 <!-- interdiff indicator --> 45 <div class="flex items-center gap-2 before:content-['|'] before:text-gray-300 dark:before:text-gray-600 before:mr-2"> 46 <span class="text-xs text-gray-600 dark:text-gray-400 uppercase tracking-wide">Interdiff</span> 47 <a 48 href="/{{ $root.RepoInfo.FullName }}/pulls/{{ $root.Pull.PullId }}/round/{{ sub $root.ActiveRound 1 }}" 49 class="px-2 py-0.5 bg-white dark:bg-gray-700 rounded font-mono text-xs hover:bg-gray-50 dark:hover:bg-gray-600 border border-gray-300 dark:border-gray-600" 50 > 51 #{{ sub $root.ActiveRound 1 }} 52 </a> 53 <span class="text-gray-400 text-xs"></span> 54 <a 55 href="/{{ $root.RepoInfo.FullName }}/pulls/{{ $root.Pull.PullId }}/round/{{ $root.ActiveRound }}" 56 class="px-2 py-0.5 bg-white dark:bg-gray-700 rounded font-mono text-xs hover:bg-gray-50 dark:hover:bg-gray-600 border border-gray-300 dark:border-gray-600" 57 > 58 #{{ $root.ActiveRound }} 59 </a> 60 </div> 61 {{ else if ne $root.ActiveRound nil }} 62 <!-- diff round indicator --> 63 <div class="flex items-center gap-2 before:content-['|'] before:text-gray-300 dark:before:text-gray-600 before:mr-2"> 64 <span class="text-xs text-gray-600 dark:text-gray-400 uppercase tracking-wide">Diff</span> 65 <span class="px-2 py-0.5 bg-white dark:bg-gray-700 rounded font-mono text-xs border border-gray-300 dark:border-gray-600"> 66 <span class="hidden md:inline">round </span>#{{ $root.ActiveRound }} 67 </span> 68 </div> 69 {{ end }} 70 {{ end }} 71 72 <!-- spacer --> 73 <div class="flex-grow"></div> 74 75 <!-- collapse diffs --> 76 {{ template "collapseToggle" }} 77 78 <!-- diff options --> 79 {{ template "repo/fragments/diffOpts" $opts }} 80 81 <!-- right panel toggle --> 82 {{ block "subsToggle" $ }} {{ end }} 83 </div> 84 85{{ end }} 86 87{{ define "resize-grip" }} 88 {{ $id := index . 0 }} 89 {{ $target := index . 1 }} 90 {{ $direction := index . 2 }} 91 <div id="{{ $id }}" 92 data-resizer="vertical" 93 data-target="{{ $target }}" 94 data-direction="{{ $direction }}" 95 class="resizer-vertical hidden md:flex w-4 sticky top-12 max-h-screen flex-col items-center justify-center group"> 96 <div class="w-1 h-16 group-hover:h-24 group-[.resizing]:h-24 transition-all rounded-full bg-gray-400 dark:bg-gray-500 group-hover:bg-gray-500 group-hover:dark:bg-gray-400"></div> 97 </div> 98{{ end }} 99 100{{ define "diffLayout" }} 101 {{ $diff := index . 0 }} 102 {{ $opts := index . 1 }} 103 104 <div class="flex col-span-full flex-grow"> 105 <!-- left panel --> 106 <div id="files" class="w-0 hidden md:block overflow-hidden sticky top-12 max-h-screen overflow-y-auto pb-12"> 107 <section class="overflow-x-auto text-sm px-6 py-2 border border-gray-200 dark:border-gray-700 w-full mx-auto min-h-full rounded bg-white dark:bg-gray-800 drop-shadow-sm"> 108 {{ template "repo/fragments/fileTree" $diff.FileTree }} 109 </section> 110 </div> 111 112 {{ template "resize-grip" (list "resize-files" "files" "before") }} 113 114 <!-- main content --> 115 <div id="diff-files" class="flex-1 min-w-0 sticky top-12 pb-12"> 116 {{ template "diffFiles" (list $diff $opts) }} 117 </div> 118 119 </div> 120{{ end }} 121 122{{ define "diffFiles" }} 123 {{ $diff := index . 0 }} 124 {{ $opts := index . 1 }} 125 {{ $files := $diff.ChangedFiles }} 126 {{ $isSplit := $opts.Split }} 127 <div class="flex flex-col gap-4"> 128 {{ if eq (len $files) 0 }} 129 <div class="text-center text-gray-500 dark:text-gray-400 py-8"> 130 <p>No differences found between the selected revisions.</p> 131 </div> 132 {{ else }} 133 {{ range $idx, $file := $files }} 134 {{ template "diffFile" (list $idx $file $isSplit) }} 135 {{ end }} 136 {{ end }} 137 </div> 138{{ end }} 139 140{{ define "diffFile" }} 141 {{ $idx := index . 0 }} 142 {{ $file := index . 1 }} 143 {{ $isSplit := index . 2 }} 144 {{ $isGenerated := false }} 145 {{ with $file }} 146 {{ $n := .Names }} 147 {{ if $n.New }} 148 {{ $isGenerated = isGenerated $n.New }} 149 {{ else if $n.Old }} 150 {{ $isGenerated = isGenerated $n.Old }} 151 {{ end }} 152 <details {{ if not $isGenerated }}open{{ end }} id="file-{{ .Id }}" class="group border border-gray-200 dark:border-gray-700 w-full mx-auto rounded bg-white dark:bg-gray-800 drop-shadow-sm" tabindex="{{ add $idx 1 }}"> 153 <summary class="list-none cursor-pointer sticky top-12 group-open:border-b border-gray-200 dark:border-gray-700"> 154 <div id="diff-file-header" class="rounded cursor-pointer bg-white dark:bg-gray-800 flex justify-between"> 155 <div id="left-side-items" class="p-2 flex gap-2 items-center overflow-x-auto"> 156 <span class="group-open:hidden inline">{{ i "chevron-right" "w-4 h-4" }}</span> 157 <span class="hidden group-open:inline">{{ i "chevron-down" "w-4 h-4" }}</span> 158 {{ template "repo/fragments/diffStatPill" .Stats }} 159 160 <div class="flex gap-2 items-center overflow-x-auto"> 161 {{ if and $n.New $n.Old (ne $n.New $n.Old)}} 162 {{ $n.Old }} {{ i "arrow-right" "w-4 h-4" }} {{ $n.New }} 163 {{ else if $n.New }} 164 {{ $n.New }} 165 {{ else }} 166 {{ $n.Old }} 167 {{ end }} 168 {{ if $isGenerated }} 169 <span class="text-gray-400 dark:text-gray-500" title="Generated files are collapsed by default"> 170 {{ i "circle-question-mark" "size-4" }} 171 </span> 172 {{ end }} 173 </div> 174 </div> 175 </div> 176 </summary> 177 178 <div class="transition-all duration-700 ease-in-out"> 179 {{ $reason := .CanRender }} 180 {{ if $reason }} 181 <p class="text-center text-gray-400 dark:text-gray-500 p-4">{{ $reason }}</p> 182 {{ else }} 183 {{ if $isSplit }} 184 {{- template "repo/fragments/splitDiff" .Split -}} 185 {{ else }} 186 {{- template "repo/fragments/unifiedDiff" . -}} 187 {{ end }} 188 {{- end -}} 189 </div> 190 </details> 191 {{ end }} 192{{ end }} 193 194{{ define "filesCheckbox" }} 195 <input type="checkbox" id="filesToggle" class="peer/files hidden" checked/> 196{{ end }} 197 198{{ define "filesToggle" }} 199 <label title="Toggle filetree panel" for="filesToggle" class="hidden md:inline-flex items-center justify-center rounded cursor-pointer text-normal font-normal normalcase"> 200 <span class="show-text">{{ i "panel-left-open" "size-4" }}</span> 201 <span class="hide-text">{{ i "panel-left-close" "size-4" }}</span> 202 </label> 203{{ end }} 204 205{{ define "collapseToggle" }} 206 <label 207 title="Expand/Collapse diffs" 208 for="collapseToggle" 209 class="btn font-normal normal-case p-2" 210 > 211 <input type="checkbox" id="collapseToggle" class="peer/collapse hidden" checked/> 212 <span class="peer-checked/collapse:hidden inline-flex items-center gap-2"> 213 {{ i "fold-vertical" "w-4 h-4" }} 214 <span class="hidden md:inline">expand all</span> 215 </span> 216 <span class="peer-checked/collapse:inline-flex hidden flex items-center gap-2"> 217 {{ i "unfold-vertical" "w-4 h-4" }} 218 <span class="hidden md:inline">collapse all</span> 219 </span> 220 </label> 221 <script> 222 (() => { 223 const checkbox = document.getElementById('collapseToggle'); 224 const diffArea = document.getElementById('diff-area'); 225 226 checkbox.addEventListener('change', () => { 227 document.querySelectorAll('details[id^="file-"]').forEach(detail => { 228 detail.open = checkbox.checked; 229 }); 230 }); 231 232 if (window.__collapseToggleHandler) { 233 diffArea.removeEventListener('toggle', window.__collapseToggleHandler, true); 234 } 235 236 const handler = (e) => { 237 if (!e.target.matches('details[id^="file-"]')) return; 238 const details = document.querySelectorAll('details[id^="file-"]'); 239 const allOpen = Array.from(details).every(d => d.open); 240 const allClosed = Array.from(details).every(d => !d.open); 241 242 if (allOpen) checkbox.checked = true; 243 else if (allClosed) checkbox.checked = false; 244 }; 245 246 window.__collapseToggleHandler = handler; 247 diffArea.addEventListener('toggle', handler, true); 248 })(); 249 </script> 250{{ end }} 251 252{{ define "activeFileHighlight" }} 253 <script> 254 (() => { 255 if (window.__activeFileScrollHandler) { 256 document.removeEventListener('scroll', window.__activeFileScrollHandler); 257 } 258 259 const filetreeLinks = document.querySelectorAll('.filetree-link'); 260 if (filetreeLinks.length === 0) return; 261 262 const linkMap = new Map(); 263 filetreeLinks.forEach(link => { 264 const path = link.getAttribute('data-path'); 265 if (path) linkMap.set('file-' + path, link); 266 }); 267 268 let currentActive = null; 269 const setActive = (link) => { 270 if (link && link !== currentActive) { 271 if (currentActive) currentActive.classList.remove('font-bold'); 272 link.classList.add('font-bold'); 273 currentActive = link; 274 } 275 }; 276 277 filetreeLinks.forEach(link => { 278 link.addEventListener('click', () => setActive(link)); 279 }); 280 281 const topbar = document.querySelector('.sticky.top-0.z-30'); 282 const headerHeight = topbar ? topbar.offsetHeight : 0; 283 284 const updateActiveFile = () => { 285 const diffFiles = document.querySelectorAll('details[id^="file-"]'); 286 Array.from(diffFiles).some(file => { 287 const rect = file.getBoundingClientRect(); 288 if (rect.top <= headerHeight && rect.bottom > headerHeight) { 289 setActive(linkMap.get(file.id)); 290 return true; 291 } 292 return false; 293 }); 294 }; 295 296 window.__activeFileScrollHandler = updateActiveFile; 297 document.addEventListener('scroll', updateActiveFile); 298 updateActiveFile(); 299 })(); 300 </script> 301{{ end }}