mirror of
https://gitea.com/gitea/tea.git
synced 2025-09-03 02:18:30 +02:00

Fix #772 Reviewed-on: https://gitea.com/gitea/tea/pulls/786 Reviewed-by: Bo-Yi Wu (吳柏毅) <appleboy.tw@gmail.com>
200 lines
4.9 KiB
Go
200 lines
4.9 KiB
Go
// Copyright 2020 The Gitea Authors. All rights reserved.
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
package interact
|
|
|
|
import (
|
|
"strings"
|
|
|
|
"code.gitea.io/sdk/gitea"
|
|
"code.gitea.io/tea/modules/config"
|
|
"code.gitea.io/tea/modules/task"
|
|
"code.gitea.io/tea/modules/theme"
|
|
|
|
"github.com/charmbracelet/huh"
|
|
)
|
|
|
|
// IsQuitting checks if the user has aborted the interactive prompt
|
|
func IsQuitting(err error) bool {
|
|
return err == huh.ErrUserAborted
|
|
}
|
|
|
|
// CreateIssue interactively creates an issue
|
|
func CreateIssue(login *config.Login, owner, repo string) error {
|
|
owner, repo, err := promptRepoSlug(owner, repo)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
printTitleAndContent("Target repo:", owner+"/"+repo)
|
|
|
|
var opts gitea.CreateIssueOption
|
|
if err := promptIssueProperties(login, owner, repo, &opts); err != nil {
|
|
return err
|
|
}
|
|
|
|
return task.CreateIssue(login, owner, repo, opts)
|
|
}
|
|
|
|
func promptIssueProperties(login *config.Login, owner, repo string, o *gitea.CreateIssueOption) error {
|
|
var milestoneName string
|
|
var err error
|
|
|
|
selectableChan := make(chan (issueSelectables), 1)
|
|
go fetchIssueSelectables(login, owner, repo, selectableChan)
|
|
|
|
// title
|
|
if err := huh.NewInput().
|
|
Title("Issue title:").
|
|
Value(&o.Title).
|
|
Validate(huh.ValidateNotEmpty()).
|
|
WithTheme(theme.GetTheme()).
|
|
Run(); err != nil {
|
|
return err
|
|
}
|
|
printTitleAndContent("Issue title:", o.Title)
|
|
|
|
// description
|
|
if err := huh.NewForm(
|
|
huh.NewGroup(
|
|
huh.NewText().
|
|
Title("Issue description(markdown):").
|
|
ExternalEditor(config.GetPreferences().Editor).
|
|
EditorExtension("md").
|
|
Value(&o.Body),
|
|
),
|
|
).WithTheme(theme.GetTheme()).
|
|
Run(); err != nil {
|
|
return err
|
|
}
|
|
printTitleAndContent("Issue description(markdown):", o.Body)
|
|
|
|
// wait until selectables are fetched
|
|
selectables := <-selectableChan
|
|
if selectables.Err != nil {
|
|
return selectables.Err
|
|
}
|
|
|
|
// skip remaining props if we don't have permission to set them
|
|
if !selectables.Repo.Permissions.Push {
|
|
return nil
|
|
}
|
|
|
|
// assignees
|
|
if o.Assignees, err = promptMultiSelect("Assignees:", selectables.Assignees, "[other]"); err != nil {
|
|
return err
|
|
}
|
|
printTitleAndContent("Assignees:", strings.Join(o.Assignees, "\n"))
|
|
|
|
// milestone
|
|
if len(selectables.MilestoneList) != 0 {
|
|
if milestoneName, err = promptSelect("Milestone:", selectables.MilestoneList, "", "[none]", ""); err != nil {
|
|
return err
|
|
}
|
|
o.Milestone = selectables.MilestoneMap[milestoneName]
|
|
printTitleAndContent("Milestone:", milestoneName)
|
|
}
|
|
|
|
// labels
|
|
if len(selectables.LabelList) != 0 {
|
|
options := make([]huh.Option[int64], 0, len(selectables.LabelList))
|
|
labelsMap := make(map[int64]string, len(selectables.LabelList))
|
|
for _, l := range selectables.LabelList {
|
|
options = append(options, huh.Option[int64]{Key: l, Value: selectables.LabelMap[l]})
|
|
labelsMap[selectables.LabelMap[l]] = l
|
|
}
|
|
if err := huh.NewMultiSelect[int64]().
|
|
Title("Labels:").
|
|
Options(options...).
|
|
Value(&o.Labels).
|
|
Run(); err != nil {
|
|
return err
|
|
}
|
|
var labels []string
|
|
for _, labelID := range o.Labels {
|
|
labels = append(labels, labelsMap[labelID])
|
|
}
|
|
printTitleAndContent("Labels:", strings.Join(labels, "\n"))
|
|
}
|
|
|
|
// deadline
|
|
if o.Deadline, err = promptDatetime("Due date:"); err != nil {
|
|
return err
|
|
}
|
|
deadlineStr := "No due date"
|
|
if o.Deadline != nil && !o.Deadline.IsZero() {
|
|
deadlineStr = o.Deadline.Format("2006-01-02")
|
|
}
|
|
printTitleAndContent("Due date:", deadlineStr)
|
|
|
|
return nil
|
|
}
|
|
|
|
type issueSelectables struct {
|
|
Repo *gitea.Repository
|
|
Assignees []string
|
|
MilestoneList []string
|
|
MilestoneMap map[string]int64
|
|
LabelList []string
|
|
LabelMap map[string]int64
|
|
Err error
|
|
}
|
|
|
|
func fetchIssueSelectables(login *config.Login, owner, repo string, done chan issueSelectables) {
|
|
// TODO PERF make these calls concurrent
|
|
r := issueSelectables{}
|
|
c := login.Client()
|
|
|
|
r.Repo, _, r.Err = c.GetRepo(owner, repo)
|
|
if r.Err != nil {
|
|
done <- r
|
|
return
|
|
}
|
|
// we can set the following properties only if we have write access to the repo
|
|
// so we fastpath this if not.
|
|
if !r.Repo.Permissions.Push {
|
|
done <- r
|
|
return
|
|
}
|
|
|
|
assignees, _, err := c.GetAssignees(owner, repo)
|
|
if err != nil {
|
|
r.Err = err
|
|
done <- r
|
|
return
|
|
}
|
|
r.Assignees = make([]string, len(assignees))
|
|
for i, u := range assignees {
|
|
r.Assignees[i] = u.UserName
|
|
}
|
|
|
|
milestones, _, err := c.ListRepoMilestones(owner, repo, gitea.ListMilestoneOption{})
|
|
if err != nil {
|
|
r.Err = err
|
|
done <- r
|
|
return
|
|
}
|
|
r.MilestoneMap = make(map[string]int64)
|
|
r.MilestoneList = make([]string, len(milestones))
|
|
for i, m := range milestones {
|
|
r.MilestoneMap[m.Title] = m.ID
|
|
r.MilestoneList[i] = m.Title
|
|
}
|
|
|
|
labels, _, err := c.ListRepoLabels(owner, repo, gitea.ListLabelsOptions{
|
|
ListOptions: gitea.ListOptions{Page: -1},
|
|
})
|
|
if err != nil {
|
|
r.Err = err
|
|
done <- r
|
|
return
|
|
}
|
|
r.LabelMap = make(map[string]int64)
|
|
r.LabelList = make([]string, len(labels))
|
|
for i, l := range labels {
|
|
r.LabelMap[l.Name] = l.ID
|
|
r.LabelList[i] = l.Name
|
|
}
|
|
|
|
done <- r
|
|
}
|