debugged form loading and got part of the Login code in

This commit is contained in:
2025-09-27 17:39:32 -06:00
parent 5082e2bbc2
commit 03f1d9f717
6 changed files with 99 additions and 26 deletions
+32 -4
View File
@@ -38,12 +38,41 @@ func LoginForm(ctxt ui.AmContext) (string, any, error) {
dlg, err := ui.AmLoadDialog("login") dlg, err := ui.AmLoadDialog("login")
if err == nil { if err == nil {
dlg.Field("tgt").Value = target dlg.Field("tgt").Value = target
ctxt.VarMap().Set("amsterdam_pageTitle", "Log In")
return dlg.Render(ctxt) return dlg.Render(ctxt)
} }
return ui.ErrorPage(ctxt, err) return ui.ErrorPage(ctxt, err)
} }
/* Login handles logging in to Amsterdam.
* Parameters:
* ctxt - The AmContext for the request.
* Returns:
* Command string dictating what to be rendered.
* Data as a parameter for the command string.
* Standard Go error status.
*/
func Login(ctxt ui.AmContext) (string, any, error) {
dlg, err := ui.AmLoadDialog("login")
if err == nil {
dlg.LoadFromForm(ctxt)
target := dlg.Field("tgt").Value
if target == "" {
target = "/"
}
action := dlg.WhichButton(ctxt)
if action == "cancel" {
return "redirect", target, nil
}
if action == "remind" {
// TODO: send password reminder
return dlg.Render(ctxt)
}
}
return ui.ErrorPage(ctxt, err)
}
/* NewAccountUserAgreement renders the Amsterdam user agreement for new accounts. /* NewAccountUserAgreement renders the Amsterdam user agreement for new accounts.
* Parameters: * Parameters:
* ctxt - The AmContext for the request. * ctxt - The AmContext for the request.
@@ -61,7 +90,7 @@ func NewAccountUserAgreement(ctxt ui.AmContext) (string, any, error) {
// If user is already logged in, this is an error. // If user is already logged in, this is an error.
if !ctxt.CurrentUser().IsAnon { if !ctxt.CurrentUser().IsAnon {
return ui.ErrorPage(ctxt, fmt.Errorf("You cannot create a bew account while logged in on an existing one. You must log out first.")) return ui.ErrorPage(ctxt, fmt.Errorf("You cannot create a new account while logged in on an existing one. You must log out first."))
} }
ctxt.VarMap().Set("target", target) ctxt.VarMap().Set("target", target)
@@ -86,14 +115,13 @@ func NewAccountForm(ctxt ui.AmContext) (string, any, error) {
// If user is already logged in, this is an error. // If user is already logged in, this is an error.
if !ctxt.CurrentUser().IsAnon { if !ctxt.CurrentUser().IsAnon {
return ui.ErrorPage(ctxt, fmt.Errorf("You cannot create a bew account while logged in on an existing one. You must log out first.")) return ui.ErrorPage(ctxt, fmt.Errorf("You cannot create a new account while logged in on an existing one. You must log out first."))
} }
dlg, err := ui.AmLoadDialog("newaccount") dlg, err := ui.AmLoadDialog("newaccount")
if err == nil { if err == nil {
dlg.Field("tgt").Value = target dlg.Field("tgt").Value = target
dlg.Field("country").Value = "XX" dlg.Field("country").Value = "XX"
ctxt.VarMap().Set("amsterdam_pageTitle", "Create New Account")
return dlg.Render(ctxt) return dlg.Render(ctxt)
} }
return ui.ErrorPage(ctxt, err) return ui.ErrorPage(ctxt, err)
+1
View File
@@ -43,6 +43,7 @@ func setupEcho() *echo.Echo {
e.GET("/", ui.AmWrap(TopPage)) e.GET("/", ui.AmWrap(TopPage))
e.GET("/about", ui.AmWrap(AboutPage)) e.GET("/about", ui.AmWrap(AboutPage))
e.GET("/login", ui.AmWrap(LoginForm)) e.GET("/login", ui.AmWrap(LoginForm))
e.POST("/login", ui.AmWrap(Login))
e.GET("/newacct", ui.AmWrap(NewAccountUserAgreement)) e.GET("/newacct", ui.AmWrap(NewAccountUserAgreement))
e.GET("/newacct2", ui.AmWrap(NewAccountForm)) e.GET("/newacct2", ui.AmWrap(NewAccountForm))
+16 -1
View File
@@ -28,6 +28,7 @@ type AmContext interface {
CurrentUser() *database.User CurrentUser() *database.User
FormField(string) string FormField(string) string
FormFieldInt(string) (int, error) FormFieldInt(string) (int, error)
FormFieldIsSet(string) bool
RC() int RC() int
OutputType() string OutputType() string
Parameter(string) string Parameter(string) string
@@ -71,7 +72,7 @@ func (c *amContext) FormField(name string) string {
return c.echoContext.FormValue(name) return c.echoContext.FormValue(name)
} }
/* FormField returns the value of a form field from the request, as an integer. /* FormFieldInt returns the value of a form field from the request, as an integer.
* Parameters: * Parameters:
* name - The name of the field to retrieve. * name - The name of the field to retrieve.
* Returns: * Returns:
@@ -82,6 +83,20 @@ func (c *amContext) FormFieldInt(name string) (int, error) {
return strconv.Atoi(c.echoContext.FormValue(name)) return strconv.Atoi(c.echoContext.FormValue(name))
} }
/* FormFieldIsSet returns true if a given form field is set.
* Parameters:
* name - The name of the field to test.
* Returns:
* true if the field is set, false if not.
*/
func (c *amContext) FormFieldIsSet(name string) bool {
req := c.echoContext.Request()
if req.Form == nil {
_ = req.FormValue(name) // force form to be loaded
}
return req.Form.Has(name)
}
// RC returns the HTTP result code for the current operation. // RC returns the HTTP result code for the current operation.
func (c *amContext) RC() int { func (c *amContext) RC() int {
return c.httprc return c.httprc
+48 -19
View File
@@ -61,12 +61,12 @@ func AmLoadDialog(name string) (*Dialog, error) {
if d.MenuSelector == "" { if d.MenuSelector == "" {
d.MenuSelector = "nochange" d.MenuSelector = "nochange"
} }
for _, fld := range d.Fields { for i, fld := range d.Fields {
if fld.Type == "button" && fld.Param == "" { if fld.Type == "button" && fld.Param == "" {
fld.Param = "blue" d.Fields[i].Param = "blue"
} }
if fld.Type == "date" && fld.Param == "" { if fld.Type == "date" && fld.Param == "" {
fld.Param = "year:-100" d.Fields[i].Param = "year:-100"
} }
} }
return &d, nil return &d, nil
@@ -94,8 +94,8 @@ func (fld *DialogItem) DateValues() []int {
* Pointer to the field, or nil. * Pointer to the field, or nil.
*/ */
func (d *Dialog) Field(name string) *DialogItem { func (d *Dialog) Field(name string) *DialogItem {
for i := 0; i < len(d.Fields); i++ { for i, f := range d.Fields {
if d.Fields[i].Name == name { if f.Name == name {
return &(d.Fields[i]) return &(d.Fields[i])
} }
} }
@@ -120,6 +120,7 @@ func (d *Dialog) Render(ctxt AmContext) (string, any, error) {
} }
ctxt.VarMap().Set("amsterdam_required", required) ctxt.VarMap().Set("amsterdam_required", required)
ctxt.VarMap().Set("amsterdam_dialog", d) ctxt.VarMap().Set("amsterdam_dialog", d)
ctxt.VarMap().Set("amsterdam_pageTitle", d.Title)
return "framed_template", "dialog.jet", nil return "framed_template", "dialog.jet", nil
} }
@@ -128,48 +129,76 @@ func (d *Dialog) Render(ctxt AmContext) (string, any, error) {
* ctxt - The AmContext for this request. * ctxt - The AmContext for this request.
*/ */
func (d *Dialog) LoadFromForm(ctxt AmContext) { func (d *Dialog) LoadFromForm(ctxt AmContext) {
for _, fld := range d.Fields { for i, fld := range d.Fields {
d.Fields[i].Value = ""
if fld.Type == "header" || fld.Type == "button" {
continue
}
if fld.Type == "date" { if fld.Type == "date" {
fld.Value = "" d.Fields[i].Value = ""
dvals := make([]int, 3) dvals := make([]int, 3)
var err error var err error
dvals[0], err = ctxt.FormFieldInt(fmt.Sprintf("%s_month", fld.Name)) dvals[0], err = ctxt.FormFieldInt(fmt.Sprintf("%s_month", fld.Name))
if err != nil { if err != nil {
dvals[0] = -1 dvals[0] = -1
fld.Value = fmt.Sprintf("!undefined month %s: %v", fld.Name, err) d.Fields[i].Value = fmt.Sprintf("!undefined month %s: %v", fld.Name, err)
} }
dvals[1], err = ctxt.FormFieldInt(fmt.Sprintf("%s_day", fld.Name)) dvals[1], err = ctxt.FormFieldInt(fmt.Sprintf("%s_day", fld.Name))
if err != nil { if err != nil {
dvals[1] = -1 dvals[1] = -1
if fld.Value == "" { if d.Fields[i].Value == "" {
fld.Value = fmt.Sprintf("!undefined day %s: %v", fld.Name, err) d.Fields[i].Value = fmt.Sprintf("!undefined day %s: %v", fld.Name, err)
} }
} }
dvals[2], err = ctxt.FormFieldInt(fmt.Sprintf("%s_year", fld.Name)) dvals[2], err = ctxt.FormFieldInt(fmt.Sprintf("%s_year", fld.Name))
if err != nil { if err != nil {
dvals[2] = -1 dvals[2] = -1
if fld.Value == "" { if d.Fields[i].Value == "" {
fld.Value = fmt.Sprintf("!undefined year %s: %v", fld.Name, err) d.Fields[i].Value = fmt.Sprintf("!undefined year %s: %v", fld.Name, err)
} }
} }
if dvals[0] > 0 && dvals[1] > 0 && dvals[2] > 0 { if dvals[0] > 0 && dvals[1] > 0 && dvals[2] > 0 {
fld.Value = fmt.Sprintf("%04d%02d%02d", dvals[2], dvals[0], dvals[1]) d.Fields[i].Value = fmt.Sprintf("%04d%02d%02d", dvals[2], dvals[0], dvals[1])
} else if fld.Value == "" && fld.Required { } else if d.Fields[i].Value == "" && fld.Required {
if dvals[0] <= 0 { if dvals[0] <= 0 {
fld.Value = fmt.Sprintf("!month not set %s", fld.Name) d.Fields[i].Value = fmt.Sprintf("!month not set %s", fld.Name)
} else if dvals[1] <= 0 { } else if dvals[1] <= 0 {
fld.Value = fmt.Sprintf("!day not set %s", fld.Name) d.Fields[i].Value = fmt.Sprintf("!day not set %s", fld.Name)
} else if dvals[2] <= 0 { } else if dvals[2] <= 0 {
fld.Value = fmt.Sprintf("!year not set %s", fld.Name) d.Fields[i].Value = fmt.Sprintf("!year not set %s", fld.Name)
} }
} }
fld.AuxData = dvals d.Fields[i].AuxData = dvals
} else { } else {
fld.Value = ctxt.FormField(fld.Name) d.Fields[i].Value = ctxt.FormField(fld.Name)
} }
} }
} }
// Values returns all a dialog's values as a map.
func (d *Dialog) Values() map[string]string {
rc := map[string]string{}
for _, fld := range d.Fields {
rc[fld.Name] = fld.Value
}
return rc
}
/* WhichButton returns an indication of which button on the dialog was clicked.
* Parameters:
* ctxt - The AmContext associated with the request.
* Returns:
* The name of the button field on this dialog that was clicked. If none were, the empty string is returned.
*/
func (d *Dialog) WhichButton(ctxt AmContext) string {
for _, fld := range d.Fields {
if fld.Type == "button" && ctxt.FormFieldIsSet(fld.Name) {
return fld.Name
}
}
return ""
}
// validatorFunc is a function that validates the contents of a dialog item. // validatorFunc is a function that validates the contents of a dialog item.
type validatorFunc func(*DialogItem) error type validatorFunc func(*DialogItem) error
+1 -1
View File
@@ -10,7 +10,7 @@ name: "login"
formName: "loginform" formName: "loginform"
menuSelector: "top" menuSelector: "top"
title: "Log In" title: "Log In"
action: "/TODO/login" action: "/login"
instructions: > instructions: >
Forgot your password? Enter your user name and click the <b>Reminder</b> button to receive Forgot your password? Enter your user name and click the <b>Reminder</b> button to receive
a password reminder via E-mail. a password reminder via E-mail.
+1 -1
View File
@@ -9,7 +9,7 @@
name: "newaccount" name: "newaccount"
formName: "createform" formName: "createform"
menuSelector: "top" menuSelector: "top"
title: "Create Account" title: "Create New Account"
action: "/TODO/newacct2" action: "/TODO/newacct2"
instructions: > instructions: >
To create a new account, please enter your information below. To create a new account, please enter your information below.