A vibe coded tangled fork which supports pijul.
1{{ define "title" }}
2 {{ .Pull.Title }} · pull #{{ .Pull.PullId }} · {{ .RepoInfo.FullName }}
3{{ end }}
4
5{{ define "extrameta" }}
6 {{ template "repo/pulls/fragments/og" (dict "RepoInfo" .RepoInfo "Pull" .Pull) }}
7{{ end }}
8
9{{ define "mainLayout" }}
10 <div class="px-1 flex-grow flex flex-col gap-4">
11 <div class="max-w-full md:max-w-screen-lg mx-auto">
12 {{ block "contentLayout" . }}
13 {{ block "content" . }}{{ end }}
14 {{ end }}
15 </div>
16 {{ block "contentAfterLayout" . }}
17 <main>
18 {{ block "contentAfter" . }}{{ end }}
19 </main>
20 {{ end }}
21 </div>
22 <script>
23 (function() {
24 const details = document.getElementById('bottomSheet');
25 const backdrop = document.getElementById('bottomSheetBackdrop');
26 const isDesktop = () => window.matchMedia('(min-width: 768px)').matches;
27
28 // function to update backdrop
29 const updateBackdrop = () => {
30 if (backdrop) {
31 if (details.open && !isDesktop()) {
32 backdrop.classList.remove('opacity-0', 'pointer-events-none');
33 backdrop.classList.add('opacity-100', 'pointer-events-auto');
34 document.body.style.overflow = 'hidden';
35 } else {
36 backdrop.classList.remove('opacity-100', 'pointer-events-auto');
37 backdrop.classList.add('opacity-0', 'pointer-events-none');
38 document.body.style.overflow = '';
39 }
40 }
41 };
42
43 // close on mobile initially
44 if (!isDesktop()) {
45 details.open = false;
46 }
47 updateBackdrop(); // initialize backdrop
48
49 // prevent closing on desktop
50 details.addEventListener('toggle', function(e) {
51 if (isDesktop() && !this.open) {
52 this.open = true;
53 }
54 updateBackdrop();
55 });
56
57 const mediaQuery = window.matchMedia('(min-width: 768px)');
58 mediaQuery.addEventListener('change', function(e) {
59 if (e.matches) {
60 // switched to desktop - keep open
61 details.open = true;
62 } else {
63 // switched to mobile - close
64 details.open = false;
65 }
66 updateBackdrop();
67 });
68
69 // close when clicking backdrop
70 if (backdrop) {
71 backdrop.addEventListener('click', () => {
72 if (!isDesktop()) {
73 details.open = false;
74 }
75 });
76 }
77 })();
78 </script>
79{{ end }}
80
81{{ define "repoContentLayout" }}
82 <div class="grid grid-cols-1 md:grid-cols-10 gap-4 w-full">
83 <section class="bg-white col-span-1 md:col-span-8 dark:bg-gray-800 p-6 rounded relative w-full mx-auto dark:text-white h-full flex-shrink">
84 {{ block "repoContent" . }}{{ end }}
85 </section>
86 <div class="flex flex-col gap-6 col-span-1 md:col-span-2">
87 {{ template "repo/fragments/labelPanel"
88 (dict "RepoInfo" $.RepoInfo
89 "Defs" $.LabelDefs
90 "Subject" $.Pull.AtUri
91 "State" $.Pull.Labels) }}
92 {{ template "repo/fragments/participants" $.Pull.Participants }}
93 {{ template "repo/fragments/backlinks"
94 (dict "RepoInfo" $.RepoInfo
95 "Backlinks" $.Backlinks) }}
96 {{ template "repo/fragments/externalLinkPanel" $.Pull.AtUri }}
97 </div>
98 </div>
99{{ end }}
100
101{{ define "contentAfter" }}
102 {{ template "repo/fragments/diff" (list .Diff .DiffOpts $) }}
103{{ end }}
104
105{{ define "repoContent" }}
106 {{ template "repo/pulls/fragments/pullHeader" . }}
107 {{ if .Pull.IsStacked }}
108 <div class="mt-8">
109 {{ template "repo/pulls/fragments/pullStack" . }}
110 </div>
111 {{ end }}
112{{ end }}
113
114{{ define "resize-grip" }}
115 {{ $id := index . 0 }}
116 {{ $target := index . 1 }}
117 {{ $direction := index . 2 }}
118 <div id="{{ $id }}"
119 data-resizer="vertical"
120 data-target="{{ $target }}"
121 data-direction="{{ $direction }}"
122 class="resizer-vertical hidden md:flex w-4 sticky top-12 max-h-screen flex-col items-center justify-center group">
123 <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>
124 </div>
125{{ end }}
126
127{{ define "diffLayout" }}
128 {{ $diff := index . 0 }}
129 {{ $opts := index . 1 }}
130 {{ $root := index . 2 }}
131
132 <div class="flex col-span-full">
133 <!-- left panel -->
134 <div id="files" class="w-0 hidden md:block overflow-hidden sticky top-12 max-h-screen overflow-y-auto pb-12">
135 <section class="overflow-x-auto text-sm px-6 py-2 border-b border-x border-gray-200 dark:border-gray-700 w-full mx-auto min-h-full rounded-b rounded-t-none bg-white dark:bg-gray-800 drop-shadow-sm">
136 {{ template "repo/fragments/fileTree" $diff.FileTree }}
137 </section>
138 </div>
139
140 {{ template "resize-grip" (list "resize-files" "files" "before") }}
141
142 <!-- main content -->
143 <div id="diff-files" class="flex-1 min-w-0 sticky top-12 pb-12">
144 {{ template "diffFiles" (list $diff $opts) }}
145 </div>
146
147 {{ template "resize-grip" (list "resize-subs" "subs" "after") }}
148
149 <!-- right panel -->
150 {{ template "subsPanel" $ }}
151 </div>
152{{ end }}
153
154{{ define "subsPanel" }}
155 {{ $root := index . 2 }}
156 {{ $pull := $root.Pull }}
157 {{ $bgColor := "bg-gray-600 dark:bg-gray-700" }}
158
159 {{ if $pull.State.IsOpen }}
160 {{ $bgColor = "bg-green-600 dark:bg-green-700" }}
161 {{ else if $pull.State.IsMerged }}
162 {{ $bgColor = "bg-purple-600 dark:bg-purple-700" }}
163 {{ else if $pull.State.IsDeleted }}
164 {{ $bgColor = "bg-red-600 dark:bg-red-700" }}
165 {{ end }}
166
167 <!-- backdrop overlay - only visible on mobile when open -->
168 <div id="bottomSheetBackdrop" class="fixed inset-0 bg-black/50 md:hidden opacity-0 pointer-events-none transition-opacity duration-300 z-40"></div>
169 <!-- right panel - bottom sheet on mobile, side panel on desktop -->
170 <div id="subs" class="fixed bottom-0 left-0 right-0 z-50 w-full md:static md:z-auto md:max-h-screen md:sticky md:top-12 overflow-hidden">
171 <details open id="bottomSheet" class="rounded-t-2xl md:rounded-t drop-shadow-lg md:drop-shadow-none group/panel">
172 <summary class="
173 flex gap-4 items-center justify-between
174 rounded-t-2xl md:rounded-t cursor-pointer list-none p-4 md:h-12
175 text-white md:text-black md:dark:text-white
176 {{ $bgColor }}
177 md:bg-white md:dark:bg-gray-800
178 drop-shadow-sm border-t md:border-x border-gray-200 dark:border-gray-700">
179 <h2 class="">History</h2>
180 {{ template "subsPanelSummary" $ }}
181 </summary>
182 <div class="max-h-[85vh] md:max-h-[calc(100vh-3rem-3rem)] w-full flex flex-col-reverse gap-4 overflow-y-auto bg-slate-100 dark:bg-gray-900 md:bg-transparent">
183 {{ template "submissions" $root }}
184 </div>
185 </details>
186 </div>
187{{ end }}
188
189{{ define "subsPanelSummary" }}
190 {{ $root := index . 2 }}
191 {{ $pull := $root.Pull }}
192 {{ $rounds := len $pull.Submissions }}
193 {{ $comments := $pull.TotalComments }}
194 <div class="flex items-center gap-2 text-sm">
195 <span>
196 {{ $rounds }} round{{ if ne $rounds 1 }}s{{ end }}
197 </span>
198 <span class="select-none before:content-['\00B7']"></span>
199 <span>
200 {{ $comments }} comment{{ if ne $comments 1 }}s{{ end }}
201 </span>
202
203 <span class="md:hidden inline">
204 <span class="inline group-open:hidden">{{ i "chevron-up" "size-4" }}</span>
205 <span class="hidden group-open:inline">{{ i "chevron-down" "size-4" }}</span>
206 </span>
207 </div>
208{{ end }}
209
210{{ define "subsCheckbox" }}
211 <input type="checkbox" id="subsToggle" class="peer/subs hidden" checked/>
212{{ end }}
213
214{{ define "subsToggle" }}
215 <style>
216 #subsToggle:checked ~ div div#subs {
217 width: 100%;
218 margin-left: 0;
219 }
220 #subsToggle:checked ~ div label[for="subsToggle"] .show-toggle { display: none; }
221 #subsToggle:checked ~ div label[for="subsToggle"] .hide-toggle { display: flex; }
222 #subsToggle:not(:checked) ~ div label[for="subsToggle"] .hide-toggle { display: none; }
223
224 @media (min-width: 768px) {
225 #subsToggle:checked ~ div div#subs {
226 width: 25vw;
227 max-width: 50vw;
228 }
229 #subsToggle:not(:checked) ~ div div#subs {
230 width: 0;
231 display: none;
232 margin-left: 0;
233 }
234 #subsToggle:not(:checked) ~ div div#resize-subs {
235 display: none;
236 }
237 }
238 </style>
239 <label title="Toggle review panel" for="subsToggle" class="hidden md:flex items-center justify-end rounded cursor-pointer">
240 <span class="show-toggle">{{ i "message-square-more" "size-4" }}</span>
241 <span class="hide-toggle w-[25vw] flex justify-end">{{ i "message-square" "size-4" }}</span>
242 </label>
243{{ end }}
244
245
246{{ define "submissions" }}
247 {{ $lastIdx := sub (len .Pull.Submissions) 1 }}
248 {{ if not .LoggedInUser }}
249 {{ template "loginPrompt" $ }}
250 {{ end }}
251 {{ range $ridx, $item := reverse .Pull.Submissions }}
252 {{ $idx := sub $lastIdx $ridx }}
253 {{ template "submission" (list $item $idx $lastIdx $) }}
254 {{ end }}
255{{ end }}
256
257{{ define "submission" }}
258 {{ $item := index . 0 }}
259 {{ $idx := index . 1 }}
260 {{ $lastIdx := index . 2 }}
261 {{ $root := index . 3 }}
262 {{ $round := $item.RoundNumber }}
263 <div class="
264 w-full shadow-sm bg-gray-50 dark:bg-gray-900 border border-t-0
265 {{ if eq $round 0 }}rounded-b{{ else }}rounded{{ end }}
266 {{ if eq $round $root.ActiveRound }}
267 border-blue-200 dark:border-blue-700
268 {{ else }}
269 border-gray-200 dark:border-gray-700
270 {{ end }}
271 ">
272 {{ template "submissionHeader" $ }}
273 {{ template "submissionComments" $ }}
274 </div>
275{{ end }}
276
277{{ define "submissionHeader" }}
278 {{ $item := index . 0 }}
279 {{ $lastIdx := index . 2 }}
280 {{ $root := index . 3 }}
281 {{ $round := $item.RoundNumber }}
282 <div class="
283 {{ if ne $round 0 }}rounded-t{{ end }}
284 px-6 py-4 pr-2 pt-2
285 bg-white dark:bg-gray-800
286
287 {{ if eq $round $root.ActiveRound }}
288 border-t border-t-blue-200 dark:border-t-blue-700
289 {{ else }}
290 border-t border-t-gray-200 dark:border-t-gray-700
291 {{ end }}
292
293 flex gap-2 sticky top-0 z-20">
294 <!-- left column: just profile picture -->
295 <div class="flex-shrink-0 pt-2">
296 {{ template "user/fragments/picLink" (list $root.Pull.OwnerDid "size-8") }}
297 </div>
298 <!-- right column -->
299 <div class="flex-1 min-w-0 flex flex-col gap-1">
300 {{ template "submissionInfo" $ }}
301 {{ template "submissionCommits" $ }}
302 {{ template "submissionPipeline" $ }}
303 {{ if eq $lastIdx $round }}
304 {{ block "mergeCheck" $root }} {{ end }}
305 {{ end }}
306 </div>
307 </div>
308{{ end }}
309
310{{ define "submissionInfo" }}
311 {{ $item := index . 0 }}
312 {{ $idx := index . 1 }}
313 {{ $root := index . 3 }}
314 {{ $round := $item.RoundNumber }}
315 <div class="flex gap-2 items-center justify-between mb-1">
316 <span class="inline-flex items-center gap-2 text-sm text-gray-500 dark:text-gray-400 pt-2">
317 {{ $handle := resolve $root.Pull.OwnerDid }}
318 <a class="text-gray-500 dark:text-gray-400 hover:text-gray-500 dark:hover:text-gray-300" href="/{{ $handle }}">{{ $handle }}</a>
319 submitted
320 <span class="px-2 py-0.5 text-black dark:text-white bg-gray-100 dark:bg-gray-700 border-gray-300 dark:border-gray-600 rounded font-mono text-xs border">
321 #{{ $round }}
322 </span>
323 <span class="select-none before:content-['\00B7']"></span>
324 <a class="text-gray-500 dark:text-gray-400 hover:text-gray-500" href="#round-#{{ $round }}">
325 {{ template "repo/fragments/shortTime" $item.Created }}
326 </a>
327 </span>
328 <div class="flex gap-2 items-center">
329 {{ if or $root.IsInterdiff (ne $root.ActiveRound $round) }}
330 <a class="btn-flat flex items-center gap-2 no-underline hover:no-underline text-sm"
331 href="/{{ $root.RepoInfo.FullName }}/pulls/{{ $root.Pull.PullId }}/round/{{ $round }}?{{ safeUrl $root.DiffOpts.Encode }}#round-#{{ $round }}">
332 {{ i "diff" "w-4 h-4" }}
333 diff
334 </a>
335 {{ end }}
336 {{ if and (ne $idx 0) (or (not $root.IsInterdiff) (ne $root.ActiveRound $round)) }}
337 <a class="btn-flat flex items-center gap-2 no-underline hover:no-underline text-sm"
338 href="/{{ $root.RepoInfo.FullName }}/pulls/{{ $root.Pull.PullId }}/round/{{ $round }}/interdiff?{{ safeUrl $root.DiffOpts.Encode }}">
339 {{ i "chevrons-left-right-ellipsis" "w-4 h-4 rotate-90" }}
340 interdiff
341 </a>
342 {{ end }}
343 </div>
344 </div>
345{{ end }}
346
347{{ define "submissionCommits" }}
348 {{ $item := index . 0 }}
349 {{ $root := index . 3 }}
350 {{ $round := $item.RoundNumber }}
351 {{ $patches := $item.AsFormatPatch }}
352 {{ if $patches }}
353 <details class="group/commit">
354 <summary class="list-none cursor-pointer flex items-center gap-2">
355 <span>{{ i "git-commit-horizontal" "w-4 h-4" }}</span>
356 {{ len $patches }} commit{{ if ne (len $patches) 1 }}s{{ end }}
357 <div class="text-sm text-gray-500 dark:text-gray-400">
358 <span class="group-open/commit:hidden inline">expand</span>
359 <span class="hidden group-open/commit:inline">collapse</span>
360 </div>
361 </summary>
362 {{ range $patches }}
363 {{ template "submissionCommit" (list . $item $root) }}
364 {{ end }}
365 </details>
366 {{ end }}
367{{ end }}
368
369{{ define "submissionCommit" }}
370 {{ $patch := index . 0 }}
371 {{ $item := index . 1 }}
372 {{ $root := index . 2 }}
373 {{ $round := $item.RoundNumber }}
374 {{ with $patch }}
375 <div id="commit-{{.SHA}}" class="py-1 relative w-full md:max-w-3/5 md:w-fit flex flex-col text-gray-600 dark:text-gray-300">
376 <div class="flex items-baseline gap-2">
377 <div class="text-xs">
378 <!-- attempt to resolve $fullRepo: this is possible only on non-deleted forks and branches -->
379 {{ $fullRepo := "" }}
380 {{ if and $root.Pull.IsForkBased $root.Pull.PullSource.Repo }}
381 {{ $fullRepo = printf "%s/%s" $root.Pull.OwnerDid $root.Pull.PullSource.Repo.Name }}
382 {{ else if $root.Pull.IsBranchBased }}
383 {{ $fullRepo = $root.RepoInfo.FullName }}
384 {{ end }}
385
386 <!-- if $fullRepo was resolved, link to it, otherwise just span without a link -->
387 {{ if $fullRepo }}
388 <a href="/{{ $fullRepo }}/commit/{{ .SHA }}" class="font-mono text-gray-600 dark:text-gray-300">{{ slice .SHA 0 8 }}</a>
389 {{ else }}
390 <span class="font-mono">{{ slice .SHA 0 8 }}</span>
391 {{ end }}
392 </div>
393
394 <div>
395 <span>{{ .Title | description }}</span>
396 {{ if gt (len .Body) 0 }}
397 <button
398 class="py-1/2 px-1 mx-2 bg-gray-200 hover:bg-gray-400 rounded dark:bg-gray-700 dark:hover:bg-gray-600"
399 hx-on:click="document.getElementById('body-{{$round}}-{{.SHA}}').classList.toggle('hidden')"
400 >
401 {{ i "ellipsis" "w-3 h-3" }}
402 </button>
403 {{ end }}
404 {{ if gt (len .Body) 0 }}
405 <p id="body-{{$round}}-{{.SHA}}" class="hidden mt-1 pb-2">{{ nl2br .Body }}</p>
406 {{ end }}
407 </div>
408 </div>
409 </div>
410 {{ end }}
411{{ end }}
412
413{{ define "mergeCheck" }}
414 {{ $isOpen := .Pull.State.IsOpen }}
415 {{ if and $isOpen .MergeCheck .MergeCheck.Error }}
416 <div class="flex items-center gap-2">
417 {{ i "triangle-alert" "w-4 h-4 text-red-600 dark:text-red-500" }}
418 {{ .MergeCheck.Error }}
419 </div>
420 {{ else if and $isOpen .MergeCheck .MergeCheck.IsConflicted }}
421 <details class="group/conflict">
422 <summary class="flex items-center justify-between cursor-pointer list-none">
423 <div class="flex items-center gap-2 ">
424 {{ i "triangle-alert" "text-red-600 dark:text-red-500 w-4 h-4" }}
425 <span class="font-medium">merge conflicts detected</span>
426 <div class="text-sm text-gray-500 dark:text-gray-400">
427 <span class="group-open/conflict:hidden inline">expand</span>
428 <span class="hidden group-open/conflict:inline">collapse</span>
429 </div>
430 </div>
431 </summary>
432 {{ if gt (len .MergeCheck.Conflicts) 0 }}
433 <ul class="space-y-1 mt-2 overflow-x-auto">
434 {{ range .MergeCheck.Conflicts }}
435 {{ if .Filename }}
436 <li class="flex items-center whitespace-nowrap">
437 {{ i "file-warning" "inline-flex w-4 h-4 mr-1.5 text-red-600 dark:text-red-500 flex-shrink-0" }}
438 <span class="font-mono">{{ .Filename }}</span>
439 </li>
440 {{ else if .Reason }}
441 <li class="flex items-center whitespace-nowrap">
442 {{ i "file-warning" "w-4 h-4 mr-1.5 text-red-600 dark:text-red-500 " }}
443 <span>{{.Reason}}</span>
444 </li>
445 {{ end }}
446 {{ end }}
447 </ul>
448 {{ end }}
449 </details>
450 {{ else if and $isOpen .MergeCheck }}
451 <div class="flex items-center gap-2">
452 {{ i "check" "w-4 h-4 text-green-600 dark:text-green-500" }}
453 <span>no conflicts, ready to merge</span>
454 </div>
455 {{ end }}
456{{ end }}
457
458{{ define "mergeStatus" }}
459 {{ if .Pull.State.IsClosed }}
460 <div class="bg-gray-50 dark:bg-gray-700 border border-black dark:border-gray-500 rounded drop-shadow-sm px-6 py-2 relative">
461 <div class="flex items-center gap-2 text-black dark:text-white">
462 {{ i "ban" "w-4 h-4" }}
463 <span class="font-medium">closed without merging</span
464 >
465 </div>
466 </div>
467 {{ else if .Pull.State.IsMerged }}
468 <div class="bg-purple-50 dark:bg-purple-900 border border-purple-500 rounded drop-shadow-sm px-6 py-2 relative">
469 <div class="flex items-center gap-2 text-purple-500 dark:text-purple-300">
470 {{ i "git-merge" "w-4 h-4" }}
471 <span class="font-medium">pull request successfully merged</span
472 >
473 </div>
474 </div>
475 {{ else if .Pull.State.IsDeleted }}
476 <div class="bg-red-50 dark:bg-red-900 border border-red-500 rounded drop-shadow-sm px-6 py-2 relative">
477 <div class="flex items-center gap-2 text-red-500 dark:text-red-300">
478 {{ i "git-pull-request-closed" "w-4 h-4" }}
479 <span class="font-medium">This pull has been deleted (possibly by jj abandon or jj squash)</span>
480 </div>
481 </div>
482 {{ end }}
483{{ end }}
484
485{{ define "resubmitStatus" }}
486 {{ if .ResubmitCheck.Yes }}
487 <div class="bg-amber-50 dark:bg-amber-900 border border-amber-500 rounded drop-shadow-sm px-6 py-2 relative">
488 <div class="flex items-center gap-2 text-amber-500 dark:text-amber-300">
489 {{ i "triangle-alert" "w-4 h-4" }}
490 <span class="font-medium">this branch has been updated, consider resubmitting</span>
491 </div>
492 </div>
493 {{ end }}
494{{ end }}
495
496{{ define "submissionPipeline" }}
497 {{ $item := index . 0 }}
498 {{ $root := index . 3 }}
499 {{ $pipeline := index $root.Pipelines $item.SourceRev }}
500 {{ with $pipeline }}
501 {{ $id := .Id }}
502 {{ if .Statuses }}
503 <details class="group/pipeline">
504 <summary class="cursor-pointer list-none flex items-center gap-2">
505 {{ template "repo/pipelines/fragments/pipelineSymbol" (dict "Pipeline" $pipeline "ShortSummary" false) }}
506 <div class="text-sm text-gray-500 dark:text-gray-400">
507 <span class="group-open/pipeline:hidden inline">expand</span>
508 <span class="hidden group-open/pipeline:inline">collapse</span>
509 </div>
510 </summary>
511 <div class="my-2 grid grid-cols-1 bg-white dark:bg-gray-800 rounded border border-gray-200 dark:border-gray-700 divide-y divide-gray-200 dark:divide-gray-700">
512 {{ range $name, $all := .Statuses }}
513 <a href="/{{ $root.RepoInfo.FullName }}/pipelines/{{ $id }}/workflow/{{ $name }}" class="no-underline hover:no-underline hover:bg-gray-100/25 hover:dark:bg-gray-700/25">
514 <div
515 class="flex gap-2 items-center justify-between p-2">
516 {{ $lastStatus := $all.Latest }}
517 {{ $kind := $lastStatus.Status.String }}
518
519 <div id="left" class="flex items-center gap-2 flex-shrink-0">
520 {{ template "repo/pipelines/fragments/workflowSymbol" $all }}
521 {{ $name }}
522 </div>
523 <div id="right" class="flex items-center gap-2 flex-shrink-0">
524 <span class="font-bold">{{ $kind }}</span>
525 {{ if .TimeTaken }}
526 {{ template "repo/fragments/duration" .TimeTaken }}
527 {{ else }}
528 {{ template "repo/fragments/shortTimeAgo" $lastStatus.Created }}
529 {{ end }}
530 </div>
531 </div>
532 </a>
533 {{ end }}
534 </div>
535 </details>
536 {{ end }}
537 {{ end }}
538{{ end }}
539
540{{ define "submissionComments" }}
541 {{ $item := index . 0 }}
542 {{ $idx := index . 1 }}
543 {{ $lastIdx := index . 2 }}
544 {{ $root := index . 3 }}
545 {{ $round := $item.RoundNumber }}
546 {{ $c := len $item.Comments }}
547 <details class="relative ml-10 group/comments" {{ if or (eq $c 0) (eq $root.ActiveRound $round) }}open{{ end }}>
548 <summary class="cursor-pointer list-none">
549 <div class="hidden group-open/comments:block absolute -left-8 top-0 bottom-0 w-16 transition-colors flex items-center justify-center group/border z-4">
550 <div class="absolute left-1/2 -translate-x-1/2 top-0 bottom-0 w-0.5 group-open/comments:bg-gray-200 dark:group-open/comments:bg-gray-700 group-hover/border:bg-gray-400 dark:group-hover/border:bg-gray-500 transition-colors"> </div>
551 </div>
552 <div class="group-open/comments:hidden block relative group/summary py-4">
553 <div class="absolute -left-8 top-0 bottom-0 w-16 transition-colors flex items-center justify-center z-4">
554 <div class="absolute left-1/2 -translate-x-1/2 h-1/3 top-0 bottom-0 w-0.5 bg-gray-200 dark:bg-gray-700 group-hover/summary:bg-gray-400 dark:group-hover/summary:bg-gray-500 transition-colors"></div>
555 </div>
556 <span class="text-gray-500 dark:text-gray-400 text-sm group-hover/summary:text-gray-600 dark:group-hover/summary:text-gray-300 transition-colors flex items-center gap-2 -ml-2 relative">
557 {{ i "circle-plus" "size-4 z-5" }}
558 expand {{ $c }} comment{{ if ne $c 1 }}s{{ end }}
559 </span>
560 </div>
561 </summary>
562 <div>
563 {{ range $item.Comments }}
564 {{ template "submissionComment" . }}
565 {{ end }}
566 </div>
567
568 <div class="relative -ml-10">
569 {{ if eq $lastIdx $item.RoundNumber }}
570 {{ block "mergeStatus" $root }} {{ end }}
571 {{ block "resubmitStatus" $root }} {{ end }}
572 {{ end }}
573 </div>
574 <div class="relative -ml-10 bg-gray-50 dark:bg-gray-900">
575 {{ if $root.LoggedInUser }}
576 {{ template "repo/pulls/fragments/pullActions"
577 (dict
578 "LoggedInUser" $root.LoggedInUser
579 "Pull" $root.Pull
580 "RepoInfo" $root.RepoInfo
581 "RoundNumber" $item.RoundNumber
582 "MergeCheck" $root.MergeCheck
583 "ResubmitCheck" $root.ResubmitCheck
584 "BranchDeleteStatus" $root.BranchDeleteStatus
585 "Stack" $root.Stack) }}
586 {{ end }}
587 </div>
588 </details>
589{{ end }}
590
591{{ define "submissionComment" }}
592 <div id="comment-{{.ID}}" class="flex gap-2 -ml-4 py-4 w-full mx-auto">
593 <!-- left column: profile picture -->
594 <div class="flex-shrink-0 h-fit relative">
595 {{ template "user/fragments/picLink" (list .OwnerDid "size-8") }}
596 </div>
597 <!-- right column: name and body in two rows -->
598 <div class="flex-1 min-w-0">
599 <!-- Row 1: Author and timestamp -->
600 <div class="text-sm text-gray-500 dark:text-gray-400 flex items-center gap-1">
601 {{ $handle := resolve .OwnerDid }}
602 <a class="text-gray-500 dark:text-gray-400 hover:text-gray-500 dark:hover:text-gray-300" href="/{{ $handle }}">{{ $handle }}</a>
603 <span class="before:content-['·']"></span>
604 <a class="text-gray-500 dark:text-gray-400 hover:text-gray-500 dark:hover:text-gray-300" href="#comment-{{.ID}}">
605 {{ template "repo/fragments/shortTime" .Created }}
606 </a>
607 </div>
608 <!-- Row 2: Body text -->
609 <div class="prose dark:prose-invert mt-1">
610 {{ .Body | markdown }}
611 </div>
612 </div>
613 </div>
614{{ end }}
615
616{{ define "loginPrompt" }}
617 <div class="bg-amber-50 dark:bg-amber-900 border border-amber-500 rounded drop-shadow-sm p-2 relative flex gap-2 items-center">
618 <a href="/signup" class="btn-create py-0 hover:no-underline hover:text-white flex items-center gap-2">
619 sign up
620 </a>
621 <span class="text-gray-500 dark:text-gray-400">or</span>
622 <a href="/login" class="underline">login</a>
623 to add to the discussion
624 </div>
625{{ end }}