mirror of https://github.com/go-gitea/gitea
Support org/user level projects (#22235)
Fix #13405 <img width="1151" alt="image" src="https://user-images.githubusercontent.com/81045/209442911-7baa3924-c389-47b6-b63b-a740803e640e.png"> Co-authored-by: 6543 <6543@obermui.de>pull/22551/head
parent
0c048e554b
commit
6fe3c8b398
@ -0,0 +1,128 @@ |
|||||||
|
// Copyright 2022 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package organization |
||||||
|
|
||||||
|
import ( |
||||||
|
"context" |
||||||
|
"strings" |
||||||
|
|
||||||
|
"code.gitea.io/gitea/models/db" |
||||||
|
"code.gitea.io/gitea/models/perm" |
||||||
|
repo_model "code.gitea.io/gitea/models/repo" |
||||||
|
"code.gitea.io/gitea/models/unit" |
||||||
|
|
||||||
|
"xorm.io/builder" |
||||||
|
) |
||||||
|
|
||||||
|
type TeamList []*Team |
||||||
|
|
||||||
|
func (t TeamList) LoadUnits(ctx context.Context) error { |
||||||
|
for _, team := range t { |
||||||
|
if err := team.getUnits(ctx); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
} |
||||||
|
return nil |
||||||
|
} |
||||||
|
|
||||||
|
func (t TeamList) UnitMaxAccess(tp unit.Type) perm.AccessMode { |
||||||
|
maxAccess := perm.AccessModeNone |
||||||
|
for _, team := range t { |
||||||
|
if team.IsOwnerTeam() { |
||||||
|
return perm.AccessModeOwner |
||||||
|
} |
||||||
|
for _, teamUnit := range team.Units { |
||||||
|
if teamUnit.Type != tp { |
||||||
|
continue |
||||||
|
} |
||||||
|
if teamUnit.AccessMode > maxAccess { |
||||||
|
maxAccess = teamUnit.AccessMode |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
return maxAccess |
||||||
|
} |
||||||
|
|
||||||
|
// SearchTeamOptions holds the search options
|
||||||
|
type SearchTeamOptions struct { |
||||||
|
db.ListOptions |
||||||
|
UserID int64 |
||||||
|
Keyword string |
||||||
|
OrgID int64 |
||||||
|
IncludeDesc bool |
||||||
|
} |
||||||
|
|
||||||
|
func (opts *SearchTeamOptions) toCond() builder.Cond { |
||||||
|
cond := builder.NewCond() |
||||||
|
|
||||||
|
if len(opts.Keyword) > 0 { |
||||||
|
lowerKeyword := strings.ToLower(opts.Keyword) |
||||||
|
var keywordCond builder.Cond = builder.Like{"lower_name", lowerKeyword} |
||||||
|
if opts.IncludeDesc { |
||||||
|
keywordCond = keywordCond.Or(builder.Like{"LOWER(description)", lowerKeyword}) |
||||||
|
} |
||||||
|
cond = cond.And(keywordCond) |
||||||
|
} |
||||||
|
|
||||||
|
if opts.OrgID > 0 { |
||||||
|
cond = cond.And(builder.Eq{"`team`.org_id": opts.OrgID}) |
||||||
|
} |
||||||
|
|
||||||
|
if opts.UserID > 0 { |
||||||
|
cond = cond.And(builder.Eq{"team_user.uid": opts.UserID}) |
||||||
|
} |
||||||
|
|
||||||
|
return cond |
||||||
|
} |
||||||
|
|
||||||
|
// SearchTeam search for teams. Caller is responsible to check permissions.
|
||||||
|
func SearchTeam(opts *SearchTeamOptions) (TeamList, int64, error) { |
||||||
|
sess := db.GetEngine(db.DefaultContext) |
||||||
|
|
||||||
|
opts.SetDefaultValues() |
||||||
|
cond := opts.toCond() |
||||||
|
|
||||||
|
if opts.UserID > 0 { |
||||||
|
sess = sess.Join("INNER", "team_user", "team_user.team_id = team.id") |
||||||
|
} |
||||||
|
sess = db.SetSessionPagination(sess, opts) |
||||||
|
|
||||||
|
teams := make([]*Team, 0, opts.PageSize) |
||||||
|
count, err := sess.Where(cond).OrderBy("lower_name").FindAndCount(&teams) |
||||||
|
if err != nil { |
||||||
|
return nil, 0, err |
||||||
|
} |
||||||
|
|
||||||
|
return teams, count, nil |
||||||
|
} |
||||||
|
|
||||||
|
// GetRepoTeams gets the list of teams that has access to the repository
|
||||||
|
func GetRepoTeams(ctx context.Context, repo *repo_model.Repository) (teams TeamList, err error) { |
||||||
|
return teams, db.GetEngine(ctx). |
||||||
|
Join("INNER", "team_repo", "team_repo.team_id = team.id"). |
||||||
|
Where("team.org_id = ?", repo.OwnerID). |
||||||
|
And("team_repo.repo_id=?", repo.ID). |
||||||
|
OrderBy("CASE WHEN name LIKE '" + OwnerTeamName + "' THEN '' ELSE name END"). |
||||||
|
Find(&teams) |
||||||
|
} |
||||||
|
|
||||||
|
// GetUserOrgTeams returns all teams that user belongs to in given organization.
|
||||||
|
func GetUserOrgTeams(ctx context.Context, orgID, userID int64) (teams TeamList, err error) { |
||||||
|
return teams, db.GetEngine(ctx). |
||||||
|
Join("INNER", "team_user", "team_user.team_id = team.id"). |
||||||
|
Where("team.org_id = ?", orgID). |
||||||
|
And("team_user.uid=?", userID). |
||||||
|
Find(&teams) |
||||||
|
} |
||||||
|
|
||||||
|
// GetUserRepoTeams returns user repo's teams
|
||||||
|
func GetUserRepoTeams(ctx context.Context, orgID, userID, repoID int64) (teams TeamList, err error) { |
||||||
|
return teams, db.GetEngine(ctx). |
||||||
|
Join("INNER", "team_user", "team_user.team_id = team.id"). |
||||||
|
Join("INNER", "team_repo", "team_repo.team_id = team.id"). |
||||||
|
Where("team.org_id = ?", orgID). |
||||||
|
And("team_user.uid=?", userID). |
||||||
|
And("team_repo.repo_id=?", repoID). |
||||||
|
Find(&teams) |
||||||
|
} |
@ -0,0 +1,17 @@ |
|||||||
|
// Copyright 2018 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package org_test |
||||||
|
|
||||||
|
import ( |
||||||
|
"path/filepath" |
||||||
|
"testing" |
||||||
|
|
||||||
|
"code.gitea.io/gitea/models/unittest" |
||||||
|
) |
||||||
|
|
||||||
|
func TestMain(m *testing.M) { |
||||||
|
unittest.MainTest(m, &unittest.TestOptions{ |
||||||
|
GiteaRootPath: filepath.Join("..", "..", ".."), |
||||||
|
}) |
||||||
|
} |
@ -0,0 +1,670 @@ |
|||||||
|
// Copyright 2022 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package org |
||||||
|
|
||||||
|
import ( |
||||||
|
"errors" |
||||||
|
"fmt" |
||||||
|
"net/http" |
||||||
|
"net/url" |
||||||
|
"strconv" |
||||||
|
"strings" |
||||||
|
|
||||||
|
issues_model "code.gitea.io/gitea/models/issues" |
||||||
|
project_model "code.gitea.io/gitea/models/project" |
||||||
|
"code.gitea.io/gitea/models/unit" |
||||||
|
"code.gitea.io/gitea/modules/base" |
||||||
|
"code.gitea.io/gitea/modules/context" |
||||||
|
"code.gitea.io/gitea/modules/json" |
||||||
|
"code.gitea.io/gitea/modules/setting" |
||||||
|
"code.gitea.io/gitea/modules/util" |
||||||
|
"code.gitea.io/gitea/modules/web" |
||||||
|
shared_user "code.gitea.io/gitea/routers/web/shared/user" |
||||||
|
"code.gitea.io/gitea/services/forms" |
||||||
|
) |
||||||
|
|
||||||
|
const ( |
||||||
|
tplProjects base.TplName = "org/projects/list" |
||||||
|
tplProjectsNew base.TplName = "org/projects/new" |
||||||
|
tplProjectsView base.TplName = "org/projects/view" |
||||||
|
tplGenericProjectsNew base.TplName = "user/project" |
||||||
|
) |
||||||
|
|
||||||
|
// MustEnableProjects check if projects are enabled in settings
|
||||||
|
func MustEnableProjects(ctx *context.Context) { |
||||||
|
if unit.TypeProjects.UnitGlobalDisabled() { |
||||||
|
ctx.NotFound("EnableKanbanBoard", nil) |
||||||
|
return |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
// Projects renders the home page of projects
|
||||||
|
func Projects(ctx *context.Context) { |
||||||
|
ctx.Data["Title"] = ctx.Tr("repo.project_board") |
||||||
|
|
||||||
|
sortType := ctx.FormTrim("sort") |
||||||
|
|
||||||
|
isShowClosed := strings.ToLower(ctx.FormTrim("state")) == "closed" |
||||||
|
page := ctx.FormInt("page") |
||||||
|
if page <= 1 { |
||||||
|
page = 1 |
||||||
|
} |
||||||
|
|
||||||
|
projects, total, err := project_model.FindProjects(ctx, project_model.SearchOptions{ |
||||||
|
OwnerID: ctx.ContextUser.ID, |
||||||
|
Page: page, |
||||||
|
IsClosed: util.OptionalBoolOf(isShowClosed), |
||||||
|
SortType: sortType, |
||||||
|
Type: project_model.TypeOrganization, |
||||||
|
}) |
||||||
|
if err != nil { |
||||||
|
ctx.ServerError("FindProjects", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
opTotal, err := project_model.CountProjects(ctx, project_model.SearchOptions{ |
||||||
|
OwnerID: ctx.ContextUser.ID, |
||||||
|
IsClosed: util.OptionalBoolOf(!isShowClosed), |
||||||
|
Type: project_model.TypeOrganization, |
||||||
|
}) |
||||||
|
if err != nil { |
||||||
|
ctx.ServerError("CountProjects", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if isShowClosed { |
||||||
|
ctx.Data["OpenCount"] = opTotal |
||||||
|
ctx.Data["ClosedCount"] = total |
||||||
|
} else { |
||||||
|
ctx.Data["OpenCount"] = total |
||||||
|
ctx.Data["ClosedCount"] = opTotal |
||||||
|
} |
||||||
|
|
||||||
|
ctx.Data["Projects"] = projects |
||||||
|
shared_user.RenderUserHeader(ctx) |
||||||
|
|
||||||
|
if isShowClosed { |
||||||
|
ctx.Data["State"] = "closed" |
||||||
|
} else { |
||||||
|
ctx.Data["State"] = "open" |
||||||
|
} |
||||||
|
|
||||||
|
for _, project := range projects { |
||||||
|
project.RenderedContent = project.Description |
||||||
|
} |
||||||
|
|
||||||
|
numPages := 0 |
||||||
|
if total > 0 { |
||||||
|
numPages = (int(total) - 1/setting.UI.IssuePagingNum) |
||||||
|
} |
||||||
|
|
||||||
|
pager := context.NewPagination(int(total), setting.UI.IssuePagingNum, page, numPages) |
||||||
|
pager.AddParam(ctx, "state", "State") |
||||||
|
ctx.Data["Page"] = pager |
||||||
|
|
||||||
|
ctx.Data["CanWriteProjects"] = canWriteUnit(ctx) |
||||||
|
ctx.Data["IsShowClosed"] = isShowClosed |
||||||
|
ctx.Data["PageIsViewProjects"] = true |
||||||
|
ctx.Data["SortType"] = sortType |
||||||
|
|
||||||
|
ctx.HTML(http.StatusOK, tplProjects) |
||||||
|
} |
||||||
|
|
||||||
|
func canWriteUnit(ctx *context.Context) bool { |
||||||
|
if ctx.ContextUser.IsOrganization() { |
||||||
|
return ctx.Org.CanWriteUnit(ctx, unit.TypeProjects) |
||||||
|
} |
||||||
|
return ctx.Doer != nil && ctx.ContextUser.ID == ctx.Doer.ID |
||||||
|
} |
||||||
|
|
||||||
|
// NewProject render creating a project page
|
||||||
|
func NewProject(ctx *context.Context) { |
||||||
|
ctx.Data["Title"] = ctx.Tr("repo.projects.new") |
||||||
|
ctx.Data["ProjectTypes"] = project_model.GetProjectsConfig() |
||||||
|
ctx.Data["CanWriteProjects"] = canWriteUnit(ctx) |
||||||
|
ctx.Data["HomeLink"] = ctx.ContextUser.HomeLink() |
||||||
|
shared_user.RenderUserHeader(ctx) |
||||||
|
ctx.HTML(http.StatusOK, tplProjectsNew) |
||||||
|
} |
||||||
|
|
||||||
|
// NewProjectPost creates a new project
|
||||||
|
func NewProjectPost(ctx *context.Context) { |
||||||
|
form := web.GetForm(ctx).(*forms.CreateProjectForm) |
||||||
|
ctx.Data["Title"] = ctx.Tr("repo.projects.new") |
||||||
|
shared_user.RenderUserHeader(ctx) |
||||||
|
|
||||||
|
if ctx.HasError() { |
||||||
|
ctx.Data["CanWriteProjects"] = canWriteUnit(ctx) |
||||||
|
ctx.Data["PageIsViewProjects"] = true |
||||||
|
ctx.Data["ProjectTypes"] = project_model.GetProjectsConfig() |
||||||
|
ctx.HTML(http.StatusOK, tplProjectsNew) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if err := project_model.NewProject(&project_model.Project{ |
||||||
|
OwnerID: ctx.ContextUser.ID, |
||||||
|
Title: form.Title, |
||||||
|
Description: form.Content, |
||||||
|
CreatorID: ctx.Doer.ID, |
||||||
|
BoardType: form.BoardType, |
||||||
|
Type: project_model.TypeOrganization, |
||||||
|
}); err != nil { |
||||||
|
ctx.ServerError("NewProject", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
ctx.Flash.Success(ctx.Tr("repo.projects.create_success", form.Title)) |
||||||
|
ctx.Redirect(ctx.ContextUser.HomeLink() + "/-/projects") |
||||||
|
} |
||||||
|
|
||||||
|
// ChangeProjectStatus updates the status of a project between "open" and "close"
|
||||||
|
func ChangeProjectStatus(ctx *context.Context) { |
||||||
|
toClose := false |
||||||
|
switch ctx.Params(":action") { |
||||||
|
case "open": |
||||||
|
toClose = false |
||||||
|
case "close": |
||||||
|
toClose = true |
||||||
|
default: |
||||||
|
ctx.Redirect(ctx.Repo.RepoLink + "/projects") |
||||||
|
} |
||||||
|
id := ctx.ParamsInt64(":id") |
||||||
|
|
||||||
|
if err := project_model.ChangeProjectStatusByRepoIDAndID(ctx.Repo.Repository.ID, id, toClose); err != nil { |
||||||
|
if project_model.IsErrProjectNotExist(err) { |
||||||
|
ctx.NotFound("", err) |
||||||
|
} else { |
||||||
|
ctx.ServerError("ChangeProjectStatusByIDAndRepoID", err) |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
ctx.Redirect(ctx.Repo.RepoLink + "/projects?state=" + url.QueryEscape(ctx.Params(":action"))) |
||||||
|
} |
||||||
|
|
||||||
|
// DeleteProject delete a project
|
||||||
|
func DeleteProject(ctx *context.Context) { |
||||||
|
p, err := project_model.GetProjectByID(ctx, ctx.ParamsInt64(":id")) |
||||||
|
if err != nil { |
||||||
|
if project_model.IsErrProjectNotExist(err) { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
} else { |
||||||
|
ctx.ServerError("GetProjectByID", err) |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
if p.RepoID != ctx.Repo.Repository.ID { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if err := project_model.DeleteProjectByID(ctx, p.ID); err != nil { |
||||||
|
ctx.Flash.Error("DeleteProjectByID: " + err.Error()) |
||||||
|
} else { |
||||||
|
ctx.Flash.Success(ctx.Tr("repo.projects.deletion_success")) |
||||||
|
} |
||||||
|
|
||||||
|
ctx.JSON(http.StatusOK, map[string]interface{}{ |
||||||
|
"redirect": ctx.Repo.RepoLink + "/projects", |
||||||
|
}) |
||||||
|
} |
||||||
|
|
||||||
|
// EditProject allows a project to be edited
|
||||||
|
func EditProject(ctx *context.Context) { |
||||||
|
ctx.Data["Title"] = ctx.Tr("repo.projects.edit") |
||||||
|
ctx.Data["PageIsEditProjects"] = true |
||||||
|
ctx.Data["PageIsViewProjects"] = true |
||||||
|
ctx.Data["CanWriteProjects"] = canWriteUnit(ctx) |
||||||
|
shared_user.RenderUserHeader(ctx) |
||||||
|
|
||||||
|
p, err := project_model.GetProjectByID(ctx, ctx.ParamsInt64(":id")) |
||||||
|
if err != nil { |
||||||
|
if project_model.IsErrProjectNotExist(err) { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
} else { |
||||||
|
ctx.ServerError("GetProjectByID", err) |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
if p.RepoID != ctx.Repo.Repository.ID { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
ctx.Data["title"] = p.Title |
||||||
|
ctx.Data["content"] = p.Description |
||||||
|
|
||||||
|
ctx.HTML(http.StatusOK, tplProjectsNew) |
||||||
|
} |
||||||
|
|
||||||
|
// EditProjectPost response for editing a project
|
||||||
|
func EditProjectPost(ctx *context.Context) { |
||||||
|
form := web.GetForm(ctx).(*forms.CreateProjectForm) |
||||||
|
ctx.Data["Title"] = ctx.Tr("repo.projects.edit") |
||||||
|
ctx.Data["PageIsEditProjects"] = true |
||||||
|
ctx.Data["PageIsViewProjects"] = true |
||||||
|
ctx.Data["CanWriteProjects"] = canWriteUnit(ctx) |
||||||
|
shared_user.RenderUserHeader(ctx) |
||||||
|
|
||||||
|
if ctx.HasError() { |
||||||
|
ctx.HTML(http.StatusOK, tplProjectsNew) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
p, err := project_model.GetProjectByID(ctx, ctx.ParamsInt64(":id")) |
||||||
|
if err != nil { |
||||||
|
if project_model.IsErrProjectNotExist(err) { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
} else { |
||||||
|
ctx.ServerError("GetProjectByID", err) |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
if p.RepoID != ctx.Repo.Repository.ID { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
p.Title = form.Title |
||||||
|
p.Description = form.Content |
||||||
|
if err = project_model.UpdateProject(ctx, p); err != nil { |
||||||
|
ctx.ServerError("UpdateProjects", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
ctx.Flash.Success(ctx.Tr("repo.projects.edit_success", p.Title)) |
||||||
|
ctx.Redirect(ctx.Repo.RepoLink + "/projects") |
||||||
|
} |
||||||
|
|
||||||
|
// ViewProject renders the project board for a project
|
||||||
|
func ViewProject(ctx *context.Context) { |
||||||
|
project, err := project_model.GetProjectByID(ctx, ctx.ParamsInt64(":id")) |
||||||
|
if err != nil { |
||||||
|
if project_model.IsErrProjectNotExist(err) { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
} else { |
||||||
|
ctx.ServerError("GetProjectByID", err) |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
if project.OwnerID != ctx.ContextUser.ID { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
boards, err := project_model.GetBoards(ctx, project.ID) |
||||||
|
if err != nil { |
||||||
|
ctx.ServerError("GetProjectBoards", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if boards[0].ID == 0 { |
||||||
|
boards[0].Title = ctx.Tr("repo.projects.type.uncategorized") |
||||||
|
} |
||||||
|
|
||||||
|
issuesMap, err := issues_model.LoadIssuesFromBoardList(ctx, boards) |
||||||
|
if err != nil { |
||||||
|
ctx.ServerError("LoadIssuesOfBoards", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
linkedPrsMap := make(map[int64][]*issues_model.Issue) |
||||||
|
for _, issuesList := range issuesMap { |
||||||
|
for _, issue := range issuesList { |
||||||
|
var referencedIds []int64 |
||||||
|
for _, comment := range issue.Comments { |
||||||
|
if comment.RefIssueID != 0 && comment.RefIsPull { |
||||||
|
referencedIds = append(referencedIds, comment.RefIssueID) |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
if len(referencedIds) > 0 { |
||||||
|
if linkedPrs, err := issues_model.Issues(ctx, &issues_model.IssuesOptions{ |
||||||
|
IssueIDs: referencedIds, |
||||||
|
IsPull: util.OptionalBoolTrue, |
||||||
|
}); err == nil { |
||||||
|
linkedPrsMap[issue.ID] = linkedPrs |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
project.RenderedContent = project.Description |
||||||
|
ctx.Data["LinkedPRs"] = linkedPrsMap |
||||||
|
ctx.Data["PageIsViewProjects"] = true |
||||||
|
ctx.Data["CanWriteProjects"] = canWriteUnit(ctx) |
||||||
|
ctx.Data["Project"] = project |
||||||
|
ctx.Data["IssuesMap"] = issuesMap |
||||||
|
ctx.Data["Boards"] = boards |
||||||
|
shared_user.RenderUserHeader(ctx) |
||||||
|
|
||||||
|
ctx.HTML(http.StatusOK, tplProjectsView) |
||||||
|
} |
||||||
|
|
||||||
|
func getActionIssues(ctx *context.Context) []*issues_model.Issue { |
||||||
|
commaSeparatedIssueIDs := ctx.FormString("issue_ids") |
||||||
|
if len(commaSeparatedIssueIDs) == 0 { |
||||||
|
return nil |
||||||
|
} |
||||||
|
issueIDs := make([]int64, 0, 10) |
||||||
|
for _, stringIssueID := range strings.Split(commaSeparatedIssueIDs, ",") { |
||||||
|
issueID, err := strconv.ParseInt(stringIssueID, 10, 64) |
||||||
|
if err != nil { |
||||||
|
ctx.ServerError("ParseInt", err) |
||||||
|
return nil |
||||||
|
} |
||||||
|
issueIDs = append(issueIDs, issueID) |
||||||
|
} |
||||||
|
issues, err := issues_model.GetIssuesByIDs(ctx, issueIDs) |
||||||
|
if err != nil { |
||||||
|
ctx.ServerError("GetIssuesByIDs", err) |
||||||
|
return nil |
||||||
|
} |
||||||
|
// Check access rights for all issues
|
||||||
|
issueUnitEnabled := ctx.Repo.CanRead(unit.TypeIssues) |
||||||
|
prUnitEnabled := ctx.Repo.CanRead(unit.TypePullRequests) |
||||||
|
for _, issue := range issues { |
||||||
|
if issue.RepoID != ctx.Repo.Repository.ID { |
||||||
|
ctx.NotFound("some issue's RepoID is incorrect", errors.New("some issue's RepoID is incorrect")) |
||||||
|
return nil |
||||||
|
} |
||||||
|
if issue.IsPull && !prUnitEnabled || !issue.IsPull && !issueUnitEnabled { |
||||||
|
ctx.NotFound("IssueOrPullRequestUnitNotAllowed", nil) |
||||||
|
return nil |
||||||
|
} |
||||||
|
if err = issue.LoadAttributes(ctx); err != nil { |
||||||
|
ctx.ServerError("LoadAttributes", err) |
||||||
|
return nil |
||||||
|
} |
||||||
|
} |
||||||
|
return issues |
||||||
|
} |
||||||
|
|
||||||
|
// UpdateIssueProject change an issue's project
|
||||||
|
func UpdateIssueProject(ctx *context.Context) { |
||||||
|
issues := getActionIssues(ctx) |
||||||
|
if ctx.Written() { |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
projectID := ctx.FormInt64("id") |
||||||
|
for _, issue := range issues { |
||||||
|
oldProjectID := issue.ProjectID() |
||||||
|
if oldProjectID == projectID { |
||||||
|
continue |
||||||
|
} |
||||||
|
|
||||||
|
if err := issues_model.ChangeProjectAssign(issue, ctx.Doer, projectID); err != nil { |
||||||
|
ctx.ServerError("ChangeProjectAssign", err) |
||||||
|
return |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
ctx.JSON(http.StatusOK, map[string]interface{}{ |
||||||
|
"ok": true, |
||||||
|
}) |
||||||
|
} |
||||||
|
|
||||||
|
// DeleteProjectBoard allows for the deletion of a project board
|
||||||
|
func DeleteProjectBoard(ctx *context.Context) { |
||||||
|
if ctx.Doer == nil { |
||||||
|
ctx.JSON(http.StatusForbidden, map[string]string{ |
||||||
|
"message": "Only signed in users are allowed to perform this action.", |
||||||
|
}) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
project, err := project_model.GetProjectByID(ctx, ctx.ParamsInt64(":id")) |
||||||
|
if err != nil { |
||||||
|
if project_model.IsErrProjectNotExist(err) { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
} else { |
||||||
|
ctx.ServerError("GetProjectByID", err) |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
pb, err := project_model.GetBoard(ctx, ctx.ParamsInt64(":boardID")) |
||||||
|
if err != nil { |
||||||
|
ctx.ServerError("GetProjectBoard", err) |
||||||
|
return |
||||||
|
} |
||||||
|
if pb.ProjectID != ctx.ParamsInt64(":id") { |
||||||
|
ctx.JSON(http.StatusUnprocessableEntity, map[string]string{ |
||||||
|
"message": fmt.Sprintf("ProjectBoard[%d] is not in Project[%d] as expected", pb.ID, project.ID), |
||||||
|
}) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if project.OwnerID != ctx.ContextUser.ID { |
||||||
|
ctx.JSON(http.StatusUnprocessableEntity, map[string]string{ |
||||||
|
"message": fmt.Sprintf("ProjectBoard[%d] is not in Owner[%d] as expected", pb.ID, ctx.ContextUser.ID), |
||||||
|
}) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if err := project_model.DeleteBoardByID(ctx.ParamsInt64(":boardID")); err != nil { |
||||||
|
ctx.ServerError("DeleteProjectBoardByID", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
ctx.JSON(http.StatusOK, map[string]interface{}{ |
||||||
|
"ok": true, |
||||||
|
}) |
||||||
|
} |
||||||
|
|
||||||
|
// AddBoardToProjectPost allows a new board to be added to a project.
|
||||||
|
func AddBoardToProjectPost(ctx *context.Context) { |
||||||
|
form := web.GetForm(ctx).(*forms.EditProjectBoardForm) |
||||||
|
|
||||||
|
project, err := project_model.GetProjectByID(ctx, ctx.ParamsInt64(":id")) |
||||||
|
if err != nil { |
||||||
|
if project_model.IsErrProjectNotExist(err) { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
} else { |
||||||
|
ctx.ServerError("GetProjectByID", err) |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if err := project_model.NewBoard(&project_model.Board{ |
||||||
|
ProjectID: project.ID, |
||||||
|
Title: form.Title, |
||||||
|
Color: form.Color, |
||||||
|
CreatorID: ctx.Doer.ID, |
||||||
|
}); err != nil { |
||||||
|
ctx.ServerError("NewProjectBoard", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
ctx.JSON(http.StatusOK, map[string]interface{}{ |
||||||
|
"ok": true, |
||||||
|
}) |
||||||
|
} |
||||||
|
|
||||||
|
// CheckProjectBoardChangePermissions check permission
|
||||||
|
func CheckProjectBoardChangePermissions(ctx *context.Context) (*project_model.Project, *project_model.Board) { |
||||||
|
if ctx.Doer == nil { |
||||||
|
ctx.JSON(http.StatusForbidden, map[string]string{ |
||||||
|
"message": "Only signed in users are allowed to perform this action.", |
||||||
|
}) |
||||||
|
return nil, nil |
||||||
|
} |
||||||
|
|
||||||
|
project, err := project_model.GetProjectByID(ctx, ctx.ParamsInt64(":id")) |
||||||
|
if err != nil { |
||||||
|
if project_model.IsErrProjectNotExist(err) { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
} else { |
||||||
|
ctx.ServerError("GetProjectByID", err) |
||||||
|
} |
||||||
|
return nil, nil |
||||||
|
} |
||||||
|
|
||||||
|
board, err := project_model.GetBoard(ctx, ctx.ParamsInt64(":boardID")) |
||||||
|
if err != nil { |
||||||
|
ctx.ServerError("GetProjectBoard", err) |
||||||
|
return nil, nil |
||||||
|
} |
||||||
|
if board.ProjectID != ctx.ParamsInt64(":id") { |
||||||
|
ctx.JSON(http.StatusUnprocessableEntity, map[string]string{ |
||||||
|
"message": fmt.Sprintf("ProjectBoard[%d] is not in Project[%d] as expected", board.ID, project.ID), |
||||||
|
}) |
||||||
|
return nil, nil |
||||||
|
} |
||||||
|
|
||||||
|
if project.OwnerID != ctx.ContextUser.ID { |
||||||
|
ctx.JSON(http.StatusUnprocessableEntity, map[string]string{ |
||||||
|
"message": fmt.Sprintf("ProjectBoard[%d] is not in Repository[%d] as expected", board.ID, project.ID), |
||||||
|
}) |
||||||
|
return nil, nil |
||||||
|
} |
||||||
|
return project, board |
||||||
|
} |
||||||
|
|
||||||
|
// EditProjectBoard allows a project board's to be updated
|
||||||
|
func EditProjectBoard(ctx *context.Context) { |
||||||
|
form := web.GetForm(ctx).(*forms.EditProjectBoardForm) |
||||||
|
_, board := CheckProjectBoardChangePermissions(ctx) |
||||||
|
if ctx.Written() { |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if form.Title != "" { |
||||||
|
board.Title = form.Title |
||||||
|
} |
||||||
|
|
||||||
|
board.Color = form.Color |
||||||
|
|
||||||
|
if form.Sorting != 0 { |
||||||
|
board.Sorting = form.Sorting |
||||||
|
} |
||||||
|
|
||||||
|
if err := project_model.UpdateBoard(ctx, board); err != nil { |
||||||
|
ctx.ServerError("UpdateProjectBoard", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
ctx.JSON(http.StatusOK, map[string]interface{}{ |
||||||
|
"ok": true, |
||||||
|
}) |
||||||
|
} |
||||||
|
|
||||||
|
// SetDefaultProjectBoard set default board for uncategorized issues/pulls
|
||||||
|
func SetDefaultProjectBoard(ctx *context.Context) { |
||||||
|
project, board := CheckProjectBoardChangePermissions(ctx) |
||||||
|
if ctx.Written() { |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if err := project_model.SetDefaultBoard(project.ID, board.ID); err != nil { |
||||||
|
ctx.ServerError("SetDefaultBoard", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
ctx.JSON(http.StatusOK, map[string]interface{}{ |
||||||
|
"ok": true, |
||||||
|
}) |
||||||
|
} |
||||||
|
|
||||||
|
// MoveIssues moves or keeps issues in a column and sorts them inside that column
|
||||||
|
func MoveIssues(ctx *context.Context) { |
||||||
|
if ctx.Doer == nil { |
||||||
|
ctx.JSON(http.StatusForbidden, map[string]string{ |
||||||
|
"message": "Only signed in users are allowed to perform this action.", |
||||||
|
}) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
project, err := project_model.GetProjectByID(ctx, ctx.ParamsInt64(":id")) |
||||||
|
if err != nil { |
||||||
|
if project_model.IsErrProjectNotExist(err) { |
||||||
|
ctx.NotFound("ProjectNotExist", nil) |
||||||
|
} else { |
||||||
|
ctx.ServerError("GetProjectByID", err) |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
if project.OwnerID != ctx.ContextUser.ID { |
||||||
|
ctx.NotFound("InvalidRepoID", nil) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
var board *project_model.Board |
||||||
|
|
||||||
|
if ctx.ParamsInt64(":boardID") == 0 { |
||||||
|
board = &project_model.Board{ |
||||||
|
ID: 0, |
||||||
|
ProjectID: project.ID, |
||||||
|
Title: ctx.Tr("repo.projects.type.uncategorized"), |
||||||
|
} |
||||||
|
} else { |
||||||
|
board, err = project_model.GetBoard(ctx, ctx.ParamsInt64(":boardID")) |
||||||
|
if err != nil { |
||||||
|
if project_model.IsErrProjectBoardNotExist(err) { |
||||||
|
ctx.NotFound("ProjectBoardNotExist", nil) |
||||||
|
} else { |
||||||
|
ctx.ServerError("GetProjectBoard", err) |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
if board.ProjectID != project.ID { |
||||||
|
ctx.NotFound("BoardNotInProject", nil) |
||||||
|
return |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
type movedIssuesForm struct { |
||||||
|
Issues []struct { |
||||||
|
IssueID int64 `json:"issueID"` |
||||||
|
Sorting int64 `json:"sorting"` |
||||||
|
} `json:"issues"` |
||||||
|
} |
||||||
|
|
||||||
|
form := &movedIssuesForm{} |
||||||
|
if err = json.NewDecoder(ctx.Req.Body).Decode(&form); err != nil { |
||||||
|
ctx.ServerError("DecodeMovedIssuesForm", err) |
||||||
|
} |
||||||
|
|
||||||
|
issueIDs := make([]int64, 0, len(form.Issues)) |
||||||
|
sortedIssueIDs := make(map[int64]int64) |
||||||
|
for _, issue := range form.Issues { |
||||||
|
issueIDs = append(issueIDs, issue.IssueID) |
||||||
|
sortedIssueIDs[issue.Sorting] = issue.IssueID |
||||||
|
} |
||||||
|
movedIssues, err := issues_model.GetIssuesByIDs(ctx, issueIDs) |
||||||
|
if err != nil { |
||||||
|
if issues_model.IsErrIssueNotExist(err) { |
||||||
|
ctx.NotFound("IssueNotExisting", nil) |
||||||
|
} else { |
||||||
|
ctx.ServerError("GetIssueByID", err) |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if len(movedIssues) != len(form.Issues) { |
||||||
|
ctx.ServerError("some issues do not exist", errors.New("some issues do not exist")) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if _, err = movedIssues.LoadRepositories(ctx); err != nil { |
||||||
|
ctx.ServerError("LoadRepositories", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
for _, issue := range movedIssues { |
||||||
|
if issue.RepoID != project.RepoID && issue.Repo.OwnerID != project.OwnerID { |
||||||
|
ctx.ServerError("Some issue's repoID is not equal to project's repoID", errors.New("Some issue's repoID is not equal to project's repoID")) |
||||||
|
return |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
if err = project_model.MoveIssuesOnProjectBoard(board, sortedIssueIDs); err != nil { |
||||||
|
ctx.ServerError("MoveIssuesOnProjectBoard", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
ctx.JSON(http.StatusOK, map[string]interface{}{ |
||||||
|
"ok": true, |
||||||
|
}) |
||||||
|
} |
@ -0,0 +1,28 @@ |
|||||||
|
// Copyright 2022 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package org_test |
||||||
|
|
||||||
|
import ( |
||||||
|
"testing" |
||||||
|
|
||||||
|
"code.gitea.io/gitea/models/unittest" |
||||||
|
"code.gitea.io/gitea/modules/test" |
||||||
|
"code.gitea.io/gitea/routers/web/org" |
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert" |
||||||
|
) |
||||||
|
|
||||||
|
func TestCheckProjectBoardChangePermissions(t *testing.T) { |
||||||
|
unittest.PrepareTestEnv(t) |
||||||
|
ctx := test.MockContext(t, "user2/-/projects/4/4") |
||||||
|
test.LoadUser(t, ctx, 2) |
||||||
|
ctx.ContextUser = ctx.Doer // user2
|
||||||
|
ctx.SetParams(":id", "4") |
||||||
|
ctx.SetParams(":boardID", "4") |
||||||
|
|
||||||
|
project, board := org.CheckProjectBoardChangePermissions(ctx) |
||||||
|
assert.NotNil(t, project) |
||||||
|
assert.NotNil(t, board) |
||||||
|
assert.False(t, ctx.Written()) |
||||||
|
} |
@ -0,0 +1,14 @@ |
|||||||
|
// Copyright 2022 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package user |
||||||
|
|
||||||
|
import ( |
||||||
|
"code.gitea.io/gitea/modules/context" |
||||||
|
"code.gitea.io/gitea/modules/setting" |
||||||
|
) |
||||||
|
|
||||||
|
func RenderUserHeader(ctx *context.Context) { |
||||||
|
ctx.Data["IsRepoIndexerEnabled"] = setting.Indexer.RepoIndexerEnabled |
||||||
|
ctx.Data["ContextUser"] = ctx.ContextUser |
||||||
|
} |
@ -0,0 +1,6 @@ |
|||||||
|
{{template "base/head" .}} |
||||||
|
<div class="page-content repository packages"> |
||||||
|
{{template "user/overview/header" .}} |
||||||
|
{{template "projects/list" .}} |
||||||
|
</div> |
||||||
|
{{template "base/footer" .}} |
@ -0,0 +1,6 @@ |
|||||||
|
{{template "base/head" .}} |
||||||
|
<div class="page-content repository packages"> |
||||||
|
{{template "user/overview/header" .}} |
||||||
|
{{template "projects/new" .}} |
||||||
|
</div> |
||||||
|
{{template "base/footer" .}} |
@ -0,0 +1,6 @@ |
|||||||
|
{{template "base/head" .}} |
||||||
|
<div class="page-content repository packages"> |
||||||
|
{{template "user/overview/header" .}} |
||||||
|
{{template "projects/view" .}} |
||||||
|
</div> |
||||||
|
{{template "base/footer" .}} |
@ -0,0 +1,98 @@ |
|||||||
|
<div class="page-content repository projects"> |
||||||
|
<div class="ui container"> |
||||||
|
{{if .CanWriteProjects}} |
||||||
|
<div class="navbar"> |
||||||
|
<div class="ui right"> |
||||||
|
<a class="ui green button" href="{{$.Link}}/new">{{.locale.Tr "repo.projects.new"}}</a> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
<div class="ui divider"></div> |
||||||
|
{{end}} |
||||||
|
|
||||||
|
{{template "base/alert" .}} |
||||||
|
<div class="ui compact tiny menu"> |
||||||
|
<a class="item{{if not .IsShowClosed}} active{{end}}" href="{{$.Link}}?state=open"> |
||||||
|
{{svg "octicon-project" 16 "mr-3"}} |
||||||
|
{{JsPrettyNumber .OpenCount}} {{.locale.Tr "repo.issues.open_title"}} |
||||||
|
</a> |
||||||
|
<a class="item{{if .IsShowClosed}} active{{end}}" href="{{$.Link}}?state=closed"> |
||||||
|
{{svg "octicon-check" 16 "mr-3"}} |
||||||
|
{{JsPrettyNumber .ClosedCount}} {{.locale.Tr "repo.issues.closed_title"}} |
||||||
|
</a> |
||||||
|
</div> |
||||||
|
|
||||||
|
<div class="ui right floated secondary filter menu"> |
||||||
|
<!-- Sort --> |
||||||
|
<div class="ui dropdown type jump item"> |
||||||
|
<span class="text"> |
||||||
|
{{.locale.Tr "repo.issues.filter_sort"}} |
||||||
|
{{svg "octicon-triangle-down" 14 "dropdown icon"}} |
||||||
|
</span> |
||||||
|
<div class="menu"> |
||||||
|
<a class="{{if eq .SortType "oldest"}}active {{end}}item" href="{{$.Link}}?q={{$.Keyword}}&sort=oldest&state={{$.State}}">{{.locale.Tr "repo.issues.filter_sort.oldest"}}</a> |
||||||
|
<a class="{{if eq .SortType "recentupdate"}}active {{end}}item" href="{{$.Link}}?q={{$.Keyword}}&sort=recentupdate&state={{$.State}}">{{.locale.Tr "repo.issues.filter_sort.recentupdate"}}</a> |
||||||
|
<a class="{{if eq .SortType "leastupdate"}}active {{end}}item" href="{{$.Link}}?q={{$.Keyword}}&sort=leastupdate&state={{$.State}}">{{.locale.Tr "repo.issues.filter_sort.leastupdate"}}</a> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
<div class="milestone list"> |
||||||
|
{{range .Projects}} |
||||||
|
<li class="item"> |
||||||
|
{{svg "octicon-project"}} <a href="{{$.Link}}/{{.ID}}">{{.Title}}</a> |
||||||
|
<div class="meta"> |
||||||
|
{{$closedDate:= TimeSinceUnix .ClosedDateUnix $.locale}} |
||||||
|
{{if .IsClosed}} |
||||||
|
{{svg "octicon-clock"}} {{$.locale.Tr "repo.milestones.closed" $closedDate|Str2html}} |
||||||
|
{{end}} |
||||||
|
<span class="issue-stats"> |
||||||
|
{{svg "octicon-issue-opened" 16 "mr-3"}} |
||||||
|
{{JsPrettyNumber .NumOpenIssues}} {{$.locale.Tr "repo.issues.open_title"}} |
||||||
|
{{svg "octicon-check" 16 "mr-3"}} |
||||||
|
{{JsPrettyNumber .NumClosedIssues}} {{$.locale.Tr "repo.issues.closed_title"}} |
||||||
|
</span> |
||||||
|
</div> |
||||||
|
{{if and (or $.CanWriteIssues $.CanWritePulls) (not $.Repository.IsArchived)}} |
||||||
|
<div class="ui right operate"> |
||||||
|
<a href="{{$.Link}}/{{.ID}}/edit" data-id={{.ID}} data-title={{.Title}}>{{svg "octicon-pencil"}} {{$.locale.Tr "repo.issues.label_edit"}}</a> |
||||||
|
{{if .IsClosed}} |
||||||
|
<a class="link-action" href data-url="{{$.Link}}/{{.ID}}/open">{{svg "octicon-check"}} {{$.locale.Tr "repo.projects.open"}}</a> |
||||||
|
{{else}} |
||||||
|
<a class="link-action" href data-url="{{$.Link}}/{{.ID}}/close">{{svg "octicon-skip"}} {{$.locale.Tr "repo.projects.close"}}</a> |
||||||
|
{{end}} |
||||||
|
<a class="delete-button" href="#" data-url="{{$.RepoLink}}/projects/{{.ID}}/delete" data-id="{{.ID}}">{{svg "octicon-trash"}} {{$.locale.Tr "repo.issues.label_delete"}}</a> |
||||||
|
</div> |
||||||
|
{{end}} |
||||||
|
{{if .Description}} |
||||||
|
<div class="content"> |
||||||
|
{{.RenderedContent|Str2html}} |
||||||
|
</div> |
||||||
|
{{end}} |
||||||
|
</li> |
||||||
|
{{end}} |
||||||
|
|
||||||
|
{{template "base/paginate" .}} |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
|
||||||
|
{{if or .CanWriteIssues .CanWritePulls}} |
||||||
|
<div class="ui small basic delete modal"> |
||||||
|
<div class="ui icon header"> |
||||||
|
{{svg "octicon-trash"}} |
||||||
|
{{.locale.Tr "repo.projects.deletion"}} |
||||||
|
</div> |
||||||
|
<div class="content"> |
||||||
|
<p>{{.locale.Tr "repo.projects.deletion_desc"}}</p> |
||||||
|
</div> |
||||||
|
<div class="actions"> |
||||||
|
<div class="ui red basic inverted cancel button"> |
||||||
|
<i class="remove icon"></i> |
||||||
|
{{.locale.Tr "modal.no"}} |
||||||
|
</div> |
||||||
|
<div class="ui green basic inverted ok button"> |
||||||
|
<i class="checkmark icon"></i> |
||||||
|
{{.locale.Tr "modal.yes"}} |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
{{end}} |
@ -0,0 +1,66 @@ |
|||||||
|
<div class="page-content repository projects edit-project new milestone"> |
||||||
|
<div class="ui container"> |
||||||
|
<div class="navbar"> |
||||||
|
{{if and .CanWriteProjects .PageIsEditProject}} |
||||||
|
<div class="ui right floated secondary menu"> |
||||||
|
<a class="ui green button" href="{{$.HomeLink}}/-/projects/new">{{.locale.Tr "repo.milestones.new"}}</a> |
||||||
|
</div> |
||||||
|
{{end}} |
||||||
|
</div> |
||||||
|
<div class="ui divider"></div> |
||||||
|
<h2 class="ui dividing header"> |
||||||
|
{{if .PageIsEditProjects}} |
||||||
|
{{.locale.Tr "repo.projects.edit"}} |
||||||
|
<div class="sub header">{{.locale.Tr "repo.projects.edit_subheader"}}</div> |
||||||
|
{{else}} |
||||||
|
{{.locale.Tr "repo.projects.new"}} |
||||||
|
<div class="sub header">{{.locale.Tr "repo.projects.new_subheader"}}</div> |
||||||
|
{{end}} |
||||||
|
</h2> |
||||||
|
{{template "base/alert" .}} |
||||||
|
<form class="ui form grid" action="{{.Link}}" method="post"> |
||||||
|
{{.CsrfTokenHtml}} |
||||||
|
<div class="eleven wide column"> |
||||||
|
<div class="field {{if .Err_Title}}error{{end}}"> |
||||||
|
<label>{{.locale.Tr "repo.projects.title"}}</label> |
||||||
|
<input name="title" placeholder="{{.locale.Tr "repo.projects.title"}}" value="{{.title}}" autofocus required> |
||||||
|
</div> |
||||||
|
<div class="field"> |
||||||
|
<label>{{.locale.Tr "repo.projects.description"}}</label> |
||||||
|
<textarea name="content" placeholder="{{.locale.Tr "repo.projects.description_placeholder"}}">{{.content}}</textarea> |
||||||
|
</div> |
||||||
|
|
||||||
|
{{if not .PageIsEditProjects}} |
||||||
|
<label>{{.locale.Tr "repo.projects.template.desc"}}</label> |
||||||
|
<div class="ui selection dropdown"> |
||||||
|
<input type="hidden" name="board_type" value="{{.type}}"> |
||||||
|
<div class="default text">{{.locale.Tr "repo.projects.template.desc_helper"}}</div> |
||||||
|
<div class="menu"> |
||||||
|
{{range $element := .ProjectTypes}} |
||||||
|
<div class="item" data-id="{{$element.BoardType}}" data-value="{{$element.BoardType}}">{{$.locale.Tr $element.Translation}}</div> |
||||||
|
{{end}} |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
{{end}} |
||||||
|
</div> |
||||||
|
<div class="ui container"> |
||||||
|
<div class="ui divider"></div> |
||||||
|
<div class="ui left"> |
||||||
|
{{if .PageIsEditProjects}} |
||||||
|
<a class="ui primary basic button" href="{{.RepoLink}}/projects"> |
||||||
|
{{.locale.Tr "repo.milestones.cancel"}} |
||||||
|
</a> |
||||||
|
<button class="ui green button"> |
||||||
|
{{.locale.Tr "repo.projects.modify"}} |
||||||
|
</button> |
||||||
|
{{else}} |
||||||
|
<button class="ui green button"> |
||||||
|
{{.locale.Tr "repo.projects.create"}} |
||||||
|
</button> |
||||||
|
{{end}} |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
|
||||||
|
</form> |
||||||
|
</div> |
||||||
|
</div> |
@ -0,0 +1,279 @@ |
|||||||
|
<div class="page-content repository projects view-project"> |
||||||
|
<div class="ui container"> |
||||||
|
<div class="ui two column stackable grid"> |
||||||
|
<div class="column"> |
||||||
|
</div> |
||||||
|
<div class="column right aligned"> |
||||||
|
{{if .CanWriteProjects}} |
||||||
|
<a class="ui green button show-modal item" data-modal="#new-board-item">{{.locale.Tr "new_project_board"}}</a> |
||||||
|
{{end}} |
||||||
|
<div class="ui small modal new-board-modal" id="new-board-item"> |
||||||
|
<div class="header"> |
||||||
|
{{$.locale.Tr "repo.projects.board.new"}} |
||||||
|
</div> |
||||||
|
<div class="content"> |
||||||
|
<form class="ui form"> |
||||||
|
<div class="required field"> |
||||||
|
<label for="new_board">{{$.locale.Tr "repo.projects.board.new_title"}}</label> |
||||||
|
<input class="new-board" id="new_board" name="title" required> |
||||||
|
</div> |
||||||
|
|
||||||
|
<div class="field color-field"> |
||||||
|
<label for="new_board_color">{{$.locale.Tr "repo.projects.board.color"}}</label> |
||||||
|
<div class="color picker column"> |
||||||
|
<input class="color-picker" maxlength="7" placeholder="#c320f6" id="new_board_color_picker" name="color"> |
||||||
|
<div class="column precolors"> |
||||||
|
{{template "repo/issue/label_precolors"}} |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
|
||||||
|
<div class="text right actions"> |
||||||
|
<div class="ui cancel button">{{$.locale.Tr "settings.cancel"}}</div> |
||||||
|
<button data-url="{{$.Link}}" class="ui green button" id="new_board_submit">{{$.locale.Tr "repo.projects.board.new_submit"}}</button> |
||||||
|
</div> |
||||||
|
</form> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
<div class="ui divider"></div> |
||||||
|
<div class="ui two column stackable grid"> |
||||||
|
<div class="column"> |
||||||
|
<h2 class="project-title">{{$.Project.Title}}</h2> |
||||||
|
<div class="content project-description">{{$.Project.RenderedContent|Str2html}}</div> |
||||||
|
</div> |
||||||
|
{{if or $.CanWriteIssues $.CanWritePulls}} |
||||||
|
<div class="column right aligned"> |
||||||
|
<div class="ui compact right small menu"> |
||||||
|
<a class="item" href="{{$.Link}}/edit" data-id={{$.Project.ID}} data-title={{$.Project.Title}}> |
||||||
|
{{svg "octicon-pencil"}} |
||||||
|
<span class="mx-3">{{$.locale.Tr "repo.issues.label_edit"}}</span> |
||||||
|
</a> |
||||||
|
{{if .Project.IsClosed}} |
||||||
|
<a class="item link-action" href data-url="{{$.Link}}/open"> |
||||||
|
{{svg "octicon-check"}} |
||||||
|
<span class="mx-3">{{$.locale.Tr "repo.projects.open"}}</span> |
||||||
|
</a> |
||||||
|
{{else}} |
||||||
|
<a class="item link-action" href data-url="{{$.Link}}/close"> |
||||||
|
{{svg "octicon-skip"}} |
||||||
|
<span class="mx-3">{{$.locale.Tr "repo.projects.close"}}</span> |
||||||
|
</a> |
||||||
|
{{end}} |
||||||
|
<a class="item delete-button" href="#" data-url="{{$.Link}}/delete" data-id="{{.Project.ID}}"> |
||||||
|
{{svg "octicon-trash"}} |
||||||
|
<span class="mx-3">{{$.locale.Tr "repo.issues.label_delete"}}</span> |
||||||
|
</a> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
{{end}} |
||||||
|
</div> |
||||||
|
<div class="ui divider"></div> |
||||||
|
</div> |
||||||
|
<div class="ui container fluid padded" id="project-board"> |
||||||
|
|
||||||
|
<div class="board"> |
||||||
|
{{range $board := .Boards}} |
||||||
|
|
||||||
|
<div class="ui segment board-column" style="background: {{.Color}} !important;" data-id="{{.ID}}" data-sorting="{{.Sorting}}" data-url="{{$.Link}}/{{.ID}}"> |
||||||
|
<div class="board-column-header df ac sb"> |
||||||
|
<div class="ui large label board-label py-2"> |
||||||
|
<div class="ui small circular grey label board-card-cnt"> |
||||||
|
{{.NumIssues}} |
||||||
|
</div> |
||||||
|
{{.Title}} |
||||||
|
</div> |
||||||
|
{{if and $.CanWriteProjects (ne .ID 0)}} |
||||||
|
<div class="ui dropdown jump item tooltip"> |
||||||
|
<div class="not-mobile px-3" tabindex="-1"> |
||||||
|
{{svg "octicon-kebab-horizontal"}} |
||||||
|
</div> |
||||||
|
<div class="menu user-menu" tabindex="-1"> |
||||||
|
<a class="item show-modal button" data-modal="#edit-project-board-modal-{{.ID}}"> |
||||||
|
{{svg "octicon-pencil"}} |
||||||
|
{{$.locale.Tr "repo.projects.board.edit"}} |
||||||
|
</a> |
||||||
|
{{if not .Default}} |
||||||
|
<a class="item show-modal button" data-modal="#set-default-project-board-modal-{{.ID}}"> |
||||||
|
{{svg "octicon-pin"}} |
||||||
|
{{$.locale.Tr "repo.projects.board.set_default"}} |
||||||
|
</a> |
||||||
|
{{end}} |
||||||
|
<a class="item show-modal button" data-modal="#delete-board-modal-{{.ID}}"> |
||||||
|
{{svg "octicon-trash"}} |
||||||
|
{{$.locale.Tr "repo.projects.board.delete"}} |
||||||
|
</a> |
||||||
|
|
||||||
|
<div class="ui small modal edit-project-board" id="edit-project-board-modal-{{.ID}}"> |
||||||
|
<div class="header"> |
||||||
|
{{$.locale.Tr "repo.projects.board.edit"}} |
||||||
|
</div> |
||||||
|
<div class="content"> |
||||||
|
<form class="ui form"> |
||||||
|
<div class="required field"> |
||||||
|
<label for="new_board_title">{{$.locale.Tr "repo.projects.board.edit_title"}}</label> |
||||||
|
<input class="project-board-title" id="new_board_title" name="title" value="{{.Title}}" required> |
||||||
|
</div> |
||||||
|
|
||||||
|
<div class="field color-field"> |
||||||
|
<label for="new_board_color">{{$.locale.Tr "repo.projects.board.color"}}</label> |
||||||
|
<div class="color picker column"> |
||||||
|
<input class="color-picker" maxlength="7" placeholder="#c320f6" id="new_board_color" name="color" value="{{.Color}}"> |
||||||
|
<div class="column precolors"> |
||||||
|
{{template "repo/issue/label_precolors"}} |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
|
||||||
|
<div class="text right actions"> |
||||||
|
<div class="ui cancel button">{{$.locale.Tr "settings.cancel"}}</div> |
||||||
|
<button data-url="{{$.Link}}/{{.ID}}" class="ui red button">{{$.locale.Tr "repo.projects.board.edit"}}</button> |
||||||
|
</div> |
||||||
|
</form> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
|
||||||
|
<div class="ui basic modal" id="set-default-project-board-modal-{{.ID}}"> |
||||||
|
<div class="ui icon header"> |
||||||
|
{{$.locale.Tr "repo.projects.board.set_default"}} |
||||||
|
</div> |
||||||
|
<div class="content center"> |
||||||
|
<label> |
||||||
|
{{$.locale.Tr "repo.projects.board.set_default_desc"}} |
||||||
|
</label> |
||||||
|
</div> |
||||||
|
<div class="text right actions"> |
||||||
|
<div class="ui cancel button">{{$.locale.Tr "settings.cancel"}}</div> |
||||||
|
<button class="ui red button set-default-project-board" data-url="{{$.Link}}/{{.ID}}/default">{{$.locale.Tr "repo.projects.board.set_default"}}</button> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
|
||||||
|
<div class="ui basic modal" id="delete-board-modal-{{.ID}}"> |
||||||
|
<div class="ui icon header"> |
||||||
|
{{$.locale.Tr "repo.projects.board.delete"}} |
||||||
|
</div> |
||||||
|
<div class="content center"> |
||||||
|
<label> |
||||||
|
{{$.locale.Tr "repo.projects.board.deletion_desc"}} |
||||||
|
</label> |
||||||
|
</div> |
||||||
|
<div class="text right actions"> |
||||||
|
<div class="ui cancel button">{{$.locale.Tr "settings.cancel"}}</div> |
||||||
|
<button class="ui red button delete-project-board" data-url="{{$.Link}}/{{.ID}}">{{$.locale.Tr "repo.projects.board.delete"}}</button> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
{{end}} |
||||||
|
</div> |
||||||
|
<div class="ui divider"></div> |
||||||
|
|
||||||
|
<div class="ui cards board" data-url="{{$.Link}}/{{.ID}}" data-project="{{$.Project.ID}}" data-board="{{.ID}}" id="board_{{.ID}}"> |
||||||
|
|
||||||
|
{{range (index $.IssuesMap .ID)}} |
||||||
|
|
||||||
|
<!-- start issue card --> |
||||||
|
<div class="card board-card" data-issue="{{.ID}}"> |
||||||
|
<div class="content p-0"> |
||||||
|
<div class="header"> |
||||||
|
<span class="dif ac vm {{if .IsClosed}}red{{else}}green{{end}}"> |
||||||
|
{{if .IsPull}} |
||||||
|
{{if .PullRequest.HasMerged}} |
||||||
|
{{svg "octicon-git-merge" 16 "text purple"}} |
||||||
|
{{else}} |
||||||
|
{{if .IsClosed}} |
||||||
|
{{svg "octicon-git-pull-request" 16 "text red"}} |
||||||
|
{{else}} |
||||||
|
{{svg "octicon-git-pull-request" 16 "text green"}} |
||||||
|
{{end}} |
||||||
|
{{end}} |
||||||
|
{{else}} |
||||||
|
{{if .IsClosed}} |
||||||
|
{{svg "octicon-issue-closed" 16 "text red"}} |
||||||
|
{{else}} |
||||||
|
{{svg "octicon-issue-opened" 16 "text green"}} |
||||||
|
{{end}} |
||||||
|
{{end}} |
||||||
|
</span> |
||||||
|
<a class="project-board-title vm" href="{{.Link}}"> |
||||||
|
{{.Title}} |
||||||
|
</a> |
||||||
|
</div> |
||||||
|
<div class="meta my-2"> |
||||||
|
<span class="text light grey"> |
||||||
|
{{.Repo.FullName}}#{{.Index}} |
||||||
|
{{$timeStr := TimeSinceUnix .GetLastEventTimestamp $.locale}} |
||||||
|
{{if .OriginalAuthor}} |
||||||
|
{{$.locale.Tr .GetLastEventLabelFake $timeStr (.OriginalAuthor|Escape) | Safe}} |
||||||
|
{{else if gt .Poster.ID 0}} |
||||||
|
{{$.locale.Tr .GetLastEventLabel $timeStr (.Poster.HomeLink|Escape) (.Poster.GetDisplayName | Escape) | Safe}} |
||||||
|
{{else}} |
||||||
|
{{$.locale.Tr .GetLastEventLabelFake $timeStr (.Poster.GetDisplayName | Escape) | Safe}} |
||||||
|
{{end}} |
||||||
|
</span> |
||||||
|
</div> |
||||||
|
{{- if .MilestoneID}} |
||||||
|
<div class="meta my-2"> |
||||||
|
<a class="milestone" href="{{$.RepoLink}}/milestone/{{.MilestoneID}}"> |
||||||
|
{{svg "octicon-milestone" 16 "mr-2 vm"}} |
||||||
|
<span class="vm">{{.Milestone.Name}}</span> |
||||||
|
</a> |
||||||
|
</div> |
||||||
|
{{- end}} |
||||||
|
{{- range index $.LinkedPRs .ID}} |
||||||
|
<div class="meta my-2"> |
||||||
|
<a href="{{$.RepoLink}}/pulls/{{.Index}}"> |
||||||
|
<span class="m-0 {{if .PullRequest.HasMerged}}purple{{else if .IsClosed}}red{{else}}green{{end}}">{{svg "octicon-git-merge" 16 "mr-2 vm"}}</span> |
||||||
|
<span class="vm">{{.Title}} <span class="text light grey">#{{.Index}}</span></span> |
||||||
|
</a> |
||||||
|
</div> |
||||||
|
{{- end}} |
||||||
|
</div> |
||||||
|
|
||||||
|
{{if or .Labels .Assignees}} |
||||||
|
<div class="extra content labels-list p-0 pt-2"> |
||||||
|
{{range .Labels}} |
||||||
|
<a class="ui label" target="_blank" href="{{$.RepoLink}}/issues?labels={{.ID}}" style="color: {{.ForegroundColor}}; background-color: {{.Color}};" title="{{.Description | RenderEmojiPlain}}">{{.Name | RenderEmoji}}</a> |
||||||
|
{{end}} |
||||||
|
<div class="right floated"> |
||||||
|
{{range .Assignees}} |
||||||
|
<a class="tooltip" target="_blank" href="{{.HTMLURL}}" data-content="{{$.locale.Tr "repo.projects.board.assigned_to"}} {{.Name}}">{{avatar . 28 "mini mr-3"}}</a> |
||||||
|
{{end}} |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
{{end}} |
||||||
|
</div> |
||||||
|
<!-- stop issue card --> |
||||||
|
|
||||||
|
{{end}} |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
{{end}} |
||||||
|
</div> |
||||||
|
|
||||||
|
</div> |
||||||
|
|
||||||
|
</div> |
||||||
|
|
||||||
|
{{if or .CanWriteIssues .CanWritePulls}} |
||||||
|
<div class="ui small basic delete modal"> |
||||||
|
<div class="ui icon header"> |
||||||
|
{{svg "octicon-trash"}} |
||||||
|
{{.locale.Tr "repo.projects.deletion"}} |
||||||
|
</div> |
||||||
|
<div class="content"> |
||||||
|
<p>{{.locale.Tr "repo.projects.deletion_desc"}}</p> |
||||||
|
</div> |
||||||
|
<div class="actions"> |
||||||
|
<div class="ui red basic inverted cancel button"> |
||||||
|
<i class="remove icon"></i> |
||||||
|
{{.locale.Tr "modal.no"}} |
||||||
|
</div> |
||||||
|
<div class="ui green basic inverted ok button"> |
||||||
|
<i class="checkmark icon"></i> |
||||||
|
{{.locale.Tr "modal.yes"}} |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
{{end}} |
Loading…
Reference in new issue