2025-02-22 09:42:15 +01:00
|
|
|
package component
|
|
|
|
|
|
|
|
import (
|
|
|
|
"forge.cadoles.com/wpetit/clearcase/internal/core/model"
|
|
|
|
"forge.cadoles.com/wpetit/clearcase/internal/http/form"
|
|
|
|
common "forge.cadoles.com/wpetit/clearcase/internal/http/handler/webui/common/component"
|
|
|
|
)
|
|
|
|
|
|
|
|
type IssuePageVModel struct {
|
|
|
|
SummaryForm *form.Form
|
|
|
|
IssueForm *form.Form
|
|
|
|
Projects []*model.Project
|
|
|
|
SelectedProjectID string
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewIssueSummaryForm() *form.Form {
|
|
|
|
return form.New(
|
|
|
|
form.NewField(
|
|
|
|
"project",
|
|
|
|
form.Attrs{},
|
2025-02-22 16:28:46 +01:00
|
|
|
form.NonEmpty("Ce champs ne doit pas être vide."),
|
2025-02-22 09:42:15 +01:00
|
|
|
),
|
|
|
|
form.NewField(
|
|
|
|
"summary",
|
|
|
|
form.Attrs{
|
|
|
|
"type": "textarea",
|
|
|
|
"rows": "20",
|
2025-02-22 16:28:46 +01:00
|
|
|
"placeholder": "Décrivez rapidement le sujet du problème rencontré ou de l'évolution souhaitée...",
|
2025-02-22 09:42:15 +01:00
|
|
|
},
|
2025-02-22 16:28:46 +01:00
|
|
|
form.NonEmpty("Ce champs ne doit pas être vide."),
|
2025-02-22 09:42:15 +01:00
|
|
|
),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewIssueForm() *form.Form {
|
|
|
|
return form.New(
|
2025-02-22 16:28:46 +01:00
|
|
|
form.NewField(
|
|
|
|
"title",
|
|
|
|
form.Attrs{
|
|
|
|
"type": "text",
|
|
|
|
},
|
|
|
|
form.NonEmpty("Ce champs ne doit pas être vide."),
|
|
|
|
),
|
2025-02-22 09:42:15 +01:00
|
|
|
form.NewField(
|
|
|
|
"content",
|
|
|
|
form.Attrs{
|
|
|
|
"type": "textarea",
|
2025-02-22 16:28:46 +01:00
|
|
|
"rows": "20",
|
2025-02-22 09:42:15 +01:00
|
|
|
},
|
2025-02-22 16:28:46 +01:00
|
|
|
form.NonEmpty("Ce champs ne doit pas être vide."),
|
2025-02-22 09:42:15 +01:00
|
|
|
),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
templ IssuePage(vmodel IssuePageVModel) {
|
2025-02-22 16:28:46 +01:00
|
|
|
@common.Page(common.WithTitle("Nouvelle demande")) {
|
2025-02-22 09:42:15 +01:00
|
|
|
<div class="container is-fluid">
|
|
|
|
<section class="section">
|
|
|
|
<div class="buttons is-right">
|
2025-02-22 16:28:46 +01:00
|
|
|
<a class="button is-medium" href={ common.BaseURL(ctx, common.WithPath("/auth/logout")) }>Se déconnecter</a>
|
2025-02-22 09:42:15 +01:00
|
|
|
</div>
|
|
|
|
<div class="columns">
|
2025-02-22 16:28:46 +01:00
|
|
|
<div class="column is-4">
|
2025-02-22 09:42:15 +01:00
|
|
|
<form id="summary-form" action={ common.CurrentURL(ctx) } method="post" hx-disabled-elt="#summary-form textarea, #summary-form select, #summary-form button" hx-indicator="#generation-progress">
|
2025-02-22 16:28:46 +01:00
|
|
|
<h2 class="title is-size-2">Résumé de la demande</h2>
|
2025-02-22 09:42:15 +01:00
|
|
|
<progress id="generation-progress" class="htmx-indicator progress"></progress>
|
|
|
|
@common.FormSelect(
|
2025-02-22 16:28:46 +01:00
|
|
|
vmodel.SummaryForm, "issue-project", "project", "Projet",
|
2025-02-22 09:42:15 +01:00
|
|
|
common.WithOptions(projectsToOptions(vmodel.Projects)...),
|
|
|
|
common.WithAttrs(
|
|
|
|
"hx-get", string(common.CurrentURL(ctx, common.WithoutValues("project", "*"))),
|
|
|
|
"hx-target", "body",
|
|
|
|
"hx-push-url", "true",
|
|
|
|
),
|
|
|
|
)
|
2025-02-22 15:32:10 +01:00
|
|
|
@common.FormTextarea(
|
2025-02-22 16:28:46 +01:00
|
|
|
vmodel.SummaryForm, "issue-summary", "summary", "Résumé",
|
2025-02-22 15:32:10 +01:00
|
|
|
common.WithTextareaAttrs(
|
|
|
|
"hx-on:change", "onSummaryChange(event)",
|
|
|
|
),
|
|
|
|
)
|
2025-02-22 16:28:46 +01:00
|
|
|
<div class="buttons is-right">
|
|
|
|
<button type="submit" class="button is-primary is-large">
|
|
|
|
<span class="icon">
|
|
|
|
<i class="fa fa-robot"></i>
|
|
|
|
</span>
|
|
|
|
<span>Génerer le ticket</span>
|
|
|
|
</button>
|
|
|
|
</div>
|
2025-02-22 09:42:15 +01:00
|
|
|
</form>
|
|
|
|
</div>
|
|
|
|
<div class="column">
|
2025-02-22 16:28:46 +01:00
|
|
|
<h2 class="title is-size-2">Votre demande</h2>
|
|
|
|
@common.FormField(vmodel.IssueForm, "issue-content", "title", "Titre")
|
|
|
|
@common.FormTextarea(vmodel.IssueForm, "issue-content", "content", "Contenu")
|
|
|
|
<div class="buttons is-right">
|
|
|
|
<button disabled type="submit" class="button is-primary is-large">
|
|
|
|
<span class="icon">
|
|
|
|
<i class="fa fa-rocket"></i>
|
|
|
|
</span>
|
|
|
|
<span>Créer le ticket</span>
|
|
|
|
</button>
|
2025-02-22 09:42:15 +01:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</section>
|
|
|
|
</div>
|
2025-02-22 15:32:10 +01:00
|
|
|
<script type="text/javascript">
|
|
|
|
function onSummaryChange(evt) {
|
|
|
|
const summary = evt.currentTarget.value;
|
|
|
|
const projectId = document.getElementById("issue-project").value;
|
|
|
|
sessionStorage.setItem(`summary-${projectId}`, summary);
|
|
|
|
}
|
|
|
|
htmx.onLoad(function(){
|
|
|
|
const summaryTextarea = document.getElementById("issue-summary");
|
|
|
|
const summary = summaryTextarea.value;
|
|
|
|
if (summary !== "") return;
|
|
|
|
const projectId = document.getElementById("issue-project").value;
|
|
|
|
if (!projectId) return;
|
|
|
|
const savedSummary = sessionStorage.getItem(`summary-${projectId}`);
|
|
|
|
if (!savedSummary) return;
|
|
|
|
summaryTextarea.value = savedSummary;
|
|
|
|
})
|
|
|
|
</script>
|
2025-02-22 09:42:15 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func projectsToOptions(projects []*model.Project) []string {
|
|
|
|
options := make([]string, 0, len(projects)*2)
|
|
|
|
for _, p := range projects {
|
|
|
|
options = append(options, p.Label, p.ID)
|
|
|
|
}
|
|
|
|
return options
|
|
|
|
}
|