mirror of
				https://github.com/go-gitea/gitea
				synced 2025-10-26 00:48:29 +00:00 
			
		
		
		
	Merge branch 'main' into lunny/automerge_support_delete_branch
This commit is contained in:
		| @@ -1443,11 +1443,11 @@ func ViewIssue(ctx *context.Context) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if issue.IsPull && !ctx.Repo.CanRead(unit.TypeIssues) { | 	if issue.IsPull && !ctx.Repo.CanRead(unit.TypeIssues) { | ||||||
| 		ctx.Data["IssueType"] = "pulls" | 		ctx.Data["IssueDependencySearchType"] = "pulls" | ||||||
| 	} else if !issue.IsPull && !ctx.Repo.CanRead(unit.TypePullRequests) { | 	} else if !issue.IsPull && !ctx.Repo.CanRead(unit.TypePullRequests) { | ||||||
| 		ctx.Data["IssueType"] = "issues" | 		ctx.Data["IssueDependencySearchType"] = "issues" | ||||||
| 	} else { | 	} else { | ||||||
| 		ctx.Data["IssueType"] = "all" | 		ctx.Data["IssueDependencySearchType"] = "all" | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	ctx.Data["IsProjectsEnabled"] = ctx.Repo.CanRead(unit.TypeProjects) | 	ctx.Data["IsProjectsEnabled"] = ctx.Repo.CanRead(unit.TypeProjects) | ||||||
|   | |||||||
| @@ -318,9 +318,10 @@ func pushUpdateAddTags(ctx context.Context, repo *repo_model.Repository, gitRepo | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	releases, err := db.Find[repo_model.Release](ctx, repo_model.FindReleasesOptions{ | 	releases, err := db.Find[repo_model.Release](ctx, repo_model.FindReleasesOptions{ | ||||||
| 		RepoID:      repo.ID, | 		RepoID:        repo.ID, | ||||||
| 		TagNames:    tags, | 		TagNames:      tags, | ||||||
| 		IncludeTags: true, | 		IncludeDrafts: true, | ||||||
|  | 		IncludeTags:   true, | ||||||
| 	}) | 	}) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return fmt.Errorf("db.Find[repo_model.Release]: %w", err) | 		return fmt.Errorf("db.Find[repo_model.Release]: %w", err) | ||||||
| @@ -407,13 +408,17 @@ func pushUpdateAddTags(ctx context.Context, repo *repo_model.Repository, gitRepo | |||||||
|  |  | ||||||
| 			newReleases = append(newReleases, rel) | 			newReleases = append(newReleases, rel) | ||||||
| 		} else { | 		} else { | ||||||
| 			rel.Title = parts[0] |  | ||||||
| 			rel.Note = note |  | ||||||
| 			rel.Sha1 = commit.ID.String() | 			rel.Sha1 = commit.ID.String() | ||||||
| 			rel.CreatedUnix = timeutil.TimeStamp(createdAt.Unix()) | 			rel.CreatedUnix = timeutil.TimeStamp(createdAt.Unix()) | ||||||
| 			rel.NumCommits = commitsCount | 			rel.NumCommits = commitsCount | ||||||
| 			if rel.IsTag && author != nil { | 			if rel.IsTag { | ||||||
| 				rel.PublisherID = author.ID | 				rel.Title = parts[0] | ||||||
|  | 				rel.Note = note | ||||||
|  | 				if author != nil { | ||||||
|  | 					rel.PublisherID = author.ID | ||||||
|  | 				} | ||||||
|  | 			} else { | ||||||
|  | 				rel.IsDraft = false | ||||||
| 			} | 			} | ||||||
| 			if err = repo_model.UpdateRelease(ctx, rel); err != nil { | 			if err = repo_model.UpdateRelease(ctx, rel); err != nil { | ||||||
| 				return fmt.Errorf("Update: %w", err) | 				return fmt.Errorf("Update: %w", err) | ||||||
|   | |||||||
| @@ -156,7 +156,7 @@ func (d discordConvertor) Push(p *api.PushPayload) (DiscordPayload, error) { | |||||||
| 	// for each commit, generate attachment text | 	// for each commit, generate attachment text | ||||||
| 	for i, commit := range p.Commits { | 	for i, commit := range p.Commits { | ||||||
| 		// limit the commit message display to just the summary, otherwise it would be hard to read | 		// limit the commit message display to just the summary, otherwise it would be hard to read | ||||||
| 		message := strings.TrimRight(strings.SplitN(commit.Message, "\n", 1)[0], "\r") | 		message := strings.TrimRight(strings.SplitN(commit.Message, "\n", 2)[0], "\r") | ||||||
|  |  | ||||||
| 		// a limit of 50 is set because GitHub does the same | 		// a limit of 50 is set because GitHub does the same | ||||||
| 		if utf8.RuneCountInString(message) > 50 { | 		if utf8.RuneCountInString(message) > 50 { | ||||||
|   | |||||||
| @@ -80,12 +80,26 @@ func TestDiscordPayload(t *testing.T) { | |||||||
| 		assert.Equal(t, p.Sender.AvatarURL, pl.Embeds[0].Author.IconURL) | 		assert.Equal(t, p.Sender.AvatarURL, pl.Embeds[0].Author.IconURL) | ||||||
| 	}) | 	}) | ||||||
|  |  | ||||||
| 	t.Run("PushWithLongCommitMessage", func(t *testing.T) { | 	t.Run("PushWithMultilineCommitMessage", func(t *testing.T) { | ||||||
| 		p := pushTestMultilineCommitMessagePayload() | 		p := pushTestMultilineCommitMessagePayload() | ||||||
|  |  | ||||||
| 		pl, err := dc.Push(p) | 		pl, err := dc.Push(p) | ||||||
| 		require.NoError(t, err) | 		require.NoError(t, err) | ||||||
|  |  | ||||||
|  | 		assert.Len(t, pl.Embeds, 1) | ||||||
|  | 		assert.Equal(t, "[test/repo:test] 2 new commits", pl.Embeds[0].Title) | ||||||
|  | 		assert.Equal(t, "[2020558](http://localhost:3000/test/repo/commit/2020558fe2e34debb818a514715839cabd25e778) chore: This is a commit summary - user1\n[2020558](http://localhost:3000/test/repo/commit/2020558fe2e34debb818a514715839cabd25e778) chore: This is a commit summary - user1", pl.Embeds[0].Description) | ||||||
|  | 		assert.Equal(t, p.Sender.UserName, pl.Embeds[0].Author.Name) | ||||||
|  | 		assert.Equal(t, setting.AppURL+p.Sender.UserName, pl.Embeds[0].Author.URL) | ||||||
|  | 		assert.Equal(t, p.Sender.AvatarURL, pl.Embeds[0].Author.IconURL) | ||||||
|  | 	}) | ||||||
|  |  | ||||||
|  | 	t.Run("PushWithLongCommitSummary", func(t *testing.T) { | ||||||
|  | 		p := pushTestPayloadWithCommitMessage("This is a commit summary ⚠️⚠️⚠️⚠️ containing 你好 ⚠️⚠️️\n\nThis is the message body") | ||||||
|  |  | ||||||
|  | 		pl, err := dc.Push(p) | ||||||
|  | 		require.NoError(t, err) | ||||||
|  |  | ||||||
| 		assert.Len(t, pl.Embeds, 1) | 		assert.Len(t, pl.Embeds, 1) | ||||||
| 		assert.Equal(t, "[test/repo:test] 2 new commits", pl.Embeds[0].Title) | 		assert.Equal(t, "[test/repo:test] 2 new commits", pl.Embeds[0].Title) | ||||||
| 		assert.Equal(t, "[2020558](http://localhost:3000/test/repo/commit/2020558fe2e34debb818a514715839cabd25e778) This is a commit summary ⚠️⚠️⚠️⚠️ containing 你好... - user1\n[2020558](http://localhost:3000/test/repo/commit/2020558fe2e34debb818a514715839cabd25e778) This is a commit summary ⚠️⚠️⚠️⚠️ containing 你好... - user1", pl.Embeds[0].Description) | 		assert.Equal(t, "[2020558](http://localhost:3000/test/repo/commit/2020558fe2e34debb818a514715839cabd25e778) This is a commit summary ⚠️⚠️⚠️⚠️ containing 你好... - user1\n[2020558](http://localhost:3000/test/repo/commit/2020558fe2e34debb818a514715839cabd25e778) This is a commit summary ⚠️⚠️⚠️⚠️ containing 你好... - user1", pl.Embeds[0].Description) | ||||||
|   | |||||||
| @@ -68,7 +68,7 @@ func pushTestPayload() *api.PushPayload { | |||||||
| } | } | ||||||
|  |  | ||||||
| func pushTestMultilineCommitMessagePayload() *api.PushPayload { | func pushTestMultilineCommitMessagePayload() *api.PushPayload { | ||||||
| 	return pushTestPayloadWithCommitMessage("This is a commit summary ⚠️⚠️⚠️⚠️ containing 你好 ⚠️⚠️️\n\nThis is the message body.") | 	return pushTestPayloadWithCommitMessage("chore: This is a commit summary\n\nThis is a commit description.") | ||||||
| } | } | ||||||
|  |  | ||||||
| func pushTestPayloadWithCommitMessage(message string) *api.PushPayload { | func pushTestPayloadWithCommitMessage(message string) *api.PushPayload { | ||||||
|   | |||||||
| @@ -17,12 +17,12 @@ | |||||||
| 		{{end}} | 		{{end}} | ||||||
| 	</div> | 	</div> | ||||||
| 	<div class="right-links" role="group" aria-label="{{ctx.Locale.Tr "aria.footer.links"}}"> | 	<div class="right-links" role="group" aria-label="{{ctx.Locale.Tr "aria.footer.links"}}"> | ||||||
| 		<div class="ui dropdown upward language"> | 		<div class="ui dropdown upward"> | ||||||
| 			<span class="flex-text-inline">{{svg "octicon-globe" 14}} {{ctx.Locale.LangName}}</span> | 			<span class="flex-text-inline">{{svg "octicon-globe" 14}} {{ctx.Locale.LangName}}</span> | ||||||
| 			<div class="menu language-menu"> | 			<div class="menu language-menu"> | ||||||
| 				{{range .AllLangs}} | 				{{range .AllLangs -}} | ||||||
| 					<a lang="{{.Lang}}" data-url="{{AppSubUrl}}/?lang={{.Lang}}" class="item {{if eq ctx.Locale.Lang .Lang}}active selected{{end}}">{{.Name}}</a> | 				<a lang="{{.Lang}}" data-url="{{AppSubUrl}}/?lang={{.Lang}}" class="item {{if eq ctx.Locale.Lang .Lang}}selected{{end}}">{{.Name}}</a> | ||||||
| 				{{end}} | 				{{end -}} | ||||||
| 			</div> | 			</div> | ||||||
| 		</div> | 		</div> | ||||||
| 		<a href="{{AssetUrlPrefix}}/licenses.txt">{{ctx.Locale.Tr "licenses"}}</a> | 		<a href="{{AssetUrlPrefix}}/licenses.txt">{{ctx.Locale.Tr "licenses"}}</a> | ||||||
|   | |||||||
| @@ -44,6 +44,5 @@ | |||||||
| 		</div> | 		</div> | ||||||
| 	</div> | 	</div> | ||||||
| </div> | </div> | ||||||
|  |  | ||||||
| <div class="divider"></div> | <div class="divider"></div> | ||||||
| {{end}} | {{end}} | ||||||
|   | |||||||
							
								
								
									
										13
									
								
								templates/repo/issue/sidebar/allow_maintainer_edit.tmpl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								templates/repo/issue/sidebar/allow_maintainer_edit.tmpl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | |||||||
|  | {{if and .Issue.IsPull .IsIssuePoster (not .Issue.IsClosed) .Issue.PullRequest.HeadRepo}} | ||||||
|  | 	{{if and (not (eq .Issue.PullRequest.HeadRepo.FullName .Issue.PullRequest.BaseRepo.FullName)) .CanWriteToHeadRepo}} | ||||||
|  | 	<div class="divider"></div> | ||||||
|  | 	<div class="ui checkbox loading-icon-2px" id="allow-edits-from-maintainers" | ||||||
|  | 			data-url="{{.Issue.Link}}" | ||||||
|  | 			data-tooltip-content="{{ctx.Locale.Tr "repo.pulls.allow_edits_from_maintainers_desc"}}" | ||||||
|  | 			data-prompt-error="{{ctx.Locale.Tr "repo.pulls.allow_edits_from_maintainers_err"}}" | ||||||
|  | 	> | ||||||
|  | 		<label><strong>{{ctx.Locale.Tr "repo.pulls.allow_edits_from_maintainers"}}</strong></label> | ||||||
|  | 		<input type="checkbox" {{if .Issue.PullRequest.AllowMaintainerEdit}}checked{{end}}> | ||||||
|  | 	</div> | ||||||
|  | 	{{end}} | ||||||
|  | {{end}} | ||||||
							
								
								
									
										46
									
								
								templates/repo/issue/sidebar/assignee_list.tmpl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								templates/repo/issue/sidebar/assignee_list.tmpl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | |||||||
|  | <div class="divider"></div> | ||||||
|  | <input id="assignee_id" name="assignee_id" type="hidden" value="{{.assignee_id}}"> | ||||||
|  | <div class="ui {{if or (not .HasIssuesOrPullsWritePermission) .Repository.IsArchived}}disabled{{end}} floating jump select-assignees-modify dropdown"> | ||||||
|  | 	<a class="text muted flex-text-block"> | ||||||
|  | 		<strong>{{ctx.Locale.Tr "repo.issues.new.assignees"}}</strong> | ||||||
|  | 		{{if and .HasIssuesOrPullsWritePermission (not .Repository.IsArchived)}} | ||||||
|  | 			{{svg "octicon-gear" 16 "tw-ml-1"}} | ||||||
|  | 		{{end}} | ||||||
|  | 	</a> | ||||||
|  | 	<div class="filter menu" data-action="update" data-issue-id="{{$.Issue.ID}}" data-update-url="{{$.RepoLink}}/issues/assignee"> | ||||||
|  | 		<div class="ui icon search input"> | ||||||
|  | 			<i class="icon">{{svg "octicon-search" 16}}</i> | ||||||
|  | 			<input type="text" placeholder="{{ctx.Locale.Tr "repo.issues.filter_assignees"}}"> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="no-select item">{{ctx.Locale.Tr "repo.issues.new.clear_assignees"}}</div> | ||||||
|  | 		{{range .Assignees}} | ||||||
|  |  | ||||||
|  | 			{{$AssigneeID := .ID}} | ||||||
|  | 			<a class="item{{range $.Issue.Assignees}}{{if eq .ID $AssigneeID}} checked{{end}}{{end}}" href="#" data-id="{{.ID}}" data-id-selector="#assignee_{{.ID}}"> | ||||||
|  | 				{{$checked := false}} | ||||||
|  | 				{{range $.Issue.Assignees}} | ||||||
|  | 					{{if eq .ID $AssigneeID}} | ||||||
|  | 						{{$checked = true}} | ||||||
|  | 					{{end}} | ||||||
|  | 				{{end}} | ||||||
|  | 				<span class="octicon-check {{if not $checked}}tw-invisible{{end}}">{{svg "octicon-check"}}</span> | ||||||
|  | 				<span class="text"> | ||||||
|  | 					{{ctx.AvatarUtils.Avatar . 20 "tw-mr-2"}}{{template "repo/search_name" .}} | ||||||
|  | 				</span> | ||||||
|  | 			</a> | ||||||
|  | 		{{end}} | ||||||
|  | 	</div> | ||||||
|  | </div> | ||||||
|  | <div class="ui assignees list"> | ||||||
|  | 	<span class="no-select item {{if .Issue.Assignees}}tw-hidden{{end}}">{{ctx.Locale.Tr "repo.issues.new.no_assignees"}}</span> | ||||||
|  | 	<div class="selected"> | ||||||
|  | 		{{range .Issue.Assignees}} | ||||||
|  | 			<div class="item"> | ||||||
|  | 				<a class="muted sidebar-item-link" href="{{$.RepoLink}}/{{if $.Issue.IsPull}}pulls{{else}}issues{{end}}?assignee={{.ID}}"> | ||||||
|  | 					{{ctx.AvatarUtils.Avatar . 28 "tw-mr-2"}} | ||||||
|  | 					{{.GetDisplayName}} | ||||||
|  | 				</a> | ||||||
|  | 			</div> | ||||||
|  | 		{{end}} | ||||||
|  | 	</div> | ||||||
|  | </div> | ||||||
							
								
								
									
										29
									
								
								templates/repo/issue/sidebar/due_date.tmpl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								templates/repo/issue/sidebar/due_date.tmpl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | |||||||
|  | <div class="divider"></div> | ||||||
|  | <span class="text"><strong>{{ctx.Locale.Tr "repo.issues.due_date"}}</strong></span> | ||||||
|  | <div class="ui form tw-mt-2"> | ||||||
|  | 	{{if .Issue.DeadlineUnix}} | ||||||
|  | 		<div class="tw-flex tw-justify-between tw-items-center tw-gap-2"> | ||||||
|  | 			<div class="due-date {{if .Issue.IsOverdue}}text red{{end}}" {{if .Issue.IsOverdue}}data-tooltip-content="{{ctx.Locale.Tr "repo.issues.due_date_overdue"}}"{{end}}> | ||||||
|  | 				{{svg "octicon-calendar"}} {{DateUtils.AbsoluteLong .Issue.DeadlineUnix}} | ||||||
|  | 			</div> | ||||||
|  | 			<div class="flex-text-block"> | ||||||
|  | 				{{if and .HasIssuesOrPullsWritePermission (not .Repository.IsArchived)}} | ||||||
|  | 					<a class="issue-due-edit muted" data-tooltip-content="{{ctx.Locale.Tr "repo.issues.due_date_form_edit"}}">{{svg "octicon-pencil"}}</a> | ||||||
|  | 					<a class="issue-due-remove muted" data-tooltip-content="{{ctx.Locale.Tr "repo.issues.due_date_form_remove"}}">{{svg "octicon-trash"}}</a> | ||||||
|  | 				{{end}} | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 	{{else}} | ||||||
|  | 		{{ctx.Locale.Tr "repo.issues.due_date_not_set"}} | ||||||
|  | 	{{end}} | ||||||
|  |  | ||||||
|  | 	{{if and .HasIssuesOrPullsWritePermission (not .Repository.IsArchived)}} | ||||||
|  | 		<form class="ui fluid action input issue-due-form form-fetch-action tw-mt-2 {{if .Issue.DeadlineUnix}}tw-hidden{{end}}" | ||||||
|  | 					method="post" action="{{AppSubUrl}}/{{PathEscape .Repository.Owner.Name}}/{{PathEscape .Repository.Name}}/issues/{{.Issue.Index}}/deadline" | ||||||
|  | 		> | ||||||
|  | 			{{$.CsrfTokenHtml}} | ||||||
|  | 			<input required type="date" name="deadline" placeholder="{{ctx.Locale.Tr "repo.issues.due_date_form"}}" {{if .Issue.DeadlineUnix}}value="{{.Issue.DeadlineUnix.FormatDate}}"{{end}}> | ||||||
|  | 			<button class="ui icon button">{{Iif .Issue.DeadlineUnix (svg "octicon-pencil") (svg "octicon-plus")}}</button> | ||||||
|  | 		</form> | ||||||
|  | 	{{end}} | ||||||
|  | </div> | ||||||
							
								
								
									
										149
									
								
								templates/repo/issue/sidebar/issue_dependencies.tmpl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										149
									
								
								templates/repo/issue/sidebar/issue_dependencies.tmpl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,149 @@ | |||||||
|  | {{if .Repository.IsDependenciesEnabled ctx}} | ||||||
|  | 	<div class="divider"></div> | ||||||
|  |  | ||||||
|  | 	<div class="ui depending"> | ||||||
|  | 		{{if (and (not .BlockedByDependencies) (not .BlockedByDependenciesNotPermitted) (not .BlockingDependencies) (not .BlockingDependenciesNotPermitted))}} | ||||||
|  | 			<span class="text"><strong>{{ctx.Locale.Tr "repo.issues.dependency.title"}}</strong></span> | ||||||
|  | 			<br> | ||||||
|  | 			<p> | ||||||
|  | 				{{if .Issue.IsPull}} | ||||||
|  | 					{{ctx.Locale.Tr "repo.issues.dependency.pr_no_dependencies"}} | ||||||
|  | 				{{else}} | ||||||
|  | 					{{ctx.Locale.Tr "repo.issues.dependency.issue_no_dependencies"}} | ||||||
|  | 				{{end}} | ||||||
|  | 			</p> | ||||||
|  | 		{{end}} | ||||||
|  |  | ||||||
|  | 		{{if or .BlockingDependencies .BlockingDependenciesNotPermitted}} | ||||||
|  | 			<span class="text" data-tooltip-content="{{if .Issue.IsPull}}{{ctx.Locale.Tr "repo.issues.dependency.pr_close_blocks"}}{{else}}{{ctx.Locale.Tr "repo.issues.dependency.issue_close_blocks"}}{{end}}"> | ||||||
|  | 				<strong>{{ctx.Locale.Tr "repo.issues.dependency.blocks_short"}}</strong> | ||||||
|  | 			</span> | ||||||
|  | 			<div class="ui divided list"> | ||||||
|  | 				{{range .BlockingDependencies}} | ||||||
|  | 					<div class="item dependency{{if .Issue.IsClosed}} is-closed{{end}} tw-flex tw-items-center tw-justify-between"> | ||||||
|  | 						<div class="item-left tw-flex tw-justify-center tw-flex-col tw-flex-1 gt-ellipsis"> | ||||||
|  | 							<a class="muted gt-ellipsis" href="{{.Issue.Link}}" data-tooltip-content="#{{.Issue.Index}} {{.Issue.Title | ctx.RenderUtils.RenderEmoji}}"> | ||||||
|  | 								#{{.Issue.Index}} {{.Issue.Title | ctx.RenderUtils.RenderEmoji}} | ||||||
|  | 							</a> | ||||||
|  | 							<div class="text small gt-ellipsis" data-tooltip-content="{{.Repository.OwnerName}}/{{.Repository.Name}}"> | ||||||
|  | 								{{.Repository.OwnerName}}/{{.Repository.Name}} | ||||||
|  | 							</div> | ||||||
|  | 						</div> | ||||||
|  | 						<div class="item-right tw-flex tw-items-center tw-m-1"> | ||||||
|  | 							{{if and $.CanCreateIssueDependencies (not $.Repository.IsArchived)}} | ||||||
|  | 								<a class="delete-dependency-button ci muted" data-id="{{.Issue.ID}}" data-type="blocking" data-tooltip-content="{{ctx.Locale.Tr "repo.issues.dependency.remove_info"}}"> | ||||||
|  | 									{{svg "octicon-trash" 16}} | ||||||
|  | 								</a> | ||||||
|  | 							{{end}} | ||||||
|  | 						</div> | ||||||
|  | 					</div> | ||||||
|  | 				{{end}} | ||||||
|  | 				{{if .BlockingDependenciesNotPermitted}} | ||||||
|  | 					<div class="item tw-flex tw-items-center tw-justify-between gt-ellipsis"> | ||||||
|  | 						<span>{{ctx.Locale.TrN (len .BlockingDependenciesNotPermitted) "repo.issues.dependency.no_permission_1" "repo.issues.dependency.no_permission_n" (len .BlockingDependenciesNotPermitted)}}</span> | ||||||
|  | 					</div> | ||||||
|  | 				{{end}} | ||||||
|  | 			</div> | ||||||
|  | 		{{end}} | ||||||
|  |  | ||||||
|  | 		{{if or .BlockedByDependencies .BlockedByDependenciesNotPermitted}} | ||||||
|  | 			<span class="text" data-tooltip-content="{{if .Issue.IsPull}}{{ctx.Locale.Tr "repo.issues.dependency.pr_closing_blockedby"}}{{else}}{{ctx.Locale.Tr "repo.issues.dependency.issue_closing_blockedby"}}{{end}}"> | ||||||
|  | 				<strong>{{ctx.Locale.Tr "repo.issues.dependency.blocked_by_short"}}</strong> | ||||||
|  | 			</span> | ||||||
|  | 			<div class="ui divided list"> | ||||||
|  | 				{{range .BlockedByDependencies}} | ||||||
|  | 					<div class="item dependency{{if .Issue.IsClosed}} is-closed{{end}} tw-flex tw-items-center tw-justify-between"> | ||||||
|  | 						<div class="item-left tw-flex tw-justify-center tw-flex-col tw-flex-1 gt-ellipsis"> | ||||||
|  | 							<a class="muted gt-ellipsis" href="{{.Issue.Link}}" data-tooltip-content="#{{.Issue.Index}} {{.Issue.Title | ctx.RenderUtils.RenderEmoji}}"> | ||||||
|  | 								#{{.Issue.Index}} {{.Issue.Title | ctx.RenderUtils.RenderEmoji}} | ||||||
|  | 							</a> | ||||||
|  | 							<div class="text small gt-ellipsis" data-tooltip-content="{{.Repository.OwnerName}}/{{.Repository.Name}}"> | ||||||
|  | 								{{.Repository.OwnerName}}/{{.Repository.Name}} | ||||||
|  | 							</div> | ||||||
|  | 						</div> | ||||||
|  | 						<div class="item-right tw-flex tw-items-center tw-m-1"> | ||||||
|  | 							{{if and $.CanCreateIssueDependencies (not $.Repository.IsArchived)}} | ||||||
|  | 								<a class="delete-dependency-button ci muted" data-id="{{.Issue.ID}}" data-type="blockedBy" data-tooltip-content="{{ctx.Locale.Tr "repo.issues.dependency.remove_info"}}"> | ||||||
|  | 									{{svg "octicon-trash" 16}} | ||||||
|  | 								</a> | ||||||
|  | 							{{end}} | ||||||
|  | 						</div> | ||||||
|  | 					</div> | ||||||
|  | 				{{end}} | ||||||
|  | 				{{if $.CanCreateIssueDependencies}} | ||||||
|  | 					{{range .BlockedByDependenciesNotPermitted}} | ||||||
|  | 						<div class="item dependency{{if .Issue.IsClosed}} is-closed{{end}} tw-flex tw-items-center tw-justify-between"> | ||||||
|  | 							<div class="item-left tw-flex tw-justify-center tw-flex-col tw-flex-1 gt-ellipsis"> | ||||||
|  | 								<div class="gt-ellipsis"> | ||||||
|  | 									<span data-tooltip-content="{{ctx.Locale.Tr "repo.issues.dependency.no_permission.can_remove"}}">{{svg "octicon-lock" 16}}</span> | ||||||
|  | 									<span class="gt-ellipsis" data-tooltip-content="#{{.Issue.Index}} {{.Issue.Title | ctx.RenderUtils.RenderEmoji}}"> | ||||||
|  | 										#{{.Issue.Index}} {{.Issue.Title | ctx.RenderUtils.RenderEmoji}} | ||||||
|  | 									</span> | ||||||
|  | 								</div> | ||||||
|  | 								<div class="text small gt-ellipsis" data-tooltip-content="{{.Repository.OwnerName}}/{{.Repository.Name}}"> | ||||||
|  | 									{{.Repository.OwnerName}}/{{.Repository.Name}} | ||||||
|  | 								</div> | ||||||
|  | 							</div> | ||||||
|  | 							<div class="item-right tw-flex tw-items-center tw-m-1"> | ||||||
|  | 								{{if and $.CanCreateIssueDependencies (not $.Repository.IsArchived)}} | ||||||
|  | 									<a class="delete-dependency-button ci muted" data-id="{{.Issue.ID}}" data-type="blocking" data-tooltip-content="{{ctx.Locale.Tr "repo.issues.dependency.remove_info"}}"> | ||||||
|  | 										{{svg "octicon-trash" 16}} | ||||||
|  | 									</a> | ||||||
|  | 								{{end}} | ||||||
|  | 							</div> | ||||||
|  | 						</div> | ||||||
|  | 					{{end}} | ||||||
|  | 				{{else if .BlockedByDependenciesNotPermitted}} | ||||||
|  | 					<div class="item tw-flex tw-items-center tw-justify-between gt-ellipsis"> | ||||||
|  | 						<span>{{ctx.Locale.TrN (len .BlockedByDependenciesNotPermitted) "repo.issues.dependency.no_permission_1" "repo.issues.dependency.no_permission_n" (len .BlockedByDependenciesNotPermitted)}}</span> | ||||||
|  | 					</div> | ||||||
|  | 				{{end}} | ||||||
|  | 			</div> | ||||||
|  | 		{{end}} | ||||||
|  |  | ||||||
|  | 		{{if and .CanCreateIssueDependencies (not .Repository.IsArchived)}} | ||||||
|  | 			<div> | ||||||
|  | 				<form method="post" action="{{.Issue.Link}}/dependency/add" id="addDependencyForm"> | ||||||
|  | 					{{$.CsrfTokenHtml}} | ||||||
|  | 					<div class="ui fluid action input"> | ||||||
|  | 						<div class="ui search selection dropdown" id="new-dependency-drop-list" data-issue-id="{{.Issue.ID}}"> | ||||||
|  | 							<input name="newDependency" type="hidden"> | ||||||
|  | 							{{svg "octicon-triangle-down" 14 "dropdown icon"}} | ||||||
|  | 							<input type="text" class="search"> | ||||||
|  | 							<div class="default text">{{ctx.Locale.Tr "repo.issues.dependency.add"}}</div> | ||||||
|  | 						</div> | ||||||
|  | 						<button class="ui icon button"> | ||||||
|  | 							{{svg "octicon-plus"}} | ||||||
|  | 						</button> | ||||||
|  | 					</div> | ||||||
|  | 				</form> | ||||||
|  | 			</div> | ||||||
|  | 		{{end}} | ||||||
|  | 	</div> | ||||||
|  |  | ||||||
|  | 	{{if and .CanCreateIssueDependencies (not .Repository.IsArchived)}} | ||||||
|  | 		<input type="hidden" id="crossRepoSearch" value="{{.AllowCrossRepositoryDependencies}}"> | ||||||
|  |  | ||||||
|  | 		<div class="ui g-modal-confirm modal remove-dependency"> | ||||||
|  | 			<div class="header"> | ||||||
|  | 				{{svg "octicon-trash"}} | ||||||
|  | 				{{ctx.Locale.Tr "repo.issues.dependency.remove_header"}} | ||||||
|  | 			</div> | ||||||
|  | 			<div class="content"> | ||||||
|  | 				<form method="post" action="{{.Issue.Link}}/dependency/delete" id="removeDependencyForm"> | ||||||
|  | 					{{$.CsrfTokenHtml}} | ||||||
|  | 					<input type="hidden" value="" name="removeDependencyID" id="removeDependencyID"> | ||||||
|  | 					<input type="hidden" value="" name="dependencyType" id="dependencyType"> | ||||||
|  | 				</form> | ||||||
|  | 				<p>{{if .Issue.IsPull}} | ||||||
|  | 					{{ctx.Locale.Tr "repo.issues.dependency.pr_remove_text"}} | ||||||
|  | 				{{else}} | ||||||
|  | 					{{ctx.Locale.Tr "repo.issues.dependency.issue_remove_text"}} | ||||||
|  | 				{{end}}</p> | ||||||
|  | 			</div> | ||||||
|  | 			{{$ModalButtonCancelText := ctx.Locale.Tr "repo.issues.dependency.cancel"}} | ||||||
|  | 			{{$ModalButtonOkText := ctx.Locale.Tr "repo.issues.dependency.remove"}} | ||||||
|  | 			{{template "base/modal_actions_confirm" (dict "." . "ModalButtonCancelText" $ModalButtonCancelText "ModalButtonOkText" $ModalButtonOkText)}} | ||||||
|  | 		</div> | ||||||
|  | 	{{end}} | ||||||
|  | {{end}} | ||||||
							
								
								
									
										118
									
								
								templates/repo/issue/sidebar/issue_management.tmpl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										118
									
								
								templates/repo/issue/sidebar/issue_management.tmpl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,118 @@ | |||||||
|  | {{if and .IsRepoAdmin (not .Repository.IsArchived)}} | ||||||
|  | 	<div class="divider"></div> | ||||||
|  |  | ||||||
|  | 	{{if or .PinEnabled .Issue.IsPinned}} | ||||||
|  | 		<form class="tw-mt-1 form-fetch-action single-button-form" method="post" {{if $.NewPinAllowed}}action="{{.Issue.Link}}/pin"{{else}}data-tooltip-content="{{ctx.Locale.Tr "repo.issues.max_pinned"}}"{{end}}> | ||||||
|  | 			{{$.CsrfTokenHtml}} | ||||||
|  | 			<button class="fluid ui button {{if not $.NewPinAllowed}}disabled{{end}}"> | ||||||
|  | 				{{if not .Issue.IsPinned}} | ||||||
|  | 					{{svg "octicon-pin" 16 "tw-mr-2"}} | ||||||
|  | 					{{ctx.Locale.Tr "pin"}} | ||||||
|  | 				{{else}} | ||||||
|  | 					{{svg "octicon-pin-slash" 16 "tw-mr-2"}} | ||||||
|  | 					{{ctx.Locale.Tr "unpin"}} | ||||||
|  | 				{{end}} | ||||||
|  | 			</button> | ||||||
|  | 		</form> | ||||||
|  | 	{{end}} | ||||||
|  |  | ||||||
|  | 	<button class="tw-mt-1 fluid ui show-modal button{{if .Issue.IsLocked}} red{{end}}" data-modal="#lock"> | ||||||
|  | 		{{if .Issue.IsLocked}} | ||||||
|  | 			{{svg "octicon-key"}} | ||||||
|  | 			{{ctx.Locale.Tr "repo.issues.unlock"}} | ||||||
|  | 		{{else}} | ||||||
|  | 			{{svg "octicon-lock"}} | ||||||
|  | 			{{ctx.Locale.Tr "repo.issues.lock"}} | ||||||
|  | 		{{end}} | ||||||
|  | 	</button> | ||||||
|  | 	<div class="ui tiny modal" id="lock"> | ||||||
|  | 		<div class="header"> | ||||||
|  | 			{{if .Issue.IsLocked}} | ||||||
|  | 				{{ctx.Locale.Tr "repo.issues.unlock.title"}} | ||||||
|  | 			{{else}} | ||||||
|  | 				{{ctx.Locale.Tr "repo.issues.lock.title"}} | ||||||
|  | 			{{end}} | ||||||
|  | 		</div> | ||||||
|  | 		<div class="content"> | ||||||
|  | 			<div class="ui warning message"> | ||||||
|  | 				{{if .Issue.IsLocked}} | ||||||
|  | 					{{ctx.Locale.Tr "repo.issues.unlock.notice_1"}}<br> | ||||||
|  | 					{{ctx.Locale.Tr "repo.issues.unlock.notice_2"}}<br> | ||||||
|  | 				{{else}} | ||||||
|  | 					{{ctx.Locale.Tr "repo.issues.lock.notice_1"}}<br> | ||||||
|  | 					{{ctx.Locale.Tr "repo.issues.lock.notice_2"}}<br> | ||||||
|  | 					{{ctx.Locale.Tr "repo.issues.lock.notice_3"}}<br> | ||||||
|  | 				{{end}} | ||||||
|  | 			</div> | ||||||
|  |  | ||||||
|  | 			<form class="ui form form-fetch-action" action="{{.Issue.Link}}{{if .Issue.IsLocked}}/unlock{{else}}/lock{{end}}" | ||||||
|  | 				method="post"> | ||||||
|  | 				{{.CsrfTokenHtml}} | ||||||
|  |  | ||||||
|  | 				{{if not .Issue.IsLocked}} | ||||||
|  | 					<div class="field"> | ||||||
|  | 						<strong> {{ctx.Locale.Tr "repo.issues.lock.reason"}} </strong> | ||||||
|  | 					</div> | ||||||
|  |  | ||||||
|  | 					<div class="field"> | ||||||
|  | 						<div class="ui fluid dropdown selection"> | ||||||
|  |  | ||||||
|  | 							<select name="reason"> | ||||||
|  | 								<option value=""> </option> | ||||||
|  | 								{{range .LockReasons}} | ||||||
|  | 									<option value="{{.}}">{{.}}</option> | ||||||
|  | 								{{end}} | ||||||
|  | 							</select> | ||||||
|  | 							{{svg "octicon-triangle-down" 14 "dropdown icon"}} | ||||||
|  |  | ||||||
|  | 							<div class="default text"> </div> | ||||||
|  |  | ||||||
|  | 							<div class="menu"> | ||||||
|  | 								{{range .LockReasons}} | ||||||
|  | 									<div class="item" data-value="{{.}}">{{.}}</div> | ||||||
|  | 								{{end}} | ||||||
|  | 							</div> | ||||||
|  | 						</div> | ||||||
|  | 					</div> | ||||||
|  | 				{{end}} | ||||||
|  |  | ||||||
|  | 				<div class="text right actions"> | ||||||
|  | 					<button class="ui cancel button">{{ctx.Locale.Tr "settings.cancel"}}</button> | ||||||
|  | 					<button class="ui red button"> | ||||||
|  | 						{{if .Issue.IsLocked}} | ||||||
|  | 							{{ctx.Locale.Tr "repo.issues.unlock_confirm"}} | ||||||
|  | 						{{else}} | ||||||
|  | 							{{ctx.Locale.Tr "repo.issues.lock_confirm"}} | ||||||
|  | 						{{end}} | ||||||
|  | 					</button> | ||||||
|  | 				</div> | ||||||
|  | 			</form> | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | 	<button class="tw-mt-1 fluid ui show-modal button" data-modal="#sidebar-delete-issue"> | ||||||
|  | 		{{svg "octicon-trash"}} | ||||||
|  | 		{{ctx.Locale.Tr "repo.issues.delete"}} | ||||||
|  | 	</button> | ||||||
|  | 	<div class="ui g-modal-confirm modal" id="sidebar-delete-issue"> | ||||||
|  | 		<div class="header"> | ||||||
|  | 			{{if .Issue.IsPull}} | ||||||
|  | 				{{ctx.Locale.Tr "repo.pulls.delete.title"}} | ||||||
|  | 			{{else}} | ||||||
|  | 				{{ctx.Locale.Tr "repo.issues.delete.title"}} | ||||||
|  | 			{{end}} | ||||||
|  | 		</div> | ||||||
|  | 		<div class="content"> | ||||||
|  | 			<p> | ||||||
|  | 				{{if .Issue.IsPull}} | ||||||
|  | 					{{ctx.Locale.Tr "repo.pulls.delete.text"}} | ||||||
|  | 				{{else}} | ||||||
|  | 					{{ctx.Locale.Tr "repo.issues.delete.text"}} | ||||||
|  | 				{{end}} | ||||||
|  | 			</p> | ||||||
|  | 		</div> | ||||||
|  | 		<form action="{{.Issue.Link}}/delete" method="post"> | ||||||
|  | 			{{.CsrfTokenHtml}} | ||||||
|  | 			{{template "base/modal_actions_confirm" .}} | ||||||
|  | 		</form> | ||||||
|  | 	</div> | ||||||
|  | {{end}} | ||||||
							
								
								
									
										23
									
								
								templates/repo/issue/sidebar/milestone_list.tmpl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								templates/repo/issue/sidebar/milestone_list.tmpl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | |||||||
|  | <div class="divider"></div> | ||||||
|  | <div class="ui {{if or (not .HasIssuesOrPullsWritePermission) .Repository.IsArchived}}disabled{{end}} floating jump select-milestone dropdown"> | ||||||
|  | 	<a class="text muted flex-text-block"> | ||||||
|  | 		<strong>{{ctx.Locale.Tr "repo.issues.new.milestone"}}</strong> | ||||||
|  | 		{{if and .HasIssuesOrPullsWritePermission (not .Repository.IsArchived)}} | ||||||
|  | 			{{svg "octicon-gear" 16 "tw-ml-1"}} | ||||||
|  | 		{{end}} | ||||||
|  | 	</a> | ||||||
|  | 	<div class="menu" data-action="update" data-issue-id="{{$.Issue.ID}}" data-update-url="{{$.RepoLink}}/issues/milestone"> | ||||||
|  | 		{{template "repo/issue/milestone/select_menu" .}} | ||||||
|  | 	</div> | ||||||
|  | </div> | ||||||
|  | <div class="ui select-milestone list"> | ||||||
|  | 	<span class="no-select item {{if .Issue.Milestone}}tw-hidden{{end}}">{{ctx.Locale.Tr "repo.issues.new.no_milestone"}}</span> | ||||||
|  | 	<div class="selected"> | ||||||
|  | 		{{if .Issue.Milestone}} | ||||||
|  | 			<a class="item muted sidebar-item-link" href="{{.RepoLink}}/milestone/{{.Issue.Milestone.ID}}"> | ||||||
|  | 				{{svg "octicon-milestone" 18 "tw-mr-2"}} | ||||||
|  | 				{{.Issue.Milestone.Name}} | ||||||
|  | 			</a> | ||||||
|  | 		{{end}} | ||||||
|  | 	</div> | ||||||
|  | </div> | ||||||
							
								
								
									
										11
									
								
								templates/repo/issue/sidebar/participant_list.tmpl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								templates/repo/issue/sidebar/participant_list.tmpl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | |||||||
|  | {{if .Participants}} | ||||||
|  | 	<div class="divider"></div> | ||||||
|  | 	<span class="text"><strong>{{ctx.Locale.Tr "repo.issues.num_participants" .NumParticipants}}</strong></span> | ||||||
|  | 	<div class="ui list tw-flex tw-flex-wrap"> | ||||||
|  | 		{{range .Participants}} | ||||||
|  | 			<a {{if gt .ID 0}}href="{{.HomeLink}}"{{end}} data-tooltip-content="{{.GetDisplayName}}"> | ||||||
|  | 				{{ctx.AvatarUtils.Avatar . 28 "tw-my-0.5 tw-mr-1"}} | ||||||
|  | 			</a> | ||||||
|  | 		{{end}} | ||||||
|  | 	</div> | ||||||
|  | {{end}} | ||||||
							
								
								
									
										53
									
								
								templates/repo/issue/sidebar/project_list.tmpl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								templates/repo/issue/sidebar/project_list.tmpl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,53 @@ | |||||||
|  | {{if .IsProjectsEnabled}} | ||||||
|  | 	<div class="divider"></div> | ||||||
|  |  | ||||||
|  | 	<div class="ui {{if or (not .HasIssuesOrPullsWritePermission) .Repository.IsArchived}}disabled{{end}} floating jump select-project dropdown"> | ||||||
|  | 		<a class="text muted flex-text-block"> | ||||||
|  | 			<strong>{{ctx.Locale.Tr "repo.issues.new.projects"}}</strong> | ||||||
|  | 			{{if and .HasIssuesOrPullsWritePermission (not .Repository.IsArchived)}} | ||||||
|  | 				{{svg "octicon-gear" 16 "tw-ml-1"}} | ||||||
|  | 			{{end}} | ||||||
|  | 		</a> | ||||||
|  | 		<div class="menu" data-action="update" data-issue-id="{{$.Issue.ID}}" data-update-url="{{$.RepoLink}}/issues/projects"> | ||||||
|  | 			{{if or .OpenProjects .ClosedProjects}} | ||||||
|  | 			<div class="ui icon search input"> | ||||||
|  | 				<i class="icon">{{svg "octicon-search" 16}}</i> | ||||||
|  | 				<input type="text" placeholder="{{ctx.Locale.Tr "repo.issues.filter_projects"}}"> | ||||||
|  | 			</div> | ||||||
|  | 			{{end}} | ||||||
|  | 			<div class="no-select item">{{ctx.Locale.Tr "repo.issues.new.clear_projects"}}</div> | ||||||
|  | 			{{if .OpenProjects}} | ||||||
|  | 				<div class="divider"></div> | ||||||
|  | 				<div class="header"> | ||||||
|  | 					{{ctx.Locale.Tr "repo.issues.new.open_projects"}} | ||||||
|  | 				</div> | ||||||
|  | 				{{range .OpenProjects}} | ||||||
|  | 					<a class="item muted sidebar-item-link" data-id="{{.ID}}" data-href="{{.Link ctx}}"> | ||||||
|  | 						{{svg .IconName 18 "tw-mr-2"}}{{.Title}} | ||||||
|  | 					</a> | ||||||
|  | 				{{end}} | ||||||
|  | 			{{end}} | ||||||
|  | 			{{if .ClosedProjects}} | ||||||
|  | 				<div class="divider"></div> | ||||||
|  | 				<div class="header"> | ||||||
|  | 					{{ctx.Locale.Tr "repo.issues.new.closed_projects"}} | ||||||
|  | 				</div> | ||||||
|  | 				{{range .ClosedProjects}} | ||||||
|  | 					<a class="item muted sidebar-item-link" data-id="{{.ID}}" data-href="{{.Link ctx}}"> | ||||||
|  | 						{{svg .IconName 18 "tw-mr-2"}}{{.Title}} | ||||||
|  | 					</a> | ||||||
|  | 				{{end}} | ||||||
|  | 			{{end}} | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | 	<div class="ui select-project list"> | ||||||
|  | 		<span class="no-select item {{if .Issue.Project}}tw-hidden{{end}}">{{ctx.Locale.Tr "repo.issues.new.no_projects"}}</span> | ||||||
|  | 		<div class="selected"> | ||||||
|  | 			{{if .Issue.Project}} | ||||||
|  | 				<a class="item muted sidebar-item-link" href="{{.Issue.Project.Link ctx}}"> | ||||||
|  | 					{{svg .Issue.Project.IconName 18 "tw-mr-2"}}{{.Issue.Project.Title}} | ||||||
|  | 				</a> | ||||||
|  | 			{{end}} | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | {{end}} | ||||||
							
								
								
									
										8
									
								
								templates/repo/issue/sidebar/reference_link.tmpl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								templates/repo/issue/sidebar/reference_link.tmpl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | |||||||
|  | <div class="divider"></div> | ||||||
|  | <div class="ui equal width compact grid"> | ||||||
|  | 	{{$issueReferenceLink := printf "%s#%d" .Issue.Repo.FullName .Issue.Index}} | ||||||
|  | 	<div class="row tw-items-center" data-tooltip-content="{{$issueReferenceLink}}"> | ||||||
|  | 		<span class="text column truncate">{{ctx.Locale.Tr "repo.issues.reference_link" $issueReferenceLink}}</span> | ||||||
|  | 		<button class="ui two wide button column tw-p-2" data-clipboard-text="{{$issueReferenceLink}}">{{svg "octicon-copy" 14}}</button> | ||||||
|  | 	</div> | ||||||
|  | </div> | ||||||
							
								
								
									
										116
									
								
								templates/repo/issue/sidebar/reviewer_list.tmpl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										116
									
								
								templates/repo/issue/sidebar/reviewer_list.tmpl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,116 @@ | |||||||
|  | <input id="reviewer_id" name="reviewer_id" type="hidden" value="{{.reviewer_id}}"> | ||||||
|  | <div class="ui {{if or (and (not .Reviewers) (not .TeamReviewers)) (not .CanChooseReviewer) .Repository.IsArchived}}disabled{{end}} floating jump select-reviewers-modify dropdown"> | ||||||
|  | 	<a class="text tw-flex tw-items-center muted"> | ||||||
|  | 		<strong>{{ctx.Locale.Tr "repo.issues.review.reviewers"}}</strong> | ||||||
|  | 		{{if and .CanChooseReviewer (not .Repository.IsArchived)}} | ||||||
|  | 			{{svg "octicon-gear" 16 "tw-ml-1"}} | ||||||
|  | 		{{end}} | ||||||
|  | 	</a> | ||||||
|  | 	<div class="filter menu" data-action="update" data-issue-id="{{$.Issue.ID}}" data-update-url="{{$.RepoLink}}/issues/request_review"> | ||||||
|  | 		{{if .Reviewers}} | ||||||
|  | 			<div class="ui icon search input"> | ||||||
|  | 				<i class="icon">{{svg "octicon-search" 16}}</i> | ||||||
|  | 				<input type="text" placeholder="{{ctx.Locale.Tr "repo.issues.filter_reviewers"}}"> | ||||||
|  | 			</div> | ||||||
|  | 		{{end}} | ||||||
|  | 		{{if .Reviewers}} | ||||||
|  | 			{{range .Reviewers}} | ||||||
|  | 				{{if .User}} | ||||||
|  | 					<a class="{{if not .CanChange}}ui{{end}} item {{if .Checked}}checked{{end}} {{if not .CanChange}}ban-change{{end}}" href="#" data-id="{{.ItemID}}" data-id-selector="#review_request_{{.ItemID}}" {{if not .CanChange}} data-tooltip-content="{{ctx.Locale.Tr "repo.issues.remove_request_review_block"}}"{{end}}> | ||||||
|  | 						<span class="octicon-check {{if not .Checked}}tw-invisible{{end}}">{{svg "octicon-check"}}</span> | ||||||
|  | 						<span class="text"> | ||||||
|  | 							{{ctx.AvatarUtils.Avatar .User 28 "tw-mr-2"}}{{template "repo/search_name" .User}} | ||||||
|  | 						</span> | ||||||
|  | 					</a> | ||||||
|  | 				{{end}} | ||||||
|  | 			{{end}} | ||||||
|  | 		{{end}} | ||||||
|  | 		{{if .TeamReviewers}} | ||||||
|  | 			{{if .Reviewers}} | ||||||
|  | 				<div class="divider"></div> | ||||||
|  | 			{{end}} | ||||||
|  | 			{{range .TeamReviewers}} | ||||||
|  | 				{{if .Team}} | ||||||
|  | 					<a class="{{if not .CanChange}}ui{{end}} item {{if .Checked}}checked{{end}} {{if not .CanChange}}ban-change{{end}}" href="#" data-id="{{.ItemID}}" data-id-selector="#review_request_team_{{.Team.ID}}" {{if not .CanChange}} data-tooltip-content="{{ctx.Locale.Tr "repo.issues.remove_request_review_block"}}"{{end}}> | ||||||
|  | 						<span class="octicon-check {{if not .Checked}}tw-invisible{{end}}">{{svg "octicon-check" 16}}</span> | ||||||
|  | 						<span class="text"> | ||||||
|  | 							{{svg "octicon-people" 16 "tw-ml-4 tw-mr-1"}}{{$.Issue.Repo.OwnerName}}/{{.Team.Name}} | ||||||
|  | 						</span> | ||||||
|  | 					</a> | ||||||
|  | 				{{end}} | ||||||
|  | 			{{end}} | ||||||
|  | 		{{end}} | ||||||
|  | 	</div> | ||||||
|  | </div> | ||||||
|  |  | ||||||
|  | <div class="ui assignees list"> | ||||||
|  | 	<span class="no-select item {{if or .OriginalReviews .PullReviewers}}tw-hidden{{end}}">{{ctx.Locale.Tr "repo.issues.new.no_reviewers"}}</span> | ||||||
|  | 	<div class="selected"> | ||||||
|  | 		{{range .PullReviewers}} | ||||||
|  | 			<div class="item tw-flex tw-items-center tw-py-2"> | ||||||
|  | 				<div class="tw-flex tw-items-center tw-flex-1"> | ||||||
|  | 					{{if .User}} | ||||||
|  | 						<a class="muted sidebar-item-link" href="{{.User.HomeLink}}">{{ctx.AvatarUtils.Avatar .User 20 "tw-mr-2"}}{{.User.GetDisplayName}}</a> | ||||||
|  | 					{{else if .Team}} | ||||||
|  | 						<span class="text">{{svg "octicon-people" 20 "tw-mr-2"}}{{$.Issue.Repo.OwnerName}}/{{.Team.Name}}</span> | ||||||
|  | 					{{end}} | ||||||
|  | 				</div> | ||||||
|  | 				<div class="tw-flex tw-items-center tw-gap-2"> | ||||||
|  | 					{{if (and $.Permission.IsAdmin (or (eq .Review.Type 1) (eq .Review.Type 3)) (not $.Issue.IsClosed) (not $.Issue.PullRequest.HasMerged))}} | ||||||
|  | 						<a href="#" class="ui muted icon tw-flex tw-items-center show-modal" data-tooltip-content="{{ctx.Locale.Tr "repo.issues.dismiss_review"}}" data-modal="#dismiss-review-modal-{{.Review.ID}}"> | ||||||
|  | 							{{svg "octicon-x" 20}} | ||||||
|  | 						</a> | ||||||
|  | 						<div class="ui small modal" id="dismiss-review-modal-{{.Review.ID}}"> | ||||||
|  | 							<div class="header"> | ||||||
|  | 								{{ctx.Locale.Tr "repo.issues.dismiss_review"}} | ||||||
|  | 							</div> | ||||||
|  | 							<div class="content"> | ||||||
|  | 								<div class="ui warning message"> | ||||||
|  | 									{{ctx.Locale.Tr "repo.issues.dismiss_review_warning"}} | ||||||
|  | 								</div> | ||||||
|  | 								<form class="ui form dismiss-review-form" id="dismiss-review-{{.Review.ID}}" action="{{$.RepoLink}}/issues/dismiss_review" method="post"> | ||||||
|  | 									{{$.CsrfTokenHtml}} | ||||||
|  | 									<input type="hidden" name="review_id" value="{{.Review.ID}}"> | ||||||
|  | 									<div class="field"> | ||||||
|  | 										<label for="message">{{ctx.Locale.Tr "action.review_dismissed_reason"}}</label> | ||||||
|  | 										<input id="message" name="message"> | ||||||
|  | 									</div> | ||||||
|  | 									<div class="text right actions"> | ||||||
|  | 										<button class="ui cancel button">{{ctx.Locale.Tr "settings.cancel"}}</button> | ||||||
|  | 										<button class="ui red button" type="submit">{{ctx.Locale.Tr "ok"}}</button> | ||||||
|  | 									</div> | ||||||
|  | 								</form> | ||||||
|  | 							</div> | ||||||
|  | 						</div> | ||||||
|  | 					{{end}} | ||||||
|  | 					{{if .Review.Stale}} | ||||||
|  | 						<span data-tooltip-content="{{ctx.Locale.Tr "repo.issues.is_stale"}}"> | ||||||
|  | 							{{svg "octicon-hourglass" 16}} | ||||||
|  | 						</span> | ||||||
|  | 					{{end}} | ||||||
|  | 					{{if and .CanChange (or .Checked (and (not $.Issue.IsClosed) (not $.Issue.PullRequest.HasMerged)))}} | ||||||
|  | 						<a href="#" class="ui muted icon re-request-review{{if .Checked}} checked{{end}}" data-tooltip-content="{{if .Checked}}{{ctx.Locale.Tr "repo.issues.remove_request_review"}}{{else}}{{ctx.Locale.Tr "repo.issues.re_request_review"}}{{end}}" data-issue-id="{{$.Issue.ID}}" data-id="{{.ItemID}}" data-update-url="{{$.RepoLink}}/issues/request_review">{{svg (Iif .Checked "octicon-trash" "octicon-sync")}}</a> | ||||||
|  | 					{{end}} | ||||||
|  | 					<span {{if .Review.TooltipContent}}data-tooltip-content="{{ctx.Locale.Tr .Review.TooltipContent}}"{{end}}> | ||||||
|  | 						{{svg (printf "octicon-%s" .Review.Type.Icon) 16 (printf "text %s" (.Review.HTMLTypeColorName))}} | ||||||
|  | 					</span> | ||||||
|  | 				</div> | ||||||
|  | 			</div> | ||||||
|  | 		{{end}} | ||||||
|  | 		{{range .OriginalReviews}} | ||||||
|  | 			<div class="item tw-flex tw-items-center tw-py-2"> | ||||||
|  | 				<div class="tw-flex tw-items-center tw-flex-1"> | ||||||
|  | 					<a class="muted" href="{{$.Repository.OriginalURL}}" data-tooltip-content="{{ctx.Locale.Tr "repo.migrated_from_fake" $.Repository.GetOriginalURLHostname}}"> | ||||||
|  | 						{{svg (MigrationIcon $.Repository.GetOriginalURLHostname) 20 "tw-mr-2"}} | ||||||
|  | 						{{.OriginalAuthor}} | ||||||
|  | 					</a> | ||||||
|  | 				</div> | ||||||
|  | 				<div class="tw-flex tw-items-center tw-gap-2"> | ||||||
|  | 					<span {{if .TooltipContent}}data-tooltip-content="{{ctx.Locale.Tr .TooltipContent}}"{{end}}> | ||||||
|  | 						{{svg (printf "octicon-%s" .Type.Icon) 16 (printf "text %s" (.HTMLTypeColorName))}} | ||||||
|  | 					</span> | ||||||
|  | 				</div> | ||||||
|  | 			</div> | ||||||
|  | 		{{end}} | ||||||
|  | 	</div> | ||||||
|  | </div> | ||||||
							
								
								
									
										75
									
								
								templates/repo/issue/sidebar/stopwatch_timetracker.tmpl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										75
									
								
								templates/repo/issue/sidebar/stopwatch_timetracker.tmpl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,75 @@ | |||||||
|  | {{if .Repository.IsTimetrackerEnabled ctx}} | ||||||
|  | 	{{if and .CanUseTimetracker (not .Repository.IsArchived)}} | ||||||
|  | 		<div class="divider"></div> | ||||||
|  | 		<div class="ui timetrack"> | ||||||
|  | 			<span class="text"><strong>{{ctx.Locale.Tr "repo.issues.tracker"}}</strong></span> | ||||||
|  | 			<div class="tw-mt-2"> | ||||||
|  | 				<form method="post" action="{{.Issue.Link}}/times/stopwatch/toggle" id="toggle_stopwatch_form"> | ||||||
|  | 					{{$.CsrfTokenHtml}} | ||||||
|  | 				</form> | ||||||
|  | 				<form method="post" action="{{.Issue.Link}}/times/stopwatch/cancel" id="cancel_stopwatch_form"> | ||||||
|  | 					{{$.CsrfTokenHtml}} | ||||||
|  | 				</form> | ||||||
|  | 				{{if $.IsStopwatchRunning}} | ||||||
|  | 					<button class="ui fluid button issue-stop-time"> | ||||||
|  | 						{{svg "octicon-stopwatch" 16 "tw-mr-2"}} | ||||||
|  | 						{{ctx.Locale.Tr "repo.issues.stop_tracking"}} | ||||||
|  | 					</button> | ||||||
|  | 					<button class="ui fluid button issue-cancel-time tw-mt-2"> | ||||||
|  | 						{{svg "octicon-trash" 16 "tw-mr-2"}} | ||||||
|  | 						{{ctx.Locale.Tr "repo.issues.cancel_tracking"}} | ||||||
|  | 					</button> | ||||||
|  | 				{{else}} | ||||||
|  | 					{{if .HasUserStopwatch}} | ||||||
|  | 						<div class="ui warning message"> | ||||||
|  | 							{{ctx.Locale.Tr "repo.issues.tracking_already_started" .OtherStopwatchURL}} | ||||||
|  | 						</div> | ||||||
|  | 					{{end}} | ||||||
|  | 					<button class="ui fluid button issue-start-time" data-tooltip-content='{{ctx.Locale.Tr "repo.issues.start_tracking"}}'> | ||||||
|  | 						{{svg "octicon-stopwatch" 16 "tw-mr-2"}} | ||||||
|  | 						{{ctx.Locale.Tr "repo.issues.start_tracking_short"}} | ||||||
|  | 					</button> | ||||||
|  | 					<div class="ui mini modal issue-start-time-modal"> | ||||||
|  | 						<div class="header">{{ctx.Locale.Tr "repo.issues.add_time"}}</div> | ||||||
|  | 						<div class="content"> | ||||||
|  | 							<form method="post" id="add_time_manual_form" action="{{.Issue.Link}}/times/add" class="ui input fluid tw-gap-2"> | ||||||
|  | 								{{$.CsrfTokenHtml}} | ||||||
|  | 								<input placeholder='{{ctx.Locale.Tr "repo.issues.add_time_hours"}}' type="number" name="hours"> | ||||||
|  | 								<input placeholder='{{ctx.Locale.Tr "repo.issues.add_time_minutes"}}' type="number" name="minutes" class="ui compact"> | ||||||
|  | 							</form> | ||||||
|  | 						</div> | ||||||
|  | 						<div class="actions"> | ||||||
|  | 							<button class="ui primary approve button">{{ctx.Locale.Tr "repo.issues.add_time_short"}}</button> | ||||||
|  | 							<button class="ui cancel button">{{ctx.Locale.Tr "repo.issues.add_time_cancel"}}</button> | ||||||
|  | 						</div> | ||||||
|  | 					</div> | ||||||
|  | 					<button class="ui fluid button issue-add-time tw-mt-2" data-tooltip-content='{{ctx.Locale.Tr "repo.issues.add_time"}}'> | ||||||
|  | 						{{svg "octicon-plus" 16 "tw-mr-2"}} | ||||||
|  | 						{{ctx.Locale.Tr "repo.issues.add_time_short"}} | ||||||
|  | 					</button> | ||||||
|  | 				{{end}} | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 	{{end}} | ||||||
|  | 	{{if .WorkingUsers}} | ||||||
|  | 		<div class="divider"></div> | ||||||
|  | 		<div class="ui comments"> | ||||||
|  | 			<span class="text"><strong>{{ctx.Locale.Tr "repo.issues.time_spent_from_all_authors" ($.Issue.TotalTrackedTime | Sec2Time)}}</strong></span> | ||||||
|  | 			<div> | ||||||
|  | 				{{range $user, $trackedtime := .WorkingUsers}} | ||||||
|  | 					<div class="comment tw-mt-2"> | ||||||
|  | 						<a class="avatar"> | ||||||
|  | 							{{ctx.AvatarUtils.Avatar $user}} | ||||||
|  | 						</a> | ||||||
|  | 						<div class="content"> | ||||||
|  | 							{{template "shared/user/authorlink" $user}} | ||||||
|  | 							<div class="text"> | ||||||
|  | 								{{$trackedtime|Sec2Time}} | ||||||
|  | 							</div> | ||||||
|  | 						</div> | ||||||
|  | 					</div> | ||||||
|  | 				{{end}} | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 	{{end}} | ||||||
|  | {{end}} | ||||||
							
								
								
									
										9
									
								
								templates/repo/issue/sidebar/watch_notification.tmpl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								templates/repo/issue/sidebar/watch_notification.tmpl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | {{if and $.IssueWatch (not .Repository.IsArchived)}} | ||||||
|  | 	<div class="divider"></div> | ||||||
|  | 	<div class="ui watching"> | ||||||
|  | 		<span class="text"><strong>{{ctx.Locale.Tr "notification.notifications"}}</strong></span> | ||||||
|  | 		<div class="tw-mt-2"> | ||||||
|  | 			{{template "repo/issue/view_content/watching" .}} | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | {{end}} | ||||||
							
								
								
									
										7
									
								
								templates/repo/issue/sidebar/wip_switch.tmpl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								templates/repo/issue/sidebar/wip_switch.tmpl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | |||||||
|  | {{if and (or .HasIssuesOrPullsWritePermission .IsIssuePoster) (not .HasMerged) (not .Issue.IsClosed) (not .IsPullWorkInProgress)}} | ||||||
|  | 	<div class="toggle-wip" data-title="{{.Issue.Title}}" data-wip-prefix="{{index .PullRequestWorkInProgressPrefixes 0}}" data-update-url="{{.Issue.Link}}/title"> | ||||||
|  | 		<a class="muted"> | ||||||
|  | 			{{ctx.Locale.Tr "repo.pulls.still_in_progress"}} {{ctx.Locale.Tr "repo.pulls.add_prefix" (index .PullRequestWorkInProgressPrefixes 0)}} | ||||||
|  | 		</a> | ||||||
|  | 	</div> | ||||||
|  | {{end}} | ||||||
| @@ -1,11 +1,4 @@ | |||||||
| <div class="issue-content"> | <div class="issue-content"> | ||||||
| 	<!-- I know, there is probably a better way to do this (moved from sidebar.tmpl, original author: 6543 @ 2021-02-28) --> |  | ||||||
| 	<!-- Agree, there should be a better way, eg: introduce window.config.pageData (original author: wxiaoguang @ 2021-09-05) --> |  | ||||||
| 	<input type="hidden" id="repolink" value="{{$.RepoRelPath}}"> |  | ||||||
| 	<input type="hidden" id="repoId" value="{{.Repository.ID}}"> |  | ||||||
| 	<input type="hidden" id="issueIndex" value="{{.Issue.Index}}"> |  | ||||||
| 	<input type="hidden" id="type" value="{{.IssueType}}"> |  | ||||||
|  |  | ||||||
| 	{{$createdStr:= DateUtils.TimeSince .Issue.CreatedUnix}} | 	{{$createdStr:= DateUtils.TimeSince .Issue.CreatedUnix}} | ||||||
| 	<div class="issue-content-left comment-list prevent-before-timeline"> | 	<div class="issue-content-left comment-list prevent-before-timeline"> | ||||||
| 		<div class="ui timeline"> | 		<div class="ui timeline"> | ||||||
|   | |||||||
| @@ -1,682 +1,24 @@ | |||||||
| <div class="issue-content-right ui segment"> | <div class="issue-content-right ui segment"> | ||||||
| 	{{template "repo/issue/branch_selector_field" .}} | 	{{template "repo/issue/branch_selector_field" $}} | ||||||
| 	{{if .Issue.IsPull}} |  | ||||||
| 		<input id="reviewer_id" name="reviewer_id" type="hidden" value="{{.reviewer_id}}"> |  | ||||||
| 		<div class="ui {{if or (and (not .Reviewers) (not .TeamReviewers)) (not .CanChooseReviewer) .Repository.IsArchived}}disabled{{end}} floating jump select-reviewers-modify dropdown"> |  | ||||||
| 			<a class="text tw-flex tw-items-center muted"> |  | ||||||
| 				<strong>{{ctx.Locale.Tr "repo.issues.review.reviewers"}}</strong> |  | ||||||
| 				{{if and .CanChooseReviewer (not .Repository.IsArchived)}} |  | ||||||
| 					{{svg "octicon-gear" 16 "tw-ml-1"}} |  | ||||||
| 				{{end}} |  | ||||||
| 			</a> |  | ||||||
| 			<div class="filter menu" data-action="update" data-issue-id="{{$.Issue.ID}}" data-update-url="{{$.RepoLink}}/issues/request_review"> |  | ||||||
| 				{{if .Reviewers}} |  | ||||||
| 					<div class="ui icon search input"> |  | ||||||
| 						<i class="icon">{{svg "octicon-search" 16}}</i> |  | ||||||
| 						<input type="text" placeholder="{{ctx.Locale.Tr "repo.issues.filter_reviewers"}}"> |  | ||||||
| 					</div> |  | ||||||
| 				{{end}} |  | ||||||
| 				{{if .Reviewers}} |  | ||||||
| 					{{range .Reviewers}} |  | ||||||
| 						{{if .User}} |  | ||||||
| 							<a class="{{if not .CanChange}}ui{{end}} item {{if .Checked}}checked{{end}} {{if not .CanChange}}ban-change{{end}}" href="#" data-id="{{.ItemID}}" data-id-selector="#review_request_{{.ItemID}}" {{if not .CanChange}} data-tooltip-content="{{ctx.Locale.Tr "repo.issues.remove_request_review_block"}}"{{end}}> |  | ||||||
| 								<span class="octicon-check {{if not .Checked}}tw-invisible{{end}}">{{svg "octicon-check"}}</span> |  | ||||||
| 								<span class="text"> |  | ||||||
| 									{{ctx.AvatarUtils.Avatar .User 28 "tw-mr-2"}}{{template "repo/search_name" .User}} |  | ||||||
| 								</span> |  | ||||||
| 							</a> |  | ||||||
| 						{{end}} |  | ||||||
| 					{{end}} |  | ||||||
| 				{{end}} |  | ||||||
| 				{{if .TeamReviewers}} |  | ||||||
| 					{{if .Reviewers}} |  | ||||||
| 						<div class="divider"></div> |  | ||||||
| 					{{end}} |  | ||||||
| 					{{range .TeamReviewers}} |  | ||||||
| 						{{if .Team}} |  | ||||||
| 							<a class="{{if not .CanChange}}ui{{end}} item {{if .Checked}}checked{{end}} {{if not .CanChange}}ban-change{{end}}" href="#" data-id="{{.ItemID}}" data-id-selector="#review_request_team_{{.Team.ID}}" {{if not .CanChange}} data-tooltip-content="{{ctx.Locale.Tr "repo.issues.remove_request_review_block"}}"{{end}}> |  | ||||||
| 								<span class="octicon-check {{if not .Checked}}tw-invisible{{end}}">{{svg "octicon-check" 16}}</span> |  | ||||||
| 								<span class="text"> |  | ||||||
| 									{{svg "octicon-people" 16 "tw-ml-4 tw-mr-1"}}{{$.Issue.Repo.OwnerName}}/{{.Team.Name}} |  | ||||||
| 								</span> |  | ||||||
| 							</a> |  | ||||||
| 						{{end}} |  | ||||||
| 					{{end}} |  | ||||||
| 				{{end}} |  | ||||||
| 			</div> |  | ||||||
| 		</div> |  | ||||||
|  |  | ||||||
| 		<div class="ui assignees list"> | 	{{if .Issue.IsPull}} | ||||||
| 			<span class="no-select item {{if or .OriginalReviews .PullReviewers}}tw-hidden{{end}}">{{ctx.Locale.Tr "repo.issues.new.no_reviewers"}}</span> | 		{{template "repo/issue/sidebar/reviewer_list" $}} | ||||||
| 			<div class="selected"> | 		{{template "repo/issue/sidebar/wip_switch" $}} | ||||||
| 				{{range .PullReviewers}} |  | ||||||
| 					<div class="item tw-flex tw-items-center tw-py-2"> |  | ||||||
| 						<div class="tw-flex tw-items-center tw-flex-1"> |  | ||||||
| 							{{if .User}} |  | ||||||
| 								<a class="muted sidebar-item-link" href="{{.User.HomeLink}}">{{ctx.AvatarUtils.Avatar .User 20 "tw-mr-2"}}{{.User.GetDisplayName}}</a> |  | ||||||
| 							{{else if .Team}} |  | ||||||
| 								<span class="text">{{svg "octicon-people" 20 "tw-mr-2"}}{{$.Issue.Repo.OwnerName}}/{{.Team.Name}}</span> |  | ||||||
| 							{{end}} |  | ||||||
| 						</div> |  | ||||||
| 						<div class="tw-flex tw-items-center tw-gap-2"> |  | ||||||
| 							{{if (and $.Permission.IsAdmin (or (eq .Review.Type 1) (eq .Review.Type 3)) (not $.Issue.IsClosed) (not $.Issue.PullRequest.HasMerged))}} |  | ||||||
| 								<a href="#" class="ui muted icon tw-flex tw-items-center show-modal" data-tooltip-content="{{ctx.Locale.Tr "repo.issues.dismiss_review"}}" data-modal="#dismiss-review-modal-{{.Review.ID}}"> |  | ||||||
| 									{{svg "octicon-x" 20}} |  | ||||||
| 								</a> |  | ||||||
| 								<div class="ui small modal" id="dismiss-review-modal-{{.Review.ID}}"> |  | ||||||
| 									<div class="header"> |  | ||||||
| 										{{ctx.Locale.Tr "repo.issues.dismiss_review"}} |  | ||||||
| 									</div> |  | ||||||
| 									<div class="content"> |  | ||||||
| 										<div class="ui warning message"> |  | ||||||
| 											{{ctx.Locale.Tr "repo.issues.dismiss_review_warning"}} |  | ||||||
| 										</div> |  | ||||||
| 										<form class="ui form dismiss-review-form" id="dismiss-review-{{.Review.ID}}" action="{{$.RepoLink}}/issues/dismiss_review" method="post"> |  | ||||||
| 											{{$.CsrfTokenHtml}} |  | ||||||
| 											<input type="hidden" name="review_id" value="{{.Review.ID}}"> |  | ||||||
| 											<div class="field"> |  | ||||||
| 												<label for="message">{{ctx.Locale.Tr "action.review_dismissed_reason"}}</label> |  | ||||||
| 												<input id="message" name="message"> |  | ||||||
| 											</div> |  | ||||||
| 											<div class="text right actions"> |  | ||||||
| 												<button class="ui cancel button">{{ctx.Locale.Tr "settings.cancel"}}</button> |  | ||||||
| 												<button class="ui red button" type="submit">{{ctx.Locale.Tr "ok"}}</button> |  | ||||||
| 											</div> |  | ||||||
| 										</form> |  | ||||||
| 									</div> |  | ||||||
| 								</div> |  | ||||||
| 							{{end}} |  | ||||||
| 							{{if .Review.Stale}} |  | ||||||
| 								<span data-tooltip-content="{{ctx.Locale.Tr "repo.issues.is_stale"}}"> |  | ||||||
| 									{{svg "octicon-hourglass" 16}} |  | ||||||
| 								</span> |  | ||||||
| 							{{end}} |  | ||||||
| 							{{if and .CanChange (or .Checked (and (not $.Issue.IsClosed) (not $.Issue.PullRequest.HasMerged)))}} |  | ||||||
| 								<a href="#" class="ui muted icon re-request-review{{if .Checked}} checked{{end}}" data-tooltip-content="{{if .Checked}}{{ctx.Locale.Tr "repo.issues.remove_request_review"}}{{else}}{{ctx.Locale.Tr "repo.issues.re_request_review"}}{{end}}" data-issue-id="{{$.Issue.ID}}" data-id="{{.ItemID}}" data-update-url="{{$.RepoLink}}/issues/request_review">{{svg (Iif .Checked "octicon-trash" "octicon-sync")}}</a> |  | ||||||
| 							{{end}} |  | ||||||
| 							<span {{if .Review.TooltipContent}}data-tooltip-content="{{ctx.Locale.Tr .Review.TooltipContent}}"{{end}}> |  | ||||||
| 								{{svg (printf "octicon-%s" .Review.Type.Icon) 16 (printf "text %s" (.Review.HTMLTypeColorName))}} |  | ||||||
| 							</span> |  | ||||||
| 						</div> |  | ||||||
| 					</div> |  | ||||||
| 				{{end}} |  | ||||||
| 				{{range .OriginalReviews}} |  | ||||||
| 					<div class="item tw-flex tw-items-center tw-py-2"> |  | ||||||
| 						<div class="tw-flex tw-items-center tw-flex-1"> |  | ||||||
| 							<a class="muted" href="{{$.Repository.OriginalURL}}" data-tooltip-content="{{ctx.Locale.Tr "repo.migrated_from_fake" $.Repository.GetOriginalURLHostname}}"> |  | ||||||
| 								{{svg (MigrationIcon $.Repository.GetOriginalURLHostname) 20 "tw-mr-2"}} |  | ||||||
| 								{{.OriginalAuthor}} |  | ||||||
| 							</a> |  | ||||||
| 						</div> |  | ||||||
| 						<div class="tw-flex tw-items-center tw-gap-2"> |  | ||||||
| 							<span {{if .TooltipContent}}data-tooltip-content="{{ctx.Locale.Tr .TooltipContent}}"{{end}}> |  | ||||||
| 								{{svg (printf "octicon-%s" .Type.Icon) 16 (printf "text %s" (.HTMLTypeColorName))}} |  | ||||||
| 							</span> |  | ||||||
| 						</div> |  | ||||||
| 					</div> |  | ||||||
| 				{{end}} |  | ||||||
| 			</div> |  | ||||||
| 		</div> |  | ||||||
| 		{{if and (or .HasIssuesOrPullsWritePermission .IsIssuePoster) (not .HasMerged) (not .Issue.IsClosed) (not .IsPullWorkInProgress)}} |  | ||||||
| 			<div class="toggle-wip" data-title="{{.Issue.Title}}" data-wip-prefix="{{index .PullRequestWorkInProgressPrefixes 0}}" data-update-url="{{.Issue.Link}}/title"> |  | ||||||
| 				<a class="muted"> |  | ||||||
| 					{{ctx.Locale.Tr "repo.pulls.still_in_progress"}} {{ctx.Locale.Tr "repo.pulls.add_prefix" (index .PullRequestWorkInProgressPrefixes 0)}} |  | ||||||
| 				</a> |  | ||||||
| 			</div> |  | ||||||
| 		{{end}} |  | ||||||
| 		<div class="divider"></div> | 		<div class="divider"></div> | ||||||
| 	{{end}} | 	{{end}} | ||||||
|  |  | ||||||
| 	{{template "repo/issue/labels/labels_selector_field" .}} | 	{{template "repo/issue/labels/labels_selector_field" $}} | ||||||
| 	{{template "repo/issue/labels/labels_sidebar" dict "root" $}} | 	{{template "repo/issue/labels/labels_sidebar" dict "root" $}} | ||||||
|  |  | ||||||
| 	<div class="divider"></div> | 	{{template "repo/issue/sidebar/milestone_list" $}} | ||||||
|  | 	{{template "repo/issue/sidebar/project_list" $}} | ||||||
| 	<div class="ui {{if or (not .HasIssuesOrPullsWritePermission) .Repository.IsArchived}}disabled{{end}} floating jump select-milestone dropdown"> | 	{{template "repo/issue/sidebar/assignee_list" $}} | ||||||
| 		<a class="text muted flex-text-block"> | 	{{template "repo/issue/sidebar/participant_list" $}} | ||||||
| 			<strong>{{ctx.Locale.Tr "repo.issues.new.milestone"}}</strong> | 	{{template "repo/issue/sidebar/watch_notification" $}} | ||||||
| 			{{if and .HasIssuesOrPullsWritePermission (not .Repository.IsArchived)}} | 	{{template "repo/issue/sidebar/stopwatch_timetracker" $}} | ||||||
| 				{{svg "octicon-gear" 16 "tw-ml-1"}} | 	{{template "repo/issue/sidebar/due_date" $}} | ||||||
| 			{{end}} | 	{{template "repo/issue/sidebar/issue_dependencies" $}} | ||||||
| 		</a> | 	{{template "repo/issue/sidebar/reference_link" $}} | ||||||
| 		<div class="menu" data-action="update" data-issue-id="{{$.Issue.ID}}" data-update-url="{{$.RepoLink}}/issues/milestone"> | 	{{template "repo/issue/sidebar/issue_management" $}} | ||||||
| 			{{template "repo/issue/milestone/select_menu" .}} | 	{{template "repo/issue/sidebar/allow_maintainer_edit" $}} | ||||||
| 		</div> |  | ||||||
| 	</div> |  | ||||||
| 	<div class="ui select-milestone list"> |  | ||||||
| 		<span class="no-select item {{if .Issue.Milestone}}tw-hidden{{end}}">{{ctx.Locale.Tr "repo.issues.new.no_milestone"}}</span> |  | ||||||
| 		<div class="selected"> |  | ||||||
| 			{{if .Issue.Milestone}} |  | ||||||
| 				<a class="item muted sidebar-item-link" href="{{.RepoLink}}/milestone/{{.Issue.Milestone.ID}}"> |  | ||||||
| 					{{svg "octicon-milestone" 18 "tw-mr-2"}} |  | ||||||
| 					{{.Issue.Milestone.Name}} |  | ||||||
| 				</a> |  | ||||||
| 			{{end}} |  | ||||||
| 		</div> |  | ||||||
| 	</div> |  | ||||||
|  |  | ||||||
| 	{{if .IsProjectsEnabled}} |  | ||||||
| 		<div class="divider"></div> |  | ||||||
|  |  | ||||||
| 		<div class="ui {{if or (not .HasIssuesOrPullsWritePermission) .Repository.IsArchived}}disabled{{end}} floating jump select-project dropdown"> |  | ||||||
| 			<a class="text muted flex-text-block"> |  | ||||||
| 				<strong>{{ctx.Locale.Tr "repo.issues.new.projects"}}</strong> |  | ||||||
| 				{{if and .HasIssuesOrPullsWritePermission (not .Repository.IsArchived)}} |  | ||||||
| 					{{svg "octicon-gear" 16 "tw-ml-1"}} |  | ||||||
| 				{{end}} |  | ||||||
| 			</a> |  | ||||||
| 			<div class="menu" data-action="update" data-issue-id="{{$.Issue.ID}}" data-update-url="{{$.RepoLink}}/issues/projects"> |  | ||||||
| 				{{if or .OpenProjects .ClosedProjects}} |  | ||||||
| 				<div class="ui icon search input"> |  | ||||||
| 					<i class="icon">{{svg "octicon-search" 16}}</i> |  | ||||||
| 					<input type="text" placeholder="{{ctx.Locale.Tr "repo.issues.filter_projects"}}"> |  | ||||||
| 				</div> |  | ||||||
| 				{{end}} |  | ||||||
| 				<div class="no-select item">{{ctx.Locale.Tr "repo.issues.new.clear_projects"}}</div> |  | ||||||
| 				{{if .OpenProjects}} |  | ||||||
| 					<div class="divider"></div> |  | ||||||
| 					<div class="header"> |  | ||||||
| 						{{ctx.Locale.Tr "repo.issues.new.open_projects"}} |  | ||||||
| 					</div> |  | ||||||
| 					{{range .OpenProjects}} |  | ||||||
| 						<a class="item muted sidebar-item-link" data-id="{{.ID}}" data-href="{{.Link ctx}}"> |  | ||||||
| 							{{svg .IconName 18 "tw-mr-2"}}{{.Title}} |  | ||||||
| 						</a> |  | ||||||
| 					{{end}} |  | ||||||
| 				{{end}} |  | ||||||
| 				{{if .ClosedProjects}} |  | ||||||
| 					<div class="divider"></div> |  | ||||||
| 					<div class="header"> |  | ||||||
| 						{{ctx.Locale.Tr "repo.issues.new.closed_projects"}} |  | ||||||
| 					</div> |  | ||||||
| 					{{range .ClosedProjects}} |  | ||||||
| 						<a class="item muted sidebar-item-link" data-id="{{.ID}}" data-href="{{.Link ctx}}"> |  | ||||||
| 							{{svg .IconName 18 "tw-mr-2"}}{{.Title}} |  | ||||||
| 						</a> |  | ||||||
| 					{{end}} |  | ||||||
| 				{{end}} |  | ||||||
| 			</div> |  | ||||||
| 		</div> |  | ||||||
| 		<div class="ui select-project list"> |  | ||||||
| 			<span class="no-select item {{if .Issue.Project}}tw-hidden{{end}}">{{ctx.Locale.Tr "repo.issues.new.no_projects"}}</span> |  | ||||||
| 			<div class="selected"> |  | ||||||
| 				{{if .Issue.Project}} |  | ||||||
| 					<a class="item muted sidebar-item-link" href="{{.Issue.Project.Link ctx}}"> |  | ||||||
| 						{{svg .Issue.Project.IconName 18 "tw-mr-2"}}{{.Issue.Project.Title}} |  | ||||||
| 					</a> |  | ||||||
| 				{{end}} |  | ||||||
| 			</div> |  | ||||||
| 		</div> |  | ||||||
| 	{{end}} |  | ||||||
|  |  | ||||||
| 	<div class="divider"></div> |  | ||||||
|  |  | ||||||
| 	<input id="assignee_id" name="assignee_id" type="hidden" value="{{.assignee_id}}"> |  | ||||||
| 	<div class="ui {{if or (not .HasIssuesOrPullsWritePermission) .Repository.IsArchived}}disabled{{end}} floating jump select-assignees-modify dropdown"> |  | ||||||
| 		<a class="text muted flex-text-block"> |  | ||||||
| 			<strong>{{ctx.Locale.Tr "repo.issues.new.assignees"}}</strong> |  | ||||||
| 			{{if and .HasIssuesOrPullsWritePermission (not .Repository.IsArchived)}} |  | ||||||
| 				{{svg "octicon-gear" 16 "tw-ml-1"}} |  | ||||||
| 			{{end}} |  | ||||||
| 		</a> |  | ||||||
| 		<div class="filter menu" data-action="update" data-issue-id="{{$.Issue.ID}}" data-update-url="{{$.RepoLink}}/issues/assignee"> |  | ||||||
| 			<div class="ui icon search input"> |  | ||||||
| 				<i class="icon">{{svg "octicon-search" 16}}</i> |  | ||||||
| 				<input type="text" placeholder="{{ctx.Locale.Tr "repo.issues.filter_assignees"}}"> |  | ||||||
| 			</div> |  | ||||||
| 			<div class="no-select item">{{ctx.Locale.Tr "repo.issues.new.clear_assignees"}}</div> |  | ||||||
| 			{{range .Assignees}} |  | ||||||
|  |  | ||||||
| 				{{$AssigneeID := .ID}} |  | ||||||
| 				<a class="item{{range $.Issue.Assignees}}{{if eq .ID $AssigneeID}} checked{{end}}{{end}}" href="#" data-id="{{.ID}}" data-id-selector="#assignee_{{.ID}}"> |  | ||||||
| 					{{$checked := false}} |  | ||||||
| 					{{range $.Issue.Assignees}} |  | ||||||
| 						{{if eq .ID $AssigneeID}} |  | ||||||
| 							{{$checked = true}} |  | ||||||
| 						{{end}} |  | ||||||
| 					{{end}} |  | ||||||
| 					<span class="octicon-check {{if not $checked}}tw-invisible{{end}}">{{svg "octicon-check"}}</span> |  | ||||||
| 					<span class="text"> |  | ||||||
| 						{{ctx.AvatarUtils.Avatar . 20 "tw-mr-2"}}{{template "repo/search_name" .}} |  | ||||||
| 					</span> |  | ||||||
| 				</a> |  | ||||||
| 			{{end}} |  | ||||||
| 		</div> |  | ||||||
| 	</div> |  | ||||||
| 	<div class="ui assignees list"> |  | ||||||
| 		<span class="no-select item {{if .Issue.Assignees}}tw-hidden{{end}}">{{ctx.Locale.Tr "repo.issues.new.no_assignees"}}</span> |  | ||||||
| 		<div class="selected"> |  | ||||||
| 			{{range .Issue.Assignees}} |  | ||||||
| 				<div class="item"> |  | ||||||
| 					<a class="muted sidebar-item-link" href="{{$.RepoLink}}/{{if $.Issue.IsPull}}pulls{{else}}issues{{end}}?assignee={{.ID}}"> |  | ||||||
| 						{{ctx.AvatarUtils.Avatar . 28 "tw-mr-2"}} |  | ||||||
| 						{{.GetDisplayName}} |  | ||||||
| 					</a> |  | ||||||
| 				</div> |  | ||||||
| 			{{end}} |  | ||||||
| 		</div> |  | ||||||
| 	</div> |  | ||||||
|  |  | ||||||
| 	<div class="divider"></div> |  | ||||||
|  |  | ||||||
| 	{{if .Participants}} |  | ||||||
| 		<span class="text"><strong>{{ctx.Locale.Tr "repo.issues.num_participants" .NumParticipants}}</strong></span> |  | ||||||
| 		<div class="ui list tw-flex tw-flex-wrap"> |  | ||||||
| 			{{range .Participants}} |  | ||||||
| 				<a {{if gt .ID 0}}href="{{.HomeLink}}"{{end}} data-tooltip-content="{{.GetDisplayName}}"> |  | ||||||
| 					{{ctx.AvatarUtils.Avatar . 28 "tw-my-0.5 tw-mr-1"}} |  | ||||||
| 				</a> |  | ||||||
| 			{{end}} |  | ||||||
| 		</div> |  | ||||||
| 	{{end}} |  | ||||||
|  |  | ||||||
| 	{{if and $.IssueWatch (not .Repository.IsArchived)}} |  | ||||||
| 		<div class="divider"></div> |  | ||||||
|  |  | ||||||
| 		<div class="ui watching"> |  | ||||||
| 			<span class="text"><strong>{{ctx.Locale.Tr "notification.notifications"}}</strong></span> |  | ||||||
| 			<div class="tw-mt-2"> |  | ||||||
| 				{{template "repo/issue/view_content/watching" .}} |  | ||||||
| 			</div> |  | ||||||
| 		</div> |  | ||||||
| 	{{end}} |  | ||||||
| 	{{if .Repository.IsTimetrackerEnabled ctx}} |  | ||||||
| 		{{if and .CanUseTimetracker (not .Repository.IsArchived)}} |  | ||||||
| 			<div class="divider"></div> |  | ||||||
| 			<div class="ui timetrack"> |  | ||||||
| 				<span class="text"><strong>{{ctx.Locale.Tr "repo.issues.tracker"}}</strong></span> |  | ||||||
| 				<div class="tw-mt-2"> |  | ||||||
| 					<form method="post" action="{{.Issue.Link}}/times/stopwatch/toggle" id="toggle_stopwatch_form"> |  | ||||||
| 						{{$.CsrfTokenHtml}} |  | ||||||
| 					</form> |  | ||||||
| 					<form method="post" action="{{.Issue.Link}}/times/stopwatch/cancel" id="cancel_stopwatch_form"> |  | ||||||
| 						{{$.CsrfTokenHtml}} |  | ||||||
| 					</form> |  | ||||||
| 					{{if $.IsStopwatchRunning}} |  | ||||||
| 						<button class="ui fluid button issue-stop-time"> |  | ||||||
| 							{{svg "octicon-stopwatch" 16 "tw-mr-2"}} |  | ||||||
| 							{{ctx.Locale.Tr "repo.issues.stop_tracking"}} |  | ||||||
| 						</button> |  | ||||||
| 						<button class="ui fluid button issue-cancel-time tw-mt-2"> |  | ||||||
| 							{{svg "octicon-trash" 16 "tw-mr-2"}} |  | ||||||
| 							{{ctx.Locale.Tr "repo.issues.cancel_tracking"}} |  | ||||||
| 						</button> |  | ||||||
| 					{{else}} |  | ||||||
| 						{{if .HasUserStopwatch}} |  | ||||||
| 							<div class="ui warning message"> |  | ||||||
| 								{{ctx.Locale.Tr "repo.issues.tracking_already_started" .OtherStopwatchURL}} |  | ||||||
| 							</div> |  | ||||||
| 						{{end}} |  | ||||||
| 						<button class="ui fluid button issue-start-time" data-tooltip-content='{{ctx.Locale.Tr "repo.issues.start_tracking"}}'> |  | ||||||
| 							{{svg "octicon-stopwatch" 16 "tw-mr-2"}} |  | ||||||
| 							{{ctx.Locale.Tr "repo.issues.start_tracking_short"}} |  | ||||||
| 						</button> |  | ||||||
| 						<div class="ui mini modal issue-start-time-modal"> |  | ||||||
| 							<div class="header">{{ctx.Locale.Tr "repo.issues.add_time"}}</div> |  | ||||||
| 							<div class="content"> |  | ||||||
| 								<form method="post" id="add_time_manual_form" action="{{.Issue.Link}}/times/add" class="ui input fluid tw-gap-2"> |  | ||||||
| 									{{$.CsrfTokenHtml}} |  | ||||||
| 									<input placeholder='{{ctx.Locale.Tr "repo.issues.add_time_hours"}}' type="number" name="hours"> |  | ||||||
| 									<input placeholder='{{ctx.Locale.Tr "repo.issues.add_time_minutes"}}' type="number" name="minutes" class="ui compact"> |  | ||||||
| 								</form> |  | ||||||
| 							</div> |  | ||||||
| 							<div class="actions"> |  | ||||||
| 								<button class="ui primary approve button">{{ctx.Locale.Tr "repo.issues.add_time_short"}}</button> |  | ||||||
| 								<button class="ui cancel button">{{ctx.Locale.Tr "repo.issues.add_time_cancel"}}</button> |  | ||||||
| 							</div> |  | ||||||
| 						</div> |  | ||||||
| 						<button class="ui fluid button issue-add-time tw-mt-2" data-tooltip-content='{{ctx.Locale.Tr "repo.issues.add_time"}}'> |  | ||||||
| 							{{svg "octicon-plus" 16 "tw-mr-2"}} |  | ||||||
| 							{{ctx.Locale.Tr "repo.issues.add_time_short"}} |  | ||||||
| 						</button> |  | ||||||
| 					{{end}} |  | ||||||
| 				</div> |  | ||||||
| 			</div> |  | ||||||
| 		{{end}} |  | ||||||
| 		{{if .WorkingUsers}} |  | ||||||
| 			<div class="divider"></div> |  | ||||||
| 			<div class="ui comments"> |  | ||||||
| 				<span class="text"><strong>{{ctx.Locale.Tr "repo.issues.time_spent_from_all_authors" ($.Issue.TotalTrackedTime | Sec2Time)}}</strong></span> |  | ||||||
| 				<div> |  | ||||||
| 					{{range $user, $trackedtime := .WorkingUsers}} |  | ||||||
| 						<div class="comment tw-mt-2"> |  | ||||||
| 							<a class="avatar"> |  | ||||||
| 								{{ctx.AvatarUtils.Avatar $user}} |  | ||||||
| 							</a> |  | ||||||
| 							<div class="content"> |  | ||||||
| 								{{template "shared/user/authorlink" $user}} |  | ||||||
| 								<div class="text"> |  | ||||||
| 									{{$trackedtime|Sec2Time}} |  | ||||||
| 								</div> |  | ||||||
| 							</div> |  | ||||||
| 						</div> |  | ||||||
| 					{{end}} |  | ||||||
| 				</div> |  | ||||||
| 			</div> |  | ||||||
| 		{{end}} |  | ||||||
| 	{{end}} |  | ||||||
|  |  | ||||||
| 	<div class="divider"></div> |  | ||||||
| 	<span class="text"><strong>{{ctx.Locale.Tr "repo.issues.due_date"}}</strong></span> |  | ||||||
| 	<div class="ui form tw-mt-2"> |  | ||||||
| 		{{if .Issue.DeadlineUnix}} |  | ||||||
| 			<div class="tw-flex tw-justify-between tw-items-center tw-gap-2"> |  | ||||||
| 				<div class="due-date {{if .Issue.IsOverdue}}text red{{end}}" {{if .Issue.IsOverdue}}data-tooltip-content="{{ctx.Locale.Tr "repo.issues.due_date_overdue"}}"{{end}}> |  | ||||||
| 					{{svg "octicon-calendar"}} {{DateUtils.AbsoluteLong .Issue.DeadlineUnix}} |  | ||||||
| 				</div> |  | ||||||
| 				<div class="flex-text-block"> |  | ||||||
| 					{{if and .HasIssuesOrPullsWritePermission (not .Repository.IsArchived)}} |  | ||||||
| 						<a class="issue-due-edit muted" data-tooltip-content="{{ctx.Locale.Tr "repo.issues.due_date_form_edit"}}">{{svg "octicon-pencil"}}</a> |  | ||||||
| 						<a class="issue-due-remove muted" data-tooltip-content="{{ctx.Locale.Tr "repo.issues.due_date_form_remove"}}">{{svg "octicon-trash"}}</a> |  | ||||||
| 					{{end}} |  | ||||||
| 				</div> |  | ||||||
| 			</div> |  | ||||||
| 		{{else}} |  | ||||||
| 			{{ctx.Locale.Tr "repo.issues.due_date_not_set"}} |  | ||||||
| 		{{end}} |  | ||||||
|  |  | ||||||
| 		{{if and .HasIssuesOrPullsWritePermission (not .Repository.IsArchived)}} |  | ||||||
| 			<form class="ui fluid action input issue-due-form form-fetch-action tw-mt-2 {{if .Issue.DeadlineUnix}}tw-hidden{{end}}" |  | ||||||
| 						method="post" action="{{AppSubUrl}}/{{PathEscape .Repository.Owner.Name}}/{{PathEscape .Repository.Name}}/issues/{{.Issue.Index}}/deadline" |  | ||||||
| 			> |  | ||||||
| 				{{$.CsrfTokenHtml}} |  | ||||||
| 				<input required type="date" name="deadline" placeholder="{{ctx.Locale.Tr "repo.issues.due_date_form"}}" {{if .Issue.DeadlineUnix}}value="{{.Issue.DeadlineUnix.FormatDate}}"{{end}}> |  | ||||||
| 				<button class="ui icon button">{{Iif .Issue.DeadlineUnix (svg "octicon-pencil") (svg "octicon-plus")}}</button> |  | ||||||
| 			</form> |  | ||||||
| 		{{end}} |  | ||||||
| 	</div> |  | ||||||
|  |  | ||||||
| 	{{if .Repository.IsDependenciesEnabled ctx}} |  | ||||||
| 		<div class="divider"></div> |  | ||||||
|  |  | ||||||
| 		<div class="ui depending"> |  | ||||||
| 			{{if (and (not .BlockedByDependencies) (not .BlockedByDependenciesNotPermitted) (not .BlockingDependencies) (not .BlockingDependenciesNotPermitted))}} |  | ||||||
| 				<span class="text"><strong>{{ctx.Locale.Tr "repo.issues.dependency.title"}}</strong></span> |  | ||||||
| 				<br> |  | ||||||
| 				<p> |  | ||||||
| 					{{if .Issue.IsPull}} |  | ||||||
| 						{{ctx.Locale.Tr "repo.issues.dependency.pr_no_dependencies"}} |  | ||||||
| 					{{else}} |  | ||||||
| 						{{ctx.Locale.Tr "repo.issues.dependency.issue_no_dependencies"}} |  | ||||||
| 					{{end}} |  | ||||||
| 				</p> |  | ||||||
| 			{{end}} |  | ||||||
|  |  | ||||||
| 			{{if or .BlockingDependencies .BlockingDependenciesNotPermitted}} |  | ||||||
| 				<span class="text" data-tooltip-content="{{if .Issue.IsPull}}{{ctx.Locale.Tr "repo.issues.dependency.pr_close_blocks"}}{{else}}{{ctx.Locale.Tr "repo.issues.dependency.issue_close_blocks"}}{{end}}"> |  | ||||||
| 					<strong>{{ctx.Locale.Tr "repo.issues.dependency.blocks_short"}}</strong> |  | ||||||
| 				</span> |  | ||||||
| 				<div class="ui relaxed divided list"> |  | ||||||
| 					{{range .BlockingDependencies}} |  | ||||||
| 						<div class="item dependency{{if .Issue.IsClosed}} is-closed{{end}} tw-flex tw-items-center tw-justify-between"> |  | ||||||
| 							<div class="item-left tw-flex tw-justify-center tw-flex-col tw-flex-1 gt-ellipsis"> |  | ||||||
| 								<a class="title muted" href="{{.Issue.Link}}" data-tooltip-content="#{{.Issue.Index}} {{.Issue.Title | ctx.RenderUtils.RenderEmoji}}"> |  | ||||||
| 									#{{.Issue.Index}} {{.Issue.Title | ctx.RenderUtils.RenderEmoji}} |  | ||||||
| 								</a> |  | ||||||
| 								<div class="text small gt-ellipsis" data-tooltip-content="{{.Repository.OwnerName}}/{{.Repository.Name}}"> |  | ||||||
| 									{{.Repository.OwnerName}}/{{.Repository.Name}} |  | ||||||
| 								</div> |  | ||||||
| 							</div> |  | ||||||
| 							<div class="item-right tw-flex tw-items-center tw-m-1"> |  | ||||||
| 								{{if and $.CanCreateIssueDependencies (not $.Repository.IsArchived)}} |  | ||||||
| 									<a class="delete-dependency-button ci muted" data-id="{{.Issue.ID}}" data-type="blocking" data-tooltip-content="{{ctx.Locale.Tr "repo.issues.dependency.remove_info"}}"> |  | ||||||
| 										{{svg "octicon-trash" 16}} |  | ||||||
| 									</a> |  | ||||||
| 								{{end}} |  | ||||||
| 							</div> |  | ||||||
| 						</div> |  | ||||||
| 					{{end}} |  | ||||||
| 					{{if .BlockingDependenciesNotPermitted}} |  | ||||||
| 						<div class="item tw-flex tw-items-center tw-justify-between gt-ellipsis"> |  | ||||||
| 							<span>{{ctx.Locale.TrN (len .BlockingDependenciesNotPermitted) "repo.issues.dependency.no_permission_1" "repo.issues.dependency.no_permission_n" (len .BlockingDependenciesNotPermitted)}}</span> |  | ||||||
| 						</div> |  | ||||||
| 					{{end}} |  | ||||||
| 				</div> |  | ||||||
| 			{{end}} |  | ||||||
|  |  | ||||||
| 			{{if or .BlockedByDependencies .BlockedByDependenciesNotPermitted}} |  | ||||||
| 				<span class="text" data-tooltip-content="{{if .Issue.IsPull}}{{ctx.Locale.Tr "repo.issues.dependency.pr_closing_blockedby"}}{{else}}{{ctx.Locale.Tr "repo.issues.dependency.issue_closing_blockedby"}}{{end}}"> |  | ||||||
| 					<strong>{{ctx.Locale.Tr "repo.issues.dependency.blocked_by_short"}}</strong> |  | ||||||
| 				</span> |  | ||||||
| 				<div class="ui relaxed divided list"> |  | ||||||
| 					{{range .BlockedByDependencies}} |  | ||||||
| 						<div class="item dependency{{if .Issue.IsClosed}} is-closed{{end}} tw-flex tw-items-center tw-justify-between"> |  | ||||||
| 							<div class="item-left tw-flex tw-justify-center tw-flex-col tw-flex-1 gt-ellipsis"> |  | ||||||
| 								<a class="title muted" href="{{.Issue.Link}}" data-tooltip-content="#{{.Issue.Index}} {{.Issue.Title | ctx.RenderUtils.RenderEmoji}}"> |  | ||||||
| 									#{{.Issue.Index}} {{.Issue.Title | ctx.RenderUtils.RenderEmoji}} |  | ||||||
| 								</a> |  | ||||||
| 								<div class="text small gt-ellipsis" data-tooltip-content="{{.Repository.OwnerName}}/{{.Repository.Name}}"> |  | ||||||
| 									{{.Repository.OwnerName}}/{{.Repository.Name}} |  | ||||||
| 								</div> |  | ||||||
| 							</div> |  | ||||||
| 							<div class="item-right tw-flex tw-items-center tw-m-1"> |  | ||||||
| 								{{if and $.CanCreateIssueDependencies (not $.Repository.IsArchived)}} |  | ||||||
| 									<a class="delete-dependency-button ci muted" data-id="{{.Issue.ID}}" data-type="blockedBy" data-tooltip-content="{{ctx.Locale.Tr "repo.issues.dependency.remove_info"}}"> |  | ||||||
| 										{{svg "octicon-trash" 16}} |  | ||||||
| 									</a> |  | ||||||
| 								{{end}} |  | ||||||
| 							</div> |  | ||||||
| 						</div> |  | ||||||
| 					{{end}} |  | ||||||
| 					{{if $.CanCreateIssueDependencies}} |  | ||||||
| 						{{range .BlockedByDependenciesNotPermitted}} |  | ||||||
| 							<div class="item dependency{{if .Issue.IsClosed}} is-closed{{end}} tw-flex tw-items-center tw-justify-between"> |  | ||||||
| 								<div class="item-left tw-flex tw-justify-center tw-flex-col tw-flex-1 gt-ellipsis"> |  | ||||||
| 									<div class="gt-ellipsis"> |  | ||||||
| 										<span data-tooltip-content="{{ctx.Locale.Tr "repo.issues.dependency.no_permission.can_remove"}}">{{svg "octicon-lock" 16}}</span> |  | ||||||
| 										<span class="title" data-tooltip-content="#{{.Issue.Index}} {{.Issue.Title | ctx.RenderUtils.RenderEmoji}}"> |  | ||||||
| 											#{{.Issue.Index}} {{.Issue.Title | ctx.RenderUtils.RenderEmoji}} |  | ||||||
| 										</span> |  | ||||||
| 									</div> |  | ||||||
| 									<div class="text small gt-ellipsis" data-tooltip-content="{{.Repository.OwnerName}}/{{.Repository.Name}}"> |  | ||||||
| 										{{.Repository.OwnerName}}/{{.Repository.Name}} |  | ||||||
| 									</div> |  | ||||||
| 								</div> |  | ||||||
| 								<div class="item-right tw-flex tw-items-center tw-m-1"> |  | ||||||
| 									{{if and $.CanCreateIssueDependencies (not $.Repository.IsArchived)}} |  | ||||||
| 										<a class="delete-dependency-button ci muted" data-id="{{.Issue.ID}}" data-type="blocking" data-tooltip-content="{{ctx.Locale.Tr "repo.issues.dependency.remove_info"}}"> |  | ||||||
| 											{{svg "octicon-trash" 16}} |  | ||||||
| 										</a> |  | ||||||
| 									{{end}} |  | ||||||
| 								</div> |  | ||||||
| 							</div> |  | ||||||
| 						{{end}} |  | ||||||
| 					{{else if .BlockedByDependenciesNotPermitted}} |  | ||||||
| 						<div class="item tw-flex tw-items-center tw-justify-between gt-ellipsis"> |  | ||||||
| 							<span>{{ctx.Locale.TrN (len .BlockedByDependenciesNotPermitted) "repo.issues.dependency.no_permission_1" "repo.issues.dependency.no_permission_n" (len .BlockedByDependenciesNotPermitted)}}</span> |  | ||||||
| 						</div> |  | ||||||
| 					{{end}} |  | ||||||
| 				</div> |  | ||||||
| 			{{end}} |  | ||||||
|  |  | ||||||
| 			{{if and .CanCreateIssueDependencies (not .Repository.IsArchived)}} |  | ||||||
| 				<div> |  | ||||||
| 					<form method="post" action="{{.Issue.Link}}/dependency/add" id="addDependencyForm"> |  | ||||||
| 						{{$.CsrfTokenHtml}} |  | ||||||
| 						<div class="ui fluid action input"> |  | ||||||
| 							<div class="ui search selection dropdown" id="new-dependency-drop-list" data-issue-id="{{.Issue.ID}}"> |  | ||||||
| 								<input name="newDependency" type="hidden"> |  | ||||||
| 								{{svg "octicon-triangle-down" 14 "dropdown icon"}} |  | ||||||
| 								<input type="text" class="search"> |  | ||||||
| 								<div class="default text">{{ctx.Locale.Tr "repo.issues.dependency.add"}}</div> |  | ||||||
| 							</div> |  | ||||||
| 							<button class="ui icon button"> |  | ||||||
| 								{{svg "octicon-plus"}} |  | ||||||
| 							</button> |  | ||||||
| 						</div> |  | ||||||
| 					</form> |  | ||||||
| 				</div> |  | ||||||
| 			{{end}} |  | ||||||
| 		</div> |  | ||||||
|  |  | ||||||
| 		{{if and .CanCreateIssueDependencies (not .Repository.IsArchived)}} |  | ||||||
| 			<input type="hidden" id="crossRepoSearch" value="{{.AllowCrossRepositoryDependencies}}"> |  | ||||||
|  |  | ||||||
| 			<div class="ui g-modal-confirm modal remove-dependency"> |  | ||||||
| 				<div class="header"> |  | ||||||
| 					{{svg "octicon-trash"}} |  | ||||||
| 					{{ctx.Locale.Tr "repo.issues.dependency.remove_header"}} |  | ||||||
| 				</div> |  | ||||||
| 				<div class="content"> |  | ||||||
| 					<form method="post" action="{{.Issue.Link}}/dependency/delete" id="removeDependencyForm"> |  | ||||||
| 						{{$.CsrfTokenHtml}} |  | ||||||
| 						<input type="hidden" value="" name="removeDependencyID" id="removeDependencyID"> |  | ||||||
| 						<input type="hidden" value="" name="dependencyType" id="dependencyType"> |  | ||||||
| 					</form> |  | ||||||
| 					<p>{{if .Issue.IsPull}} |  | ||||||
| 						{{ctx.Locale.Tr "repo.issues.dependency.pr_remove_text"}} |  | ||||||
| 					{{else}} |  | ||||||
| 						{{ctx.Locale.Tr "repo.issues.dependency.issue_remove_text"}} |  | ||||||
| 					{{end}}</p> |  | ||||||
| 				</div> |  | ||||||
| 				{{$ModalButtonCancelText := ctx.Locale.Tr "repo.issues.dependency.cancel"}} |  | ||||||
| 				{{$ModalButtonOkText := ctx.Locale.Tr "repo.issues.dependency.remove"}} |  | ||||||
| 				{{template "base/modal_actions_confirm" (dict "." . "ModalButtonCancelText" $ModalButtonCancelText "ModalButtonOkText" $ModalButtonOkText)}} |  | ||||||
| 			</div> |  | ||||||
| 		{{end}} |  | ||||||
| 	{{end}} |  | ||||||
|  |  | ||||||
| 	<div class="divider"></div> |  | ||||||
| 	<div class="ui equal width compact grid"> |  | ||||||
| 		{{$issueReferenceLink := printf "%s#%d" .Issue.Repo.FullName .Issue.Index}} |  | ||||||
| 		<div class="row tw-items-center" data-tooltip-content="{{$issueReferenceLink}}"> |  | ||||||
| 			<span class="text column truncate">{{ctx.Locale.Tr "repo.issues.reference_link" $issueReferenceLink}}</span> |  | ||||||
| 			<button class="ui two wide button column tw-p-2" data-clipboard-text="{{$issueReferenceLink}}">{{svg "octicon-copy" 14}}</button> |  | ||||||
| 		</div> |  | ||||||
| 	</div> |  | ||||||
|  |  | ||||||
| 	{{if and .IsRepoAdmin (not .Repository.IsArchived)}} |  | ||||||
| 		<div class="divider"></div> |  | ||||||
|  |  | ||||||
| 		{{if or .PinEnabled .Issue.IsPinned}} |  | ||||||
| 			<form class="tw-mt-1 form-fetch-action single-button-form" method="post" {{if $.NewPinAllowed}}action="{{.Issue.Link}}/pin"{{else}}data-tooltip-content="{{ctx.Locale.Tr "repo.issues.max_pinned"}}"{{end}}> |  | ||||||
| 				{{$.CsrfTokenHtml}} |  | ||||||
| 				<button class="fluid ui button {{if not $.NewPinAllowed}}disabled{{end}}"> |  | ||||||
| 					{{if not .Issue.IsPinned}} |  | ||||||
| 						{{svg "octicon-pin" 16 "tw-mr-2"}} |  | ||||||
| 						{{ctx.Locale.Tr "pin"}} |  | ||||||
| 					{{else}} |  | ||||||
| 						{{svg "octicon-pin-slash" 16 "tw-mr-2"}} |  | ||||||
| 						{{ctx.Locale.Tr "unpin"}} |  | ||||||
| 					{{end}} |  | ||||||
| 				</button> |  | ||||||
| 			</form> |  | ||||||
| 		{{end}} |  | ||||||
|  |  | ||||||
| 		<button class="tw-mt-1 fluid ui show-modal button{{if .Issue.IsLocked}} red{{end}}" data-modal="#lock"> |  | ||||||
| 			{{if .Issue.IsLocked}} |  | ||||||
| 				{{svg "octicon-key"}} |  | ||||||
| 				{{ctx.Locale.Tr "repo.issues.unlock"}} |  | ||||||
| 			{{else}} |  | ||||||
| 				{{svg "octicon-lock"}} |  | ||||||
| 				{{ctx.Locale.Tr "repo.issues.lock"}} |  | ||||||
| 			{{end}} |  | ||||||
| 		</button> |  | ||||||
| 		<div class="ui tiny modal" id="lock"> |  | ||||||
| 			<div class="header"> |  | ||||||
| 				{{if .Issue.IsLocked}} |  | ||||||
| 					{{ctx.Locale.Tr "repo.issues.unlock.title"}} |  | ||||||
| 				{{else}} |  | ||||||
| 					{{ctx.Locale.Tr "repo.issues.lock.title"}} |  | ||||||
| 				{{end}} |  | ||||||
| 			</div> |  | ||||||
| 			<div class="content"> |  | ||||||
| 				<div class="ui warning message"> |  | ||||||
| 					{{if .Issue.IsLocked}} |  | ||||||
| 						{{ctx.Locale.Tr "repo.issues.unlock.notice_1"}}<br> |  | ||||||
| 						{{ctx.Locale.Tr "repo.issues.unlock.notice_2"}}<br> |  | ||||||
| 					{{else}} |  | ||||||
| 						{{ctx.Locale.Tr "repo.issues.lock.notice_1"}}<br> |  | ||||||
| 						{{ctx.Locale.Tr "repo.issues.lock.notice_2"}}<br> |  | ||||||
| 						{{ctx.Locale.Tr "repo.issues.lock.notice_3"}}<br> |  | ||||||
| 					{{end}} |  | ||||||
| 				</div> |  | ||||||
|  |  | ||||||
| 				<form class="ui form form-fetch-action" action="{{.Issue.Link}}{{if .Issue.IsLocked}}/unlock{{else}}/lock{{end}}" |  | ||||||
| 					method="post"> |  | ||||||
| 					{{.CsrfTokenHtml}} |  | ||||||
|  |  | ||||||
| 					{{if not .Issue.IsLocked}} |  | ||||||
| 						<div class="field"> |  | ||||||
| 							<strong> {{ctx.Locale.Tr "repo.issues.lock.reason"}} </strong> |  | ||||||
| 						</div> |  | ||||||
|  |  | ||||||
| 						<div class="field"> |  | ||||||
| 							<div class="ui fluid dropdown selection"> |  | ||||||
|  |  | ||||||
| 								<select name="reason"> |  | ||||||
| 									<option value=""> </option> |  | ||||||
| 									{{range .LockReasons}} |  | ||||||
| 										<option value="{{.}}">{{.}}</option> |  | ||||||
| 									{{end}} |  | ||||||
| 								</select> |  | ||||||
| 								{{svg "octicon-triangle-down" 14 "dropdown icon"}} |  | ||||||
|  |  | ||||||
| 								<div class="default text"> </div> |  | ||||||
|  |  | ||||||
| 								<div class="menu"> |  | ||||||
| 									{{range .LockReasons}} |  | ||||||
| 										<div class="item" data-value="{{.}}">{{.}}</div> |  | ||||||
| 									{{end}} |  | ||||||
| 								</div> |  | ||||||
| 							</div> |  | ||||||
| 						</div> |  | ||||||
| 					{{end}} |  | ||||||
|  |  | ||||||
| 					<div class="text right actions"> |  | ||||||
| 						<button class="ui cancel button">{{ctx.Locale.Tr "settings.cancel"}}</button> |  | ||||||
| 						<button class="ui red button"> |  | ||||||
| 							{{if .Issue.IsLocked}} |  | ||||||
| 								{{ctx.Locale.Tr "repo.issues.unlock_confirm"}} |  | ||||||
| 							{{else}} |  | ||||||
| 								{{ctx.Locale.Tr "repo.issues.lock_confirm"}} |  | ||||||
| 							{{end}} |  | ||||||
| 						</button> |  | ||||||
| 					</div> |  | ||||||
| 				</form> |  | ||||||
| 			</div> |  | ||||||
| 		</div> |  | ||||||
| 		<button class="tw-mt-1 fluid ui show-modal button" data-modal="#sidebar-delete-issue"> |  | ||||||
| 			{{svg "octicon-trash"}} |  | ||||||
| 			{{ctx.Locale.Tr "repo.issues.delete"}} |  | ||||||
| 		</button> |  | ||||||
| 		<div class="ui g-modal-confirm modal" id="sidebar-delete-issue"> |  | ||||||
| 			<div class="header"> |  | ||||||
| 				{{if .Issue.IsPull}} |  | ||||||
| 					{{ctx.Locale.Tr "repo.pulls.delete.title"}} |  | ||||||
| 				{{else}} |  | ||||||
| 					{{ctx.Locale.Tr "repo.issues.delete.title"}} |  | ||||||
| 				{{end}} |  | ||||||
| 			</div> |  | ||||||
| 			<div class="content"> |  | ||||||
| 				<p> |  | ||||||
| 					{{if .Issue.IsPull}} |  | ||||||
| 						{{ctx.Locale.Tr "repo.pulls.delete.text"}} |  | ||||||
| 					{{else}} |  | ||||||
| 						{{ctx.Locale.Tr "repo.issues.delete.text"}} |  | ||||||
| 					{{end}} |  | ||||||
| 				</p> |  | ||||||
| 			</div> |  | ||||||
| 			<form action="{{.Issue.Link}}/delete" method="post"> |  | ||||||
| 				{{.CsrfTokenHtml}} |  | ||||||
| 				{{template "base/modal_actions_confirm" .}} |  | ||||||
| 			</form> |  | ||||||
| 		</div> |  | ||||||
| 	{{end}} |  | ||||||
|  |  | ||||||
| 	{{if and .Issue.IsPull .IsIssuePoster (not .Issue.IsClosed) .Issue.PullRequest.HeadRepo}} |  | ||||||
| 		{{if and (not (eq .Issue.PullRequest.HeadRepo.FullName .Issue.PullRequest.BaseRepo.FullName)) .CanWriteToHeadRepo}} |  | ||||||
| 			<div class="divider"></div> |  | ||||||
| 			<div class="inline field"> |  | ||||||
| 				<div class="ui checkbox loading-icon-2px" id="allow-edits-from-maintainers" |  | ||||||
| 						data-url="{{.Issue.Link}}" |  | ||||||
| 						data-tooltip-content="{{ctx.Locale.Tr "repo.pulls.allow_edits_from_maintainers_desc"}}" |  | ||||||
| 						data-prompt-error="{{ctx.Locale.Tr "repo.pulls.allow_edits_from_maintainers_err"}}" |  | ||||||
| 					> |  | ||||||
| 					<label><strong>{{ctx.Locale.Tr "repo.pulls.allow_edits_from_maintainers"}}</strong></label> |  | ||||||
| 					<input type="checkbox" {{if .Issue.PullRequest.AllowMaintainerEdit}}checked{{end}}> |  | ||||||
| 				</div> |  | ||||||
| 			</div> |  | ||||||
| 		{{end}} |  | ||||||
| 	{{end}} |  | ||||||
| </div> | </div> | ||||||
|   | |||||||
| @@ -3,6 +3,12 @@ | |||||||
| 		{{template "base/alert" .}} | 		{{template "base/alert" .}} | ||||||
| 	</div> | 	</div> | ||||||
| {{end}} | {{end}} | ||||||
|  | <div class="tw-hidden" id="issue-page-info" | ||||||
|  | 	data-issue-index="{{$.Issue.Index}}" | ||||||
|  | 	data-issue-dependency-search-type="{{$.IssueDependencySearchType}}" | ||||||
|  | 	data-issue-repo-link="{{$.RepoLink}}" | ||||||
|  | 	data-issue-repo-id="{{$.Repository.ID}}" | ||||||
|  | ></div> | ||||||
| <div class="issue-title-header"> | <div class="issue-title-header"> | ||||||
| 	{{$canEditIssueTitle := and (or .HasIssuesOrPullsWritePermission .IsIssuePoster) (not .Repository.IsArchived)}} | 	{{$canEditIssueTitle := and (or .HasIssuesOrPullsWritePermission .IsIssuePoster) (not .Repository.IsArchived)}} | ||||||
| 	<div class="issue-title" id="issue-title-display"> | 	<div class="issue-title" id="issue-title-display"> | ||||||
|   | |||||||
| @@ -1,8 +1,4 @@ | |||||||
| {{template "base/head" .}} | {{template "base/head" .}} | ||||||
|  |  | ||||||
| <input type="hidden" id="repolink" value="{{$.RepoRelPath}}"> |  | ||||||
| <input type="hidden" id="issueIndex" value="{{.Issue.Index}}"> |  | ||||||
|  |  | ||||||
| <div role="main" aria-label="{{.Title}}" class="page-content repository view issue pull files diff"> | <div role="main" aria-label="{{.Title}}" class="page-content repository view issue pull files diff"> | ||||||
| 	{{template "repo/header" .}} | 	{{template "repo/header" .}} | ||||||
| 	<div class="ui container fluid padded"> | 	<div class="ui container fluid padded"> | ||||||
|   | |||||||
| @@ -4,17 +4,20 @@ | |||||||
| package integration | package integration | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"fmt" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"net/url" | 	"net/url" | ||||||
| 	"testing" | 	"testing" | ||||||
|  |  | ||||||
| 	"code.gitea.io/gitea/models" | 	"code.gitea.io/gitea/models" | ||||||
|  | 	auth_model "code.gitea.io/gitea/models/auth" | ||||||
| 	"code.gitea.io/gitea/models/db" | 	"code.gitea.io/gitea/models/db" | ||||||
| 	git_model "code.gitea.io/gitea/models/git" | 	git_model "code.gitea.io/gitea/models/git" | ||||||
| 	repo_model "code.gitea.io/gitea/models/repo" | 	repo_model "code.gitea.io/gitea/models/repo" | ||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
| 	user_model "code.gitea.io/gitea/models/user" | 	user_model "code.gitea.io/gitea/models/user" | ||||||
| 	"code.gitea.io/gitea/modules/git" | 	"code.gitea.io/gitea/modules/git" | ||||||
|  | 	api "code.gitea.io/gitea/modules/structs" | ||||||
| 	"code.gitea.io/gitea/services/release" | 	"code.gitea.io/gitea/services/release" | ||||||
| 	"code.gitea.io/gitea/tests" | 	"code.gitea.io/gitea/tests" | ||||||
|  |  | ||||||
| @@ -117,3 +120,47 @@ func TestCreateNewTagProtected(t *testing.T) { | |||||||
| 		assert.NoError(t, err) | 		assert.NoError(t, err) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func TestRepushTag(t *testing.T) { | ||||||
|  | 	onGiteaRun(t, func(t *testing.T, u *url.URL) { | ||||||
|  | 		repo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: 1}) | ||||||
|  | 		owner := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: repo.OwnerID}) | ||||||
|  | 		session := loginUser(t, owner.LowerName) | ||||||
|  | 		token := getTokenForLoggedInUser(t, session, auth_model.AccessTokenScopeWriteRepository) | ||||||
|  |  | ||||||
|  | 		httpContext := NewAPITestContext(t, owner.Name, repo.Name) | ||||||
|  |  | ||||||
|  | 		dstPath := t.TempDir() | ||||||
|  |  | ||||||
|  | 		u.Path = httpContext.GitPath() | ||||||
|  | 		u.User = url.UserPassword(owner.Name, userPassword) | ||||||
|  |  | ||||||
|  | 		doGitClone(dstPath, u)(t) | ||||||
|  |  | ||||||
|  | 		// create and push a tag | ||||||
|  | 		_, _, err := git.NewCommand(git.DefaultContext, "tag", "v2.0").RunStdString(&git.RunOpts{Dir: dstPath}) | ||||||
|  | 		assert.NoError(t, err) | ||||||
|  | 		_, _, err = git.NewCommand(git.DefaultContext, "push", "origin", "--tags", "v2.0").RunStdString(&git.RunOpts{Dir: dstPath}) | ||||||
|  | 		assert.NoError(t, err) | ||||||
|  | 		// create a release for the tag | ||||||
|  | 		createdRelease := createNewReleaseUsingAPI(t, token, owner, repo, "v2.0", "", "Release of v2.0", "desc") | ||||||
|  | 		assert.False(t, createdRelease.IsDraft) | ||||||
|  | 		// delete the tag | ||||||
|  | 		_, _, err = git.NewCommand(git.DefaultContext, "push", "origin", "--delete", "v2.0").RunStdString(&git.RunOpts{Dir: dstPath}) | ||||||
|  | 		assert.NoError(t, err) | ||||||
|  | 		// query the release by API and it should be a draft | ||||||
|  | 		req := NewRequest(t, "GET", fmt.Sprintf("/api/v1/repos/%s/%s/releases/tags/%s", owner.Name, repo.Name, "v2.0")) | ||||||
|  | 		resp := MakeRequest(t, req, http.StatusOK) | ||||||
|  | 		var respRelease *api.Release | ||||||
|  | 		DecodeJSON(t, resp, &respRelease) | ||||||
|  | 		assert.True(t, respRelease.IsDraft) | ||||||
|  | 		// re-push the tag | ||||||
|  | 		_, _, err = git.NewCommand(git.DefaultContext, "push", "origin", "--tags", "v2.0").RunStdString(&git.RunOpts{Dir: dstPath}) | ||||||
|  | 		assert.NoError(t, err) | ||||||
|  | 		// query the release by API and it should not be a draft | ||||||
|  | 		req = NewRequest(t, "GET", fmt.Sprintf("/api/v1/repos/%s/%s/releases/tags/%s", owner.Name, repo.Name, "v2.0")) | ||||||
|  | 		resp = MakeRequest(t, req, http.StatusOK) | ||||||
|  | 		DecodeJSON(t, resp, &respRelease) | ||||||
|  | 		assert.False(t, respRelease.IsDraft) | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|   | |||||||
| @@ -73,7 +73,7 @@ | |||||||
|   margin-left: 5px; |   margin-left: 5px; | ||||||
| } | } | ||||||
|  |  | ||||||
| .page-footer .ui.dropdown.language .menu { | .page-footer .ui.dropdown .menu.language-menu { | ||||||
|   max-height: min(500px, calc(100vh - 60px)); |   max-height: min(500px, calc(100vh - 60px)); | ||||||
|   overflow-y: auto; |   overflow-y: auto; | ||||||
|   margin-bottom: 10px; |   margin-bottom: 10px; | ||||||
|   | |||||||
| @@ -62,23 +62,6 @@ | |||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| .repository .issue-content-right .ui.list .dependency { |  | ||||||
|   padding: 0; |  | ||||||
|   white-space: nowrap; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .repository .issue-content-right .ui.list .title { |  | ||||||
|   overflow: hidden; |  | ||||||
|   text-overflow: ellipsis; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .repository .issue-content-right #deadlineForm input { |  | ||||||
|   width: 12.8rem; |  | ||||||
|   border-radius: var(--border-radius) 0 0 var(--border-radius); |  | ||||||
|   border-right: 0; |  | ||||||
|   white-space: nowrap; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .repository .issue-content-right .filter.menu { | .repository .issue-content-right .filter.menu { | ||||||
|   max-height: 500px; |   max-height: 500px; | ||||||
|   overflow-x: auto; |   overflow-x: auto; | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| import $ from 'jquery'; |  | ||||||
| import {GET} from '../modules/fetch.ts'; | import {GET} from '../modules/fetch.ts'; | ||||||
| import {showGlobalErrorMessage} from '../bootstrap.ts'; | import {showGlobalErrorMessage} from '../bootstrap.ts'; | ||||||
|  | import {fomanticQuery} from '../modules/fomantic/base.ts'; | ||||||
|  | import {queryElems} from '../utils/dom.ts'; | ||||||
|  |  | ||||||
| const {appUrl} = window.config; | const {appUrl} = window.config; | ||||||
|  |  | ||||||
| @@ -17,18 +18,18 @@ export function initHeadNavbarContentToggle() { | |||||||
| } | } | ||||||
|  |  | ||||||
| export function initFootLanguageMenu() { | export function initFootLanguageMenu() { | ||||||
|   async function linkLanguageAction() { |   document.querySelector('.ui.dropdown .menu.language-menu')?.addEventListener('click', async (e) => { | ||||||
|     const $this = $(this); |     const item = (e.target as HTMLElement).closest('.item'); | ||||||
|     await GET($this.data('url')); |     if (!item) return; | ||||||
|  |     e.preventDefault(); | ||||||
|  |     await GET(item.getAttribute('data-url')); | ||||||
|     window.location.reload(); |     window.location.reload(); | ||||||
|   } |   }); | ||||||
|  |  | ||||||
|   $('.language-menu a[lang]').on('click', linkLanguageAction); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| export function initGlobalDropdown() { | export function initGlobalDropdown() { | ||||||
|   // Semantic UI modules. |   // Semantic UI modules. | ||||||
|   const $uiDropdowns = $('.ui.dropdown'); |   const $uiDropdowns = fomanticQuery('.ui.dropdown'); | ||||||
|  |  | ||||||
|   // do not init "custom" dropdowns, "custom" dropdowns are managed by their own code. |   // do not init "custom" dropdowns, "custom" dropdowns are managed by their own code. | ||||||
|   $uiDropdowns.filter(':not(.custom)').dropdown(); |   $uiDropdowns.filter(':not(.custom)').dropdown(); | ||||||
| @@ -46,14 +47,14 @@ export function initGlobalDropdown() { | |||||||
|     }, |     }, | ||||||
|     onHide() { |     onHide() { | ||||||
|       this._tippy?.enable(); |       this._tippy?.enable(); | ||||||
|  |       // eslint-disable-next-line unicorn/no-this-assignment | ||||||
|  |       const elDropdown = this; | ||||||
|  |  | ||||||
|       // hide all tippy elements of items after a while. eg: use Enter to click "Copy Link" in the Issue Context Menu |       // hide all tippy elements of items after a while. eg: use Enter to click "Copy Link" in the Issue Context Menu | ||||||
|       setTimeout(() => { |       setTimeout(() => { | ||||||
|         const $dropdown = $(this); |         const $dropdown = fomanticQuery(elDropdown); | ||||||
|         if ($dropdown.dropdown('is hidden')) { |         if ($dropdown.dropdown('is hidden')) { | ||||||
|           $(this).find('.menu > .item').each((_, item) => { |           queryElems(elDropdown, '.menu > .item', (el) => el._tippy?.hide()); | ||||||
|             item._tippy?.hide(); |  | ||||||
|           }); |  | ||||||
|         } |         } | ||||||
|       }, 2000); |       }, 2000); | ||||||
|     }, |     }, | ||||||
| @@ -71,7 +72,7 @@ export function initGlobalDropdown() { | |||||||
| } | } | ||||||
|  |  | ||||||
| export function initGlobalTabularMenu() { | export function initGlobalTabularMenu() { | ||||||
|   $('.ui.menu.tabular:not(.custom) .item').tab({autoTabActivation: false}); |   fomanticQuery('.ui.menu.tabular:not(.custom) .item').tab({autoTabActivation: false}); | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
|   | |||||||
| @@ -1,14 +1,14 @@ | |||||||
| import $ from 'jquery'; |  | ||||||
| import {GET} from '../modules/fetch.ts'; | import {GET} from '../modules/fetch.ts'; | ||||||
| import {hideElem, loadElem, queryElemChildren, queryElems} from '../utils/dom.ts'; | import {hideElem, loadElem, queryElemChildren, queryElems} from '../utils/dom.ts'; | ||||||
| import {parseDom} from '../utils.ts'; | import {parseDom} from '../utils.ts'; | ||||||
|  | import {fomanticQuery} from '../modules/fomantic/base.ts'; | ||||||
|  |  | ||||||
| function getDefaultSvgBoundsIfUndefined(text, src) { | function getDefaultSvgBoundsIfUndefined(text, src) { | ||||||
|   const defaultSize = 300; |   const defaultSize = 300; | ||||||
|   const maxSize = 99999; |   const maxSize = 99999; | ||||||
|  |  | ||||||
|   const svgDoc = parseDom(text, 'image/svg+xml'); |   const svgDoc = parseDom(text, 'image/svg+xml'); | ||||||
|   const svg = svgDoc.documentElement; |   const svg = (svgDoc.documentElement as unknown) as SVGSVGElement; | ||||||
|   const width = svg?.width?.baseVal; |   const width = svg?.width?.baseVal; | ||||||
|   const height = svg?.height?.baseVal; |   const height = svg?.height?.baseVal; | ||||||
|   if (width === undefined || height === undefined) { |   if (width === undefined || height === undefined) { | ||||||
| @@ -68,12 +68,14 @@ function createContext(imageAfter, imageBefore) { | |||||||
| } | } | ||||||
|  |  | ||||||
| class ImageDiff { | class ImageDiff { | ||||||
|   async init(containerEl) { |   containerEl: HTMLElement; | ||||||
|  |   diffContainerWidth: number; | ||||||
|  |  | ||||||
|  |   async init(containerEl: HTMLElement) { | ||||||
|     this.containerEl = containerEl; |     this.containerEl = containerEl; | ||||||
|     containerEl.setAttribute('data-image-diff-loaded', 'true'); |     containerEl.setAttribute('data-image-diff-loaded', 'true'); | ||||||
|  |  | ||||||
|     // the only jQuery usage in this file |     fomanticQuery(containerEl).find('.ui.menu.tabular .item').tab({autoTabActivation: false}); | ||||||
|     $(containerEl).find('.ui.menu.tabular .item').tab({autoTabActivation: false}); |  | ||||||
|  |  | ||||||
|     // the container may be hidden by "viewed" checkbox, so use the parent's width for reference |     // the container may be hidden by "viewed" checkbox, so use the parent's width for reference | ||||||
|     this.diffContainerWidth = Math.max(containerEl.closest('.diff-file-box').clientWidth - 300, 100); |     this.diffContainerWidth = Math.max(containerEl.closest('.diff-file-box').clientWidth - 300, 100); | ||||||
| @@ -81,12 +83,12 @@ class ImageDiff { | |||||||
|     const imageInfos = [{ |     const imageInfos = [{ | ||||||
|       path: containerEl.getAttribute('data-path-after'), |       path: containerEl.getAttribute('data-path-after'), | ||||||
|       mime: containerEl.getAttribute('data-mime-after'), |       mime: containerEl.getAttribute('data-mime-after'), | ||||||
|       images: containerEl.querySelectorAll('img.image-after'), // matches 3 <img> |       images: containerEl.querySelectorAll<HTMLImageElement>('img.image-after'), // matches 3 <img> | ||||||
|       boundsInfo: containerEl.querySelector('.bounds-info-after'), |       boundsInfo: containerEl.querySelector('.bounds-info-after'), | ||||||
|     }, { |     }, { | ||||||
|       path: containerEl.getAttribute('data-path-before'), |       path: containerEl.getAttribute('data-path-before'), | ||||||
|       mime: containerEl.getAttribute('data-mime-before'), |       mime: containerEl.getAttribute('data-mime-before'), | ||||||
|       images: containerEl.querySelectorAll('img.image-before'), // matches 3 <img> |       images: containerEl.querySelectorAll<HTMLImageElement>('img.image-before'), // matches 3 <img> | ||||||
|       boundsInfo: containerEl.querySelector('.bounds-info-before'), |       boundsInfo: containerEl.querySelector('.bounds-info-before'), | ||||||
|     }]; |     }]; | ||||||
|  |  | ||||||
| @@ -102,8 +104,8 @@ class ImageDiff { | |||||||
|         const bounds = getDefaultSvgBoundsIfUndefined(text, info.path); |         const bounds = getDefaultSvgBoundsIfUndefined(text, info.path); | ||||||
|         if (bounds) { |         if (bounds) { | ||||||
|           for (const el of info.images) { |           for (const el of info.images) { | ||||||
|             el.setAttribute('width', bounds.width); |             el.setAttribute('width', String(bounds.width)); | ||||||
|             el.setAttribute('height', bounds.height); |             el.setAttribute('height', String(bounds.height)); | ||||||
|           } |           } | ||||||
|           hideElem(info.boundsInfo); |           hideElem(info.boundsInfo); | ||||||
|         } |         } | ||||||
| @@ -151,7 +153,7 @@ class ImageDiff { | |||||||
|       const boundsInfoBeforeHeight = this.containerEl.querySelector('.bounds-info-before .bounds-info-height'); |       const boundsInfoBeforeHeight = this.containerEl.querySelector('.bounds-info-before .bounds-info-height'); | ||||||
|       if (boundsInfoBeforeHeight) { |       if (boundsInfoBeforeHeight) { | ||||||
|         boundsInfoBeforeHeight.textContent = `${sizes.imageBefore.naturalHeight}px`; |         boundsInfoBeforeHeight.textContent = `${sizes.imageBefore.naturalHeight}px`; | ||||||
|         boundsInfoBeforeHeight.classList.add('red', heightChanged); |         boundsInfoBeforeHeight.classList.toggle('red', heightChanged); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -205,7 +207,7 @@ class ImageDiff { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     // extra height for inner "position: absolute" elements |     // extra height for inner "position: absolute" elements | ||||||
|     const swipe = this.containerEl.querySelector('.diff-swipe'); |     const swipe = this.containerEl.querySelector<HTMLElement>('.diff-swipe'); | ||||||
|     if (swipe) { |     if (swipe) { | ||||||
|       swipe.style.width = `${sizes.maxSize.width * factor + 2}px`; |       swipe.style.width = `${sizes.maxSize.width * factor + 2}px`; | ||||||
|       swipe.style.height = `${sizes.maxSize.height * factor + 30}px`; |       swipe.style.height = `${sizes.maxSize.height * factor + 30}px`; | ||||||
| @@ -225,7 +227,7 @@ class ImageDiff { | |||||||
|       const rect = swipeFrame.getBoundingClientRect(); |       const rect = swipeFrame.getBoundingClientRect(); | ||||||
|       const value = Math.max(0, Math.min(e.clientX - rect.left, width)); |       const value = Math.max(0, Math.min(e.clientX - rect.left, width)); | ||||||
|       swipeBar.style.left = `${value}px`; |       swipeBar.style.left = `${value}px`; | ||||||
|       this.containerEl.querySelector('.swipe-container').style.width = `${swipeFrame.clientWidth - value}px`; |       this.containerEl.querySelector<HTMLElement>('.swipe-container').style.width = `${swipeFrame.clientWidth - value}px`; | ||||||
|     }; |     }; | ||||||
|     const removeEventListeners = () => { |     const removeEventListeners = () => { | ||||||
|       document.removeEventListener('mousemove', onSwipeMouseMove); |       document.removeEventListener('mousemove', onSwipeMouseMove); | ||||||
| @@ -264,11 +266,11 @@ class ImageDiff { | |||||||
|       overlayFrame.style.height = `${sizes.maxSize.height * factor + 2}px`; |       overlayFrame.style.height = `${sizes.maxSize.height * factor + 2}px`; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const rangeInput = this.containerEl.querySelector('input[type="range"]'); |     const rangeInput = this.containerEl.querySelector<HTMLInputElement>('input[type="range"]'); | ||||||
|  |  | ||||||
|     function updateOpacity() { |     function updateOpacity() { | ||||||
|       if (sizes.imageAfter) { |       if (sizes.imageAfter) { | ||||||
|         sizes.imageAfter.parentNode.style.opacity = `${rangeInput.value / 100}`; |         sizes.imageAfter.parentNode.style.opacity = `${Number(rangeInput.value) / 100}`; | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -278,7 +280,7 @@ class ImageDiff { | |||||||
| } | } | ||||||
|  |  | ||||||
| export function initImageDiff() { | export function initImageDiff() { | ||||||
|   for (const el of queryElems('.image-diff:not([data-image-diff-loaded])')) { |   for (const el of queryElems<HTMLImageElement>(document, '.image-diff:not([data-image-diff-loaded])')) { | ||||||
|     (new ImageDiff()).init(el); // it is async, but we don't need to await for it |     (new ImageDiff()).init(el); // it is async, but we don't need to await for it | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -31,7 +31,7 @@ async function onDownloadArchive(e) { | |||||||
| } | } | ||||||
|  |  | ||||||
| export function initRepoArchiveLinks() { | export function initRepoArchiveLinks() { | ||||||
|   queryElems('a.archive-link[href]', (el) => el.addEventListener('click', onDownloadArchive)); |   queryElems(document, 'a.archive-link[href]', (el) => el.addEventListener('click', onDownloadArchive)); | ||||||
| } | } | ||||||
|  |  | ||||||
| export function initRepoActivityTopAuthorsChart() { | export function initRepoActivityTopAuthorsChart() { | ||||||
|   | |||||||
| @@ -45,17 +45,17 @@ export function initRepoEditor() { | |||||||
|   const dropzoneUpload = document.querySelector('.page-content.repository.editor.upload .dropzone'); |   const dropzoneUpload = document.querySelector('.page-content.repository.editor.upload .dropzone'); | ||||||
|   if (dropzoneUpload) initDropzone(dropzoneUpload); |   if (dropzoneUpload) initDropzone(dropzoneUpload); | ||||||
|  |  | ||||||
|   const editArea = document.querySelector('.page-content.repository.editor textarea#edit_area'); |   const editArea = document.querySelector<HTMLTextAreaElement>('.page-content.repository.editor textarea#edit_area'); | ||||||
|   if (!editArea) return; |   if (!editArea) return; | ||||||
|  |  | ||||||
|   for (const el of queryElems('.js-quick-pull-choice-option')) { |   for (const el of queryElems<HTMLInputElement>(document, '.js-quick-pull-choice-option')) { | ||||||
|     el.addEventListener('input', () => { |     el.addEventListener('input', () => { | ||||||
|       if (el.value === 'commit-to-new-branch') { |       if (el.value === 'commit-to-new-branch') { | ||||||
|         showElem('.quick-pull-branch-name'); |         showElem('.quick-pull-branch-name'); | ||||||
|         document.querySelector('.quick-pull-branch-name input').required = true; |         document.querySelector<HTMLInputElement>('.quick-pull-branch-name input').required = true; | ||||||
|       } else { |       } else { | ||||||
|         hideElem('.quick-pull-branch-name'); |         hideElem('.quick-pull-branch-name'); | ||||||
|         document.querySelector('.quick-pull-branch-name input').required = false; |         document.querySelector<HTMLInputElement>('.quick-pull-branch-name input').required = false; | ||||||
|       } |       } | ||||||
|       document.querySelector('#commit-button').textContent = el.getAttribute('data-button-text'); |       document.querySelector('#commit-button').textContent = el.getAttribute('data-button-text'); | ||||||
|     }); |     }); | ||||||
| @@ -71,13 +71,13 @@ export function initRepoEditor() { | |||||||
|     if (filenameInput.value) { |     if (filenameInput.value) { | ||||||
|       parts.push(filenameInput.value); |       parts.push(filenameInput.value); | ||||||
|     } |     } | ||||||
|     document.querySelector('#tree_path').value = parts.join('/'); |     document.querySelector<HTMLInputElement>('#tree_path').value = parts.join('/'); | ||||||
|   } |   } | ||||||
|   filenameInput.addEventListener('input', function () { |   filenameInput.addEventListener('input', function () { | ||||||
|     const parts = filenameInput.value.split('/'); |     const parts = filenameInput.value.split('/'); | ||||||
|     const links = Array.from(document.querySelectorAll('.breadcrumb span.section')); |     const links = Array.from(document.querySelectorAll('.breadcrumb span.section')); | ||||||
|     const dividers = Array.from(document.querySelectorAll('.breadcrumb .breadcrumb-divider')); |     const dividers = Array.from(document.querySelectorAll('.breadcrumb .breadcrumb-divider')); | ||||||
|     let warningDiv = document.querySelector('.ui.warning.message.flash-message.flash-warning.space-related'); |     let warningDiv = document.querySelector<HTMLDivElement>('.ui.warning.message.flash-message.flash-warning.space-related'); | ||||||
|     let containSpace = false; |     let containSpace = false; | ||||||
|     if (parts.length > 1) { |     if (parts.length > 1) { | ||||||
|       for (let i = 0; i < parts.length; ++i) { |       for (let i = 0; i < parts.length; ++i) { | ||||||
| @@ -110,14 +110,14 @@ export function initRepoEditor() { | |||||||
|           filenameInput.value = value; |           filenameInput.value = value; | ||||||
|         } |         } | ||||||
|         this.setSelectionRange(0, 0); |         this.setSelectionRange(0, 0); | ||||||
|         containSpace |= (trimValue !== value && trimValue !== ''); |         containSpace = containSpace || (trimValue !== value && trimValue !== ''); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     containSpace |= Array.from(links).some((link) => { |     containSpace = containSpace || Array.from(links).some((link) => { | ||||||
|       const value = link.querySelector('a').textContent; |       const value = link.querySelector('a').textContent; | ||||||
|       return value.trim() !== value; |       return value.trim() !== value; | ||||||
|     }); |     }); | ||||||
|     containSpace |= parts[parts.length - 1].trim() !== parts[parts.length - 1]; |     containSpace = containSpace || parts[parts.length - 1].trim() !== parts[parts.length - 1]; | ||||||
|     if (containSpace) { |     if (containSpace) { | ||||||
|       if (!warningDiv) { |       if (!warningDiv) { | ||||||
|         warningDiv = document.createElement('div'); |         warningDiv = document.createElement('div'); | ||||||
| @@ -135,8 +135,8 @@ export function initRepoEditor() { | |||||||
|     joinTreePath(); |     joinTreePath(); | ||||||
|   }); |   }); | ||||||
|   filenameInput.addEventListener('keydown', function (e) { |   filenameInput.addEventListener('keydown', function (e) { | ||||||
|     const sections = queryElems('.breadcrumb span.section'); |     const sections = queryElems(document, '.breadcrumb span.section'); | ||||||
|     const dividers = queryElems('.breadcrumb .breadcrumb-divider'); |     const dividers = queryElems(document, '.breadcrumb .breadcrumb-divider'); | ||||||
|     // Jump back to last directory once the filename is empty |     // Jump back to last directory once the filename is empty | ||||||
|     if (e.code === 'Backspace' && filenameInput.selectionStart === 0 && sections.length > 0) { |     if (e.code === 'Backspace' && filenameInput.selectionStart === 0 && sections.length > 0) { | ||||||
|       e.preventDefault(); |       e.preventDefault(); | ||||||
| @@ -159,7 +159,7 @@ export function initRepoEditor() { | |||||||
|  |  | ||||||
|     // Using events from https://github.com/codedance/jquery.AreYouSure#advanced-usage |     // Using events from https://github.com/codedance/jquery.AreYouSure#advanced-usage | ||||||
|     // to enable or disable the commit button |     // to enable or disable the commit button | ||||||
|     const commitButton = document.querySelector('#commit-button'); |     const commitButton = document.querySelector<HTMLButtonElement>('#commit-button'); | ||||||
|     const $editForm = $('.ui.edit.form'); |     const $editForm = $('.ui.edit.form'); | ||||||
|     const dirtyFileClass = 'dirty-file'; |     const dirtyFileClass = 'dirty-file'; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -3,8 +3,8 @@ import {svg} from '../svg.ts'; | |||||||
| import {showErrorToast} from '../modules/toast.ts'; | import {showErrorToast} from '../modules/toast.ts'; | ||||||
| import {GET, POST} from '../modules/fetch.ts'; | import {GET, POST} from '../modules/fetch.ts'; | ||||||
| import {showElem} from '../utils/dom.ts'; | import {showElem} from '../utils/dom.ts'; | ||||||
|  | import {parseIssuePageInfo} from '../utils.ts'; | ||||||
|  |  | ||||||
| const {appSubUrl} = window.config; |  | ||||||
| let i18nTextEdited; | let i18nTextEdited; | ||||||
| let i18nTextOptions; | let i18nTextOptions; | ||||||
| let i18nTextDeleteFromHistory; | let i18nTextDeleteFromHistory; | ||||||
| @@ -121,15 +121,14 @@ function showContentHistoryMenu(issueBaseUrl, $item, commentId) { | |||||||
| } | } | ||||||
|  |  | ||||||
| export async function initRepoIssueContentHistory() { | export async function initRepoIssueContentHistory() { | ||||||
|   const issueIndex = $('#issueIndex').val(); |   const issuePageInfo = parseIssuePageInfo(); | ||||||
|   if (!issueIndex) return; |   if (!issuePageInfo.issueNumber) return; | ||||||
|  |  | ||||||
|   const $itemIssue = $('.repository.issue .timeline-item.comment.first'); // issue(PR) main content |   const $itemIssue = $('.repository.issue .timeline-item.comment.first'); // issue(PR) main content | ||||||
|   const $comments = $('.repository.issue .comment-list .comment'); // includes: issue(PR) comments, review comments, code comments |   const $comments = $('.repository.issue .comment-list .comment'); // includes: issue(PR) comments, review comments, code comments | ||||||
|   if (!$itemIssue.length && !$comments.length) return; |   if (!$itemIssue.length && !$comments.length) return; | ||||||
|  |  | ||||||
|   const repoLink = $('#repolink').val(); |   const issueBaseUrl = `${issuePageInfo.repoLink}/issues/${issuePageInfo.issueNumber}`; | ||||||
|   const issueBaseUrl = `${appSubUrl}/${repoLink}/issues/${issueIndex}`; |  | ||||||
|  |  | ||||||
|   try { |   try { | ||||||
|     const response = await GET(`${issueBaseUrl}/content-history/overview`); |     const response = await GET(`${issueBaseUrl}/content-history/overview`); | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ import {createTippy, showTemporaryTooltip} from '../modules/tippy.ts'; | |||||||
| import {hideElem, showElem, toggleElem} from '../utils/dom.ts'; | import {hideElem, showElem, toggleElem} from '../utils/dom.ts'; | ||||||
| import {setFileFolding} from './file-fold.ts'; | import {setFileFolding} from './file-fold.ts'; | ||||||
| import {ComboMarkdownEditor, getComboMarkdownEditor, initComboMarkdownEditor} from './comp/ComboMarkdownEditor.ts'; | import {ComboMarkdownEditor, getComboMarkdownEditor, initComboMarkdownEditor} from './comp/ComboMarkdownEditor.ts'; | ||||||
| import {toAbsoluteUrl} from '../utils.ts'; | import {parseIssuePageInfo, toAbsoluteUrl} from '../utils.ts'; | ||||||
| import {GET, POST} from '../modules/fetch.ts'; | import {GET, POST} from '../modules/fetch.ts'; | ||||||
| import {showErrorToast} from '../modules/toast.ts'; | import {showErrorToast} from '../modules/toast.ts'; | ||||||
| import {initRepoIssueSidebar} from './repo-issue-sidebar.ts'; | import {initRepoIssueSidebar} from './repo-issue-sidebar.ts'; | ||||||
| @@ -57,13 +57,11 @@ function excludeLabel(item) { | |||||||
| } | } | ||||||
|  |  | ||||||
| export function initRepoIssueSidebarList() { | export function initRepoIssueSidebarList() { | ||||||
|   const repolink = $('#repolink').val(); |   const issuePageInfo = parseIssuePageInfo(); | ||||||
|   const repoId = $('#repoId').val(); |  | ||||||
|   const crossRepoSearch = $('#crossRepoSearch').val(); |   const crossRepoSearch = $('#crossRepoSearch').val(); | ||||||
|   const tp = $('#type').val(); |   let issueSearchUrl = `${issuePageInfo.repoLink}/issues/search?q={query}&type=${issuePageInfo.issueDependencySearchType}`; | ||||||
|   let issueSearchUrl = `${appSubUrl}/${repolink}/issues/search?q={query}&type=${tp}`; |  | ||||||
|   if (crossRepoSearch === 'true') { |   if (crossRepoSearch === 'true') { | ||||||
|     issueSearchUrl = `${appSubUrl}/issues/search?q={query}&priority_repo_id=${repoId}&type=${tp}`; |     issueSearchUrl = `${appSubUrl}/issues/search?q={query}&priority_repo_id=${issuePageInfo.repoId}&type=${issuePageInfo.issueDependencySearchType}`; | ||||||
|   } |   } | ||||||
|   $('#new-dependency-drop-list') |   $('#new-dependency-drop-list') | ||||||
|     .dropdown({ |     .dropdown({ | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ const {appSubUrl, csrfToken} = window.config; | |||||||
|  |  | ||||||
| function initRepoSettingsCollaboration() { | function initRepoSettingsCollaboration() { | ||||||
|   // Change collaborator access mode |   // Change collaborator access mode | ||||||
|   for (const dropdownEl of queryElems('.page-content.repository .ui.dropdown.access-mode')) { |   for (const dropdownEl of queryElems(document, '.page-content.repository .ui.dropdown.access-mode')) { | ||||||
|     const textEl = dropdownEl.querySelector(':scope > .text'); |     const textEl = dropdownEl.querySelector(':scope > .text'); | ||||||
|     $(dropdownEl).dropdown({ |     $(dropdownEl).dropdown({ | ||||||
|       async action(text, value) { |       async action(text, value) { | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | import $ from 'jquery'; | ||||||
| let ariaIdCounter = 0; | let ariaIdCounter = 0; | ||||||
|  |  | ||||||
| export function generateAriaId() { | export function generateAriaId() { | ||||||
| @@ -16,3 +17,6 @@ export function linkLabelAndInput(label, input) { | |||||||
|     label.setAttribute('for', id); |     label.setAttribute('for', id); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // eslint-disable-next-line no-jquery/variable-pattern | ||||||
|  | export const fomanticQuery = $; | ||||||
|   | |||||||
| @@ -179,11 +179,9 @@ export function initGlobalTooltips() { | |||||||
| } | } | ||||||
|  |  | ||||||
| export function showTemporaryTooltip(target: Element, content: Content) { | export function showTemporaryTooltip(target: Element, content: Content) { | ||||||
|   // if the target is inside a dropdown, don't show the tooltip because when the dropdown |   // if the target is inside a dropdown, the menu will be hidden soon | ||||||
|   // closes, the tippy would be pushed unsightly to the top-left of the screen like seen |   // so display the tooltip on the dropdown instead | ||||||
|   // on the issue comment menu. |   target = target.closest('.ui.dropdown') || target; | ||||||
|   if (target.closest('.ui.dropdown > .menu')) return; |  | ||||||
|  |  | ||||||
|   const tippy = target._tippy ?? attachTooltip(target, content); |   const tippy = target._tippy ?? attachTooltip(target, content); | ||||||
|   tippy.setContent(content); |   tippy.setContent(content); | ||||||
|   if (!tippy.state.isShown) tippy.show(); |   if (!tippy.state.isShown) tippy.show(); | ||||||
|   | |||||||
| @@ -37,6 +37,13 @@ export type IssuePathInfo = { | |||||||
|   indexString?: string, |   indexString?: string, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | export type IssuePageInfo = { | ||||||
|  |   repoLink: string, | ||||||
|  |   repoId: number, | ||||||
|  |   issueNumber: number, | ||||||
|  |   issueDependencySearchType: string, | ||||||
|  | } | ||||||
|  |  | ||||||
| export type Issue = { | export type Issue = { | ||||||
|   id: number; |   id: number; | ||||||
|   number: number; |   number: number; | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| import {encode, decode} from 'uint8-to-base64'; | import {decode, encode} from 'uint8-to-base64'; | ||||||
| import type {IssuePathInfo} from './types.ts'; | import type {IssuePageInfo, IssuePathInfo} from './types.ts'; | ||||||
|  |  | ||||||
| // transform /path/to/file.ext to file.ext | // transform /path/to/file.ext to file.ext | ||||||
| export function basename(path: string): string { | export function basename(path: string): string { | ||||||
| @@ -43,6 +43,16 @@ export function parseIssueNewHref(href: string): IssuePathInfo { | |||||||
|   return {ownerName, repoName, pathType, indexString}; |   return {ownerName, repoName, pathType, indexString}; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | export function parseIssuePageInfo(): IssuePageInfo { | ||||||
|  |   const el = document.querySelector('#issue-page-info'); | ||||||
|  |   return { | ||||||
|  |     issueNumber: parseInt(el?.getAttribute('data-issue-index')), | ||||||
|  |     issueDependencySearchType: el?.getAttribute('data-issue-dependency-search-type') || '', | ||||||
|  |     repoId: parseInt(el?.getAttribute('data-issue-repo-id')), | ||||||
|  |     repoLink: el?.getAttribute('data-issue-repo-link') || '', | ||||||
|  |   }; | ||||||
|  | } | ||||||
|  |  | ||||||
| // parse a URL, either relative '/path' or absolute 'https://localhost/path' | // parse a URL, either relative '/path' or absolute 'https://localhost/path' | ||||||
| export function parseUrl(str: string): URL { | export function parseUrl(str: string): URL { | ||||||
|   return new URL(str, str.startsWith('http') ? undefined : window.location.origin); |   return new URL(str, str.startsWith('http') ? undefined : window.location.origin); | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ import type $ from 'jquery'; | |||||||
| type ElementArg = Element | string | NodeListOf<Element> | Array<Element> | ReturnType<typeof $>; | type ElementArg = Element | string | NodeListOf<Element> | Array<Element> | ReturnType<typeof $>; | ||||||
| type ElementsCallback = (el: Element) => Promisable<any>; | type ElementsCallback = (el: Element) => Promisable<any>; | ||||||
| type ElementsCallbackWithArgs = (el: Element, ...args: any[]) => Promisable<any>; | type ElementsCallbackWithArgs = (el: Element, ...args: any[]) => Promisable<any>; | ||||||
| type IterableElements = NodeListOf<Element> | Array<Element>; | type ArrayLikeIterable<T> = ArrayLike<T> & Iterable<T>; // for NodeListOf and Array | ||||||
|  |  | ||||||
| function elementsCall(el: ElementArg, func: ElementsCallbackWithArgs, ...args: any[]) { | function elementsCall(el: ElementArg, func: ElementsCallbackWithArgs, ...args: any[]) { | ||||||
|   if (typeof el === 'string' || el instanceof String) { |   if (typeof el === 'string' || el instanceof String) { | ||||||
| @@ -15,7 +15,7 @@ function elementsCall(el: ElementArg, func: ElementsCallbackWithArgs, ...args: a | |||||||
|     func(el, ...args); |     func(el, ...args); | ||||||
|   } else if (el.length !== undefined) { |   } else if (el.length !== undefined) { | ||||||
|     // this works for: NodeList, HTMLCollection, Array, jQuery |     // this works for: NodeList, HTMLCollection, Array, jQuery | ||||||
|     for (const e of (el as IterableElements)) { |     for (const e of (el as ArrayLikeIterable<Element>)) { | ||||||
|       func(e, ...args); |       func(e, ...args); | ||||||
|     } |     } | ||||||
|   } else { |   } else { | ||||||
| @@ -58,7 +58,7 @@ export function isElemHidden(el: ElementArg) { | |||||||
|   return res[0]; |   return res[0]; | ||||||
| } | } | ||||||
|  |  | ||||||
| function applyElemsCallback(elems: IterableElements, fn?: ElementsCallback) { | function applyElemsCallback<T extends Element>(elems: ArrayLikeIterable<T>, fn?: ElementsCallback): ArrayLikeIterable<T> { | ||||||
|   if (fn) { |   if (fn) { | ||||||
|     for (const el of elems) { |     for (const el of elems) { | ||||||
|       fn(el); |       fn(el); | ||||||
| @@ -67,19 +67,22 @@ function applyElemsCallback(elems: IterableElements, fn?: ElementsCallback) { | |||||||
|   return elems; |   return elems; | ||||||
| } | } | ||||||
|  |  | ||||||
| export function queryElemSiblings(el: Element, selector = '*', fn?: ElementsCallback) { | export function queryElemSiblings<T extends Element>(el: Element, selector = '*', fn?: ElementsCallback): ArrayLikeIterable<T> { | ||||||
|   return applyElemsCallback(Array.from(el.parentNode.children).filter((child: Element) => { |   const elems = Array.from(el.parentNode.children) as T[]; | ||||||
|  |   return applyElemsCallback<T>(elems.filter((child: Element) => { | ||||||
|     return child !== el && child.matches(selector); |     return child !== el && child.matches(selector); | ||||||
|   }), fn); |   }), fn); | ||||||
| } | } | ||||||
|  |  | ||||||
| // it works like jQuery.children: only the direct children are selected | // it works like jQuery.children: only the direct children are selected | ||||||
| export function queryElemChildren(parent: Element | ParentNode, selector = '*', fn?: ElementsCallback) { | export function queryElemChildren<T extends Element>(parent: Element | ParentNode, selector = '*', fn?: ElementsCallback): ArrayLikeIterable<T> { | ||||||
|   return applyElemsCallback(parent.querySelectorAll(`:scope > ${selector}`), fn); |   return applyElemsCallback<T>(parent.querySelectorAll(`:scope > ${selector}`), fn); | ||||||
| } | } | ||||||
|  |  | ||||||
| export function queryElems(selector: string, fn?: ElementsCallback) { | // it works like parent.querySelectorAll: all descendants are selected | ||||||
|   return applyElemsCallback(document.querySelectorAll(selector), fn); | // in the future, all "queryElems(document, ...)" should be refactored to use a more specific parent | ||||||
|  | export function queryElems<T extends Element>(parent: Element | ParentNode, selector: string, fn?: ElementsCallback): ArrayLikeIterable<T> { | ||||||
|  |   return applyElemsCallback<T>(parent.querySelectorAll(selector), fn); | ||||||
| } | } | ||||||
|  |  | ||||||
| export function onDomReady(cb: () => Promisable<void>) { | export function onDomReady(cb: () => Promisable<void>) { | ||||||
| @@ -92,7 +95,7 @@ export function onDomReady(cb: () => Promisable<void>) { | |||||||
|  |  | ||||||
| // checks whether an element is owned by the current document, and whether it is a document fragment or element node | // checks whether an element is owned by the current document, and whether it is a document fragment or element node | ||||||
| // if it is, it means it is a "normal" element managed by us, which can be modified safely. | // if it is, it means it is a "normal" element managed by us, which can be modified safely. | ||||||
| export function isDocumentFragmentOrElementNode(el: Element | Node) { | export function isDocumentFragmentOrElementNode(el: Node) { | ||||||
|   try { |   try { | ||||||
|     return el.ownerDocument === document && el.nodeType === Node.ELEMENT_NODE || el.nodeType === Node.DOCUMENT_FRAGMENT_NODE; |     return el.ownerDocument === document && el.nodeType === Node.ELEMENT_NODE || el.nodeType === Node.DOCUMENT_FRAGMENT_NODE; | ||||||
|   } catch { |   } catch { | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user