https://github.com/bonfy/go-mega/blob/master/04-web-form.md
从这网站学的
随着我们项目的扩大,代码量会愈来愈多,我们需要建立这样的数据结构来使整个项目看起来没有那么臃肿
package model - 负责数据建模
vm - View Model,定义各种结构体来表示不同的视图模型
controller - http路由,就是对各个uri进行处理的函数,引用model包中的数据模型来执行业务逻辑,并使用vm包中的视图模型来准备数据以供渲染页面使用
g.go 负责存放该package的全局变量以及init函数
我直接开了一个新的文件夹来存放着一章的教程
下面我们来看一下这个login.html
{{define "content"}}<h1>Login</h1><form action="/login" method="post" name="login"><p><input type="text" name="username" value="" placeholder="Username or Email"></p><p><input type="password" name="password" value="" placeholder="Password"></p><p><input type="submit" name="submit" value="Login"></p></form>
{{end}}
我们看这段代码的表单第一行
<form action="/login" method="post" name="login">
这里的action指定我们需要到这个目标URL
post是指表单提交的方法,post一般是指比较敏感的信息
name = "login"
是指表单的名字是login,这样可以在JavaScript中通过这个名称来引用这个表单
JavaScript,css大家可以去菜鸟上了解一下基本的用途
<p><input type="text" name="username" value="" placeholder="Username or Email"></p>
主要解释一下placeholder
,这个是指用户可以在其中输入用户名或电子邮件地址
value=""
表示输入字段的初始值为空
之前看过那个web基础,感觉这个类似于在做个项目,比go web基础好太多了,当然可能是我变强了,笑哭
接收表单数据
就是我们会发现我们在点login的时候,客户端的响应是没有的,只是说账号和密码都清空了,像qq它给的回应都是一个界面了对吧,我们这里只需要有个回应就好了,所以我们要对loginHandler进行修改
...func loginHandler(w http.ResponseWriter, r *http.Request) {tpName := "login.html"vop := vm.IndexViewModelOp{}v := vop.GetVM()if r.Method == http.MethodGet {templates[tpName].Execute(w, &v)/*templates.Execute通常都是用模板来渲染画面的*/}if r.Method == http.MethodPost {r.ParseForm()username := r.Form.Get("username")password := r.Form.Get("password")fmt.Fprintf(w, "Username:%s Password:%s", username, password)}
}
表单后端验证
一般验证用户名密码正确性检查只能在后端验证
我们在LoginViewModel里面添加一个Err字段,用于输出检查的错误返回
package vmtype LoginViewModel struct {BaseViewModelErrs []string
}type LoginViewModelOp struct {
}func (LoginViewModelOp) GetVM() LoginViewModel {v := LoginViewModel{}v.SetTitle("Login")return v
}
func (v *LoginViewModel) AddError(errs ...string) {v.Errs = append(v.Errs, errs...)
}
我们主要来解析一下新增的login.html代码
这里就不按照
...{{if .Errs}} <!--判断属性Errs是否为空,如果不为空--><ul> <!--ul表示无序列表,无序列表中通常以项目符号表示-->{{range .Errs}}<li>{{.}}</li><!--每个列表项使用<li>标签来定义-->{{end}}</ul>{{end}}
...
我们在controller中的home.go中新增check方法和修改loginHandler方法
...func check(username, password string) bool {if username == "bonfy" && password == "abc123" {return true}return false
}func loginHandler(w http.ResponseWriter, r *http.Request) {tpName := "login.html"vop := vm.LoginViewModelOp{}v := vop.GetVM()//因为客户端在访问的时候其实是get方法//在提交信息的时候是POST方法if r.Method == http.MethodGet {templates[tpName].Execute(w, &v)}if r.Method == http.MethodPost {r.ParseForm()username := r.Form.Get("username")password := r.Form.Get("password")if len(username) < 3 {v.AddError("username must longer than 3")}if len(password) < 6 {v.AddError("password must longer than 6")}if !check(username, password) {v.AddError("username password not correct,please input again")}if len(v.Errs) > 0 { //重新执行这个模板,然后显示这个错误templates[tpName].Execute(w, &v)} else {http.Redirect(w, r, "/", http.StatusSeeOther)//成功的话就会转到一个新的网站}fmt.Fprintf(w, "Username:%s Password:%s", username, password)}
}
这里提醒一下大家,那个html文件改变较大,如果出错了,可以重点看一下那个文件
{{define "content"}}
<h1>Login</h1>
<form action="/login" method="post" name="login"><p><input type="text" name="username" value="" placeholder="Username or Email"></p><p><input type="password" name="password" value="" placeholder="Password"></p><p><input type="submit" name="submit" value="Login"></p>
</form>
{{if .Errs}}
<ul>{{range .Errs}}<li>{{.}}</li>{{end}}
</ul>
{{end}}
{{end}}
等会要调试一下,看一下怎么走的,总结一下
就是注册一个路由处理器,里面有路由处理器,对每个uri路径进行处理