package component import ( "bytes" "context" "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" "github.com/pkg/errors" "github.com/yuin/goldmark" "log/slog" "strings" ) type IssuePageVModel struct { IssueURL string SummaryForm *form.Form IssueForm *form.Form IssueTips string Projects []*model.Project SelectedProjectID string } const summaryPlaceholder = ` Décrivez rapidement le sujet du problème rencontré ou de l'évolution souhaitée, ClearCase utilisera le modèle de demande présent dans le dépôt (ou un modèle par défaut) afin de générer une version mise en forme et complétée. Afin de fournir plus d'information de contexte au LLM, vous pouvez faire référence à d'autres tickets du dépôt via un ou plusieurs '#' et/ou des chemins vers des fichiers présents dans celui ci. ` const bodyPlaceholder = ` Une fois votre demande générée, vous pourrez l'éditer puis la créer directement en cliquant sur le bouton 'Créer' ci-dessous. ` func NewIssueSummaryForm() *form.Form { return form.New( form.NewField( "project", form.Attrs{}, form.NonEmpty("Ce champs ne doit pas être vide."), ), form.NewField( "summary", form.Attrs{ "type": "textarea", "rows": "20", "placeholder": strings.TrimSpace(summaryPlaceholder), }, form.NonEmpty("Ce champ ne doit pas être vide."), ), ) } func NewIssueForm() *form.Form { return form.New( form.NewField( "title", form.Attrs{ "type": "text", "placeholder": "Écrivez le résumé de votre demande et cliquez sur 'Générer' pour remplir automatiquement ces champs.", }, form.NonEmpty("Ce champ ne doit pas être vide."), ), form.NewField( "body", form.Attrs{ "type": "textarea", "rows": "20", "placeholder": strings.TrimSpace(bodyPlaceholder), }, form.NonEmpty("Ce champ ne doit pas être vide."), ), ) } templ IssuePage(vmodel IssuePageVModel) { @common.Page(common.WithTitle("Nouvelle demande")) {

ClearCase

if vmodel.IssueURL != "" {

Demande créée !

Votre demande a été créée et est disponible à l'adresse suivante: { vmodel.IssueURL }.
@templ.JSFuncCall("clearSummary", vmodel.SelectedProjectID) @templ.JSFuncCall("openIssue", vmodel.IssueURL) }

Résumé de la demande

@common.FormSelect( vmodel.SummaryForm, "issue-project", "project", "Projet", common.WithOptions(projectsToOptions(vmodel.Projects)...), common.WithAttrs( "hx-get", string(common.CurrentURL(ctx, common.WithoutValues("project", "*"))), "hx-target", "body", "hx-push-url", "true", ), ) @common.FormTextarea( vmodel.SummaryForm, "issue-summary", "summary", "Résumé", common.WithTextareaAttrs( "hx-on:change", "onSummaryChange(event)", ), )

Votre demande

@common.FormField(vmodel.IssueForm, "issue-title", "title", "Titre") @common.FormTextarea(vmodel.IssueForm, "issue-body", "body", "Corps")
if vmodel.IssueTips != "" { {{ html := markdownToHTML(ctx, vmodel.IssueTips) }} if html != "" {

Suggestions

@templ.Raw(html)
} }
} } func projectsToOptions(projects []*model.Project) []string { options := make([]string, 0, len(projects)*2) for _, p := range projects { options = append(options, p.Name, p.ID) } return options } func markdownToHTML(ctx context.Context, text string) string { var buff bytes.Buffer if err := goldmark.Convert([]byte(text), &buff); err != nil { slog.ErrorContext(ctx, "could not convert markdown to html", slog.Any("error", errors.WithStack(err))) return "" } return buff.String() }