A vibe coded tangled fork which supports pijul.
at sl/tap-appview 625 lines 24 kB view raw
1{{ define "title" }} 2 {{ .Pull.Title }} &middot; pull #{{ .Pull.PullId }} &middot; {{ .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 }}