mirror of
https://github.com/go-gitea/gitea.git
synced 2025-07-19 00:01:16 -04:00
Compare commits
3 Commits
71d2a6a41a
...
9bbb4d8d6d
Author | SHA1 | Date | |
---|---|---|---|
|
9bbb4d8d6d | ||
|
5703a0d3e3 | ||
|
85bad22ff8 |
@ -108,6 +108,14 @@ Admin operations:
|
|||||||
- `--all`, `-A`: Force a password change for all users
|
- `--all`, `-A`: Force a password change for all users
|
||||||
- `--exclude username`, `-e username`: Exclude the given user. Can be set multiple times.
|
- `--exclude username`, `-e username`: Exclude the given user. Can be set multiple times.
|
||||||
- `--unset`: Revoke forced password change for the given users
|
- `--unset`: Revoke forced password change for the given users
|
||||||
|
- `generate-access-token`:
|
||||||
|
- Options:
|
||||||
|
- `--username value`, `-u value`: Username. Required.
|
||||||
|
- `--token-name value`, `-t value`: Token name. Required.
|
||||||
|
- `--scopes value`: Comma-separated list of scopes. Scopes follow the format `[read|write]:<block>` or `all` where `<block>` is one of the available visual groups you can see when opening the API page showing the available routes (for example `repo`).
|
||||||
|
- Examples:
|
||||||
|
- `gitea admin user generate-access-token --username myname --token-name mytoken`
|
||||||
|
- `gitea admin user generate-access-token --help`
|
||||||
- `regenerate`
|
- `regenerate`
|
||||||
- Options:
|
- Options:
|
||||||
- `hooks`: Regenerate Git Hooks for all repositories
|
- `hooks`: Regenerate Git Hooks for all repositories
|
||||||
|
@ -229,39 +229,41 @@ func (issues IssueList) loadMilestones(ctx context.Context) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (issues IssueList) getProjectIDs() []int64 {
|
func (issues IssueList) LoadProjects(ctx context.Context) error {
|
||||||
ids := make(container.Set[int64], len(issues))
|
issueIDs := issues.getIssueIDs()
|
||||||
for _, issue := range issues {
|
projectMaps := make(map[int64]*project_model.Project, len(issues))
|
||||||
ids.Add(issue.ProjectID())
|
left := len(issueIDs)
|
||||||
}
|
|
||||||
return ids.Values()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (issues IssueList) loadProjects(ctx context.Context) error {
|
type projectWithIssueID struct {
|
||||||
projectIDs := issues.getProjectIDs()
|
*project_model.Project `xorm:"extends"`
|
||||||
if len(projectIDs) == 0 {
|
IssueID int64
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
projectMaps := make(map[int64]*project_model.Project, len(projectIDs))
|
|
||||||
left := len(projectIDs)
|
|
||||||
for left > 0 {
|
for left > 0 {
|
||||||
limit := db.DefaultMaxInSize
|
limit := db.DefaultMaxInSize
|
||||||
if left < limit {
|
if left < limit {
|
||||||
limit = left
|
limit = left
|
||||||
}
|
}
|
||||||
|
|
||||||
|
projects := make([]*projectWithIssueID, 0, limit)
|
||||||
err := db.GetEngine(ctx).
|
err := db.GetEngine(ctx).
|
||||||
In("id", projectIDs[:limit]).
|
Table("project").
|
||||||
Find(&projectMaps)
|
Select("project.*, project_issue.issue_id").
|
||||||
|
Join("INNER", "project_issue", "project.id = project_issue.project_id").
|
||||||
|
In("project_issue.issue_id", issueIDs[:limit]).
|
||||||
|
Find(&projects)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
for _, project := range projects {
|
||||||
|
projectMaps[project.IssueID] = project.Project
|
||||||
|
}
|
||||||
left -= limit
|
left -= limit
|
||||||
projectIDs = projectIDs[limit:]
|
issueIDs = issueIDs[limit:]
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, issue := range issues {
|
for _, issue := range issues {
|
||||||
issue.Project = projectMaps[issue.ProjectID()]
|
issue.Project = projectMaps[issue.ID]
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -541,7 +543,7 @@ func (issues IssueList) loadAttributes(ctx context.Context) error {
|
|||||||
return fmt.Errorf("issue.loadAttributes: loadMilestones: %w", err)
|
return fmt.Errorf("issue.loadAttributes: loadMilestones: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := issues.loadProjects(ctx); err != nil {
|
if err := issues.LoadProjects(ctx); err != nil {
|
||||||
return fmt.Errorf("issue.loadAttributes: loadProjects: %w", err)
|
return fmt.Errorf("issue.loadAttributes: loadProjects: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -66,8 +66,12 @@ func TestIssueList_LoadAttributes(t *testing.T) {
|
|||||||
}
|
}
|
||||||
if issue.ID == int64(1) {
|
if issue.ID == int64(1) {
|
||||||
assert.Equal(t, int64(400), issue.TotalTrackedTime)
|
assert.Equal(t, int64(400), issue.TotalTrackedTime)
|
||||||
|
assert.NotNil(t, issue.Project)
|
||||||
} else if issue.ID == int64(2) {
|
} else if issue.ID == int64(2) {
|
||||||
assert.Equal(t, int64(3682), issue.TotalTrackedTime)
|
assert.Equal(t, int64(3682), issue.TotalTrackedTime)
|
||||||
|
assert.Nil(t, issue.Project)
|
||||||
|
} else {
|
||||||
|
assert.Nil(t, issue.Project)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -27,11 +27,6 @@ func (issue *Issue) LoadProject(ctx context.Context) (err error) {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// ProjectID return project id if issue was assigned to one
|
|
||||||
func (issue *Issue) ProjectID() int64 {
|
|
||||||
return issue.projectID(db.DefaultContext)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (issue *Issue) projectID(ctx context.Context) int64 {
|
func (issue *Issue) projectID(ctx context.Context) int64 {
|
||||||
var ip project_model.ProjectIssue
|
var ip project_model.ProjectIssue
|
||||||
has, err := db.GetEngine(ctx).Where("issue_id=?", issue.ID).Get(&ip)
|
has, err := db.GetEngine(ctx).Where("issue_id=?", issue.ID).Get(&ip)
|
||||||
|
@ -383,7 +383,7 @@ func ViewProject(ctx *context.Context) {
|
|||||||
ctx.HTML(http.StatusOK, tplProjectsView)
|
ctx.HTML(http.StatusOK, tplProjectsView)
|
||||||
}
|
}
|
||||||
|
|
||||||
func getActionIssues(ctx *context.Context) []*issues_model.Issue {
|
func getActionIssues(ctx *context.Context) issues_model.IssueList {
|
||||||
commaSeparatedIssueIDs := ctx.FormString("issue_ids")
|
commaSeparatedIssueIDs := ctx.FormString("issue_ids")
|
||||||
if len(commaSeparatedIssueIDs) == 0 {
|
if len(commaSeparatedIssueIDs) == 0 {
|
||||||
return nil
|
return nil
|
||||||
@ -429,9 +429,14 @@ func UpdateIssueProject(ctx *context.Context) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if err := issues.LoadProjects(ctx); err != nil {
|
||||||
|
ctx.ServerError("LoadProjects", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
projectID := ctx.FormInt64("id")
|
projectID := ctx.FormInt64("id")
|
||||||
for _, issue := range issues {
|
for _, issue := range issues {
|
||||||
oldProjectID := issue.ProjectID()
|
oldProjectID := issue.Project.ID
|
||||||
if oldProjectID == projectID {
|
if oldProjectID == projectID {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
@ -1971,7 +1971,7 @@ func checkIssueRights(ctx *context.Context, issue *issues_model.Issue) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func getActionIssues(ctx *context.Context) []*issues_model.Issue {
|
func getActionIssues(ctx *context.Context) issues_model.IssueList {
|
||||||
commaSeparatedIssueIDs := ctx.FormString("issue_ids")
|
commaSeparatedIssueIDs := ctx.FormString("issue_ids")
|
||||||
if len(commaSeparatedIssueIDs) == 0 {
|
if len(commaSeparatedIssueIDs) == 0 {
|
||||||
return nil
|
return nil
|
||||||
@ -2722,7 +2722,7 @@ func UpdateIssueStatus(ctx *context.Context) {
|
|||||||
log.Warn("Unrecognized action: %s", action)
|
log.Warn("Unrecognized action: %s", action)
|
||||||
}
|
}
|
||||||
|
|
||||||
if _, err := issues_model.IssueList(issues).LoadRepositories(ctx); err != nil {
|
if _, err := issues.LoadRepositories(ctx); err != nil {
|
||||||
ctx.ServerError("LoadRepositories", err)
|
ctx.ServerError("LoadRepositories", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -378,9 +378,14 @@ func UpdateIssueProject(ctx *context.Context) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if err := issues.LoadProjects(ctx); err != nil {
|
||||||
|
ctx.ServerError("LoadProjects", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
projectID := ctx.FormInt64("id")
|
projectID := ctx.FormInt64("id")
|
||||||
for _, issue := range issues {
|
for _, issue := range issues {
|
||||||
oldProjectID := issue.ProjectID()
|
oldProjectID := issue.Project.ID
|
||||||
if oldProjectID == projectID {
|
if oldProjectID == projectID {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
@ -154,13 +154,6 @@
|
|||||||
<div class="item">item</div>
|
<div class="item">item</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="ui button dropdown">
|
|
||||||
<span class="text">button dropdown</span>
|
|
||||||
{{svg "octicon-triangle-down" 14 "dropdown icon"}}
|
|
||||||
<div class="menu">
|
|
||||||
<div class="item">item</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="ui search selection dropdown">
|
<div class="ui search selection dropdown">
|
||||||
<span class="text">search ...</span>
|
<span class="text">search ...</span>
|
||||||
<input name="value" class="search">
|
<input name="value" class="search">
|
||||||
@ -201,18 +194,39 @@
|
|||||||
|
|
||||||
<div>
|
<div>
|
||||||
<div class="ui dropdown mini button">
|
<div class="ui dropdown mini button">
|
||||||
<span class="text">small dropdown</span>
|
<span class="text">mini dropdown</span>
|
||||||
{{svg "octicon-triangle-down" 12 "dropdown icon"}}
|
{{svg "octicon-triangle-down" 14 "dropdown icon"}}
|
||||||
<div class="menu">
|
</div>
|
||||||
<div class="item">item</div>
|
<div class="ui dropdown tiny button">
|
||||||
</div>
|
<span class="text">tiny dropdown</span>
|
||||||
|
{{svg "octicon-triangle-down" 14 "dropdown icon"}}
|
||||||
|
</div>
|
||||||
|
<div class="ui button dropdown">
|
||||||
|
<span class="text">button dropdown</span>
|
||||||
|
{{svg "octicon-triangle-down" 14 "dropdown icon"}}
|
||||||
</div>
|
</div>
|
||||||
<div class="ui dropdown large button">
|
<div class="ui dropdown large button">
|
||||||
<span class="text">large dropdown</span>
|
<span class="text">large dropdown</span>
|
||||||
{{svg "octicon-triangle-down" 18 "dropdown icon"}}
|
{{svg "octicon-triangle-down" 14 "dropdown icon"}}
|
||||||
<div class="menu">
|
</div>
|
||||||
<div class="item">item</div>
|
</div>
|
||||||
</div>
|
|
||||||
|
<div>
|
||||||
|
<div class="ui dropdown mini compact button">
|
||||||
|
<span class="text">mini compact</span>
|
||||||
|
{{svg "octicon-triangle-down" 14 "dropdown icon"}}
|
||||||
|
</div>
|
||||||
|
<div class="ui dropdown tiny compact button">
|
||||||
|
<span class="text">tiny compact</span>
|
||||||
|
{{svg "octicon-triangle-down" 14 "dropdown icon"}}
|
||||||
|
</div>
|
||||||
|
<div class="ui button compact dropdown">
|
||||||
|
<span class="text">button compact</span>
|
||||||
|
{{svg "octicon-triangle-down" 14 "dropdown icon"}}
|
||||||
|
</div>
|
||||||
|
<div class="ui dropdown large compact button">
|
||||||
|
<span class="text">large compact</span>
|
||||||
|
{{svg "octicon-triangle-down" 14 "dropdown icon"}}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -26,7 +26,8 @@
|
|||||||
{{.locale.Tr "repo.diff.browse_source"}}
|
{{.locale.Tr "repo.diff.browse_source"}}
|
||||||
</a>
|
</a>
|
||||||
{{if and ($.Permission.CanWrite $.UnitTypeCode) (not $.Repository.IsArchived) (not .IsDeleted)}}{{- /* */ -}}
|
{{if and ($.Permission.CanWrite $.UnitTypeCode) (not $.Repository.IsArchived) (not .IsDeleted)}}{{- /* */ -}}
|
||||||
<div class="ui primary tiny floating dropdown icon button">{{.locale.Tr "repo.commit.operations"}}
|
<div class="ui dropdown primary tiny button">
|
||||||
|
{{.locale.Tr "repo.commit.operations"}}
|
||||||
{{svg "octicon-triangle-down" 14 "dropdown icon"}}
|
{{svg "octicon-triangle-down" 14 "dropdown icon"}}
|
||||||
<div class="menu">
|
<div class="menu">
|
||||||
<div class="ui header">{{.locale.Tr "repo.commit.operations"}}</div>
|
<div class="ui header">{{.locale.Tr "repo.commit.operations"}}</div>
|
||||||
|
@ -192,9 +192,9 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="ui select-project list">
|
<div class="ui select-project list">
|
||||||
<span class="no-select item {{if .Issue.ProjectID}}gt-hidden{{end}}">{{.locale.Tr "repo.issues.new.no_projects"}}</span>
|
<span class="no-select item {{if .Issue.Project}}gt-hidden{{end}}">{{.locale.Tr "repo.issues.new.no_projects"}}</span>
|
||||||
<div class="selected">
|
<div class="selected">
|
||||||
{{if .Issue.ProjectID}}
|
{{if .Issue.Project}}
|
||||||
<a class="item muted sidebar-item-link" href="{{.Issue.Project.Link}}">
|
<a class="item muted sidebar-item-link" href="{{.Issue.Project.Link}}">
|
||||||
{{svg .Issue.Project.IconName 18 "gt-mr-3"}}{{.Issue.Project.Title}}
|
{{svg .Issue.Project.IconName 18 "gt-mr-3"}}{{.Issue.Project.Title}}
|
||||||
</a>
|
</a>
|
||||||
|
@ -2289,6 +2289,17 @@ table th[data-sortt-desc] .svg {
|
|||||||
- "> .ui.label > .delete.icon": the "x" icon for removing a label item in multiple selection dropdown
|
- "> .ui.label > .delete.icon": the "x" icon for removing a label item in multiple selection dropdown
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
.ui.dropdown.mini.button,
|
||||||
|
.ui.dropdown.tiny.button {
|
||||||
|
padding-right: 20px;
|
||||||
|
}
|
||||||
|
.ui.dropdown.button {
|
||||||
|
padding-right: 22px;
|
||||||
|
}
|
||||||
|
.ui.dropdown.large.button {
|
||||||
|
padding-right: 24px;
|
||||||
|
}
|
||||||
|
|
||||||
/* Gitea uses SVG images instead of Fomantic builtin "<i>" font icons, so we need to reset the icon styles */
|
/* Gitea uses SVG images instead of Fomantic builtin "<i>" font icons, so we need to reset the icon styles */
|
||||||
.ui.ui.dropdown > .icon.icon {
|
.ui.ui.dropdown > .icon.icon {
|
||||||
position: initial; /* plain dropdown and button dropdown use flex layout for icons */
|
position: initial; /* plain dropdown and button dropdown use flex layout for icons */
|
||||||
|
Loading…
x
Reference in New Issue
Block a user