A vibe coded tangled fork which supports pijul.
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 }}