Grails 3:通过 POST 绑定多个命令对象
Grails 3: Binding multiple command objects via POST
我有一个带有方法的控制器,我也想绑定多个命令对象。当我通过 GET 调用方法时,效果很好,两个对象都被绑定了。问题是如果我通过 POST 调用方法,只有第一个命令对象被绑定而第二个被完全忽略。
简单示例:
def register(MembershipCommand cmd1, RegisterCommand cmd2) {
println(cmd1.email);
println(cmd2.pass);
respond([:]);
}
如果我调用 /register?email=test&pass=test
然后 cmd1
和 cmd2
得到填充
如果我用 POST 数据 {email:test,pass:test}
调用 /register
,cmd1
会被填充,而 cmd2.pass
是 null
。
有没有办法使用 POST 使此数据绑定工作?我不能真正使用 GET,因为涉及文件上传,而且表格相当大。
我知道另一种选择是将每个对象的方法分成 2 个和 1 个,然后让我的表单分别提交给每个对象,但我想尽可能避免这种情况。
有什么想法吗?
使 POST 正常工作的解决方案是将我的表单数据重组为对象样式格式。
所以不是 {email:test,pass:test}
我会 {cmd1:{email:test}, cmd2:{pass:test}}
我创建了一个最小的工作项目来测试你的想法。它就像一个魅力。以下是片段。
RegisterCmd.groovy
class RegisterCmd {
String email
}
edit.gsp
<g:form resource="${this.player}" method="POST" action="customisedUpdate">
<g:hiddenField name="version" value="${this.player?.version}" />
<fieldset class="form">
<f:field bean="player" property="name" />
<f:field bean="player" property="game" />
<f:field bean="player" property="region" />
<label>Email</label><g:field type="text" name="email"/>
</fieldset>
<fieldset class="buttons">
<input class="save" type="submit" value="${message(code: 'default.button.update.label', default: 'Update')}" />
</fieldset>
</g:form>
PlayerController.groovy
@Transactional
def customisedUpdate(Player player, RegisterCmd registerCmd) {
println "Calling save ${player.dump()}"
println "RegisterCmd: ${registerCmd.dump()}"
//end::save[]
//tag::save-handleErrors[]
if (player == null) {
render status: HttpStatus.NOT_FOUND
return
}
if (player.hasErrors()) {
respond player.errors, view: 'create'
return
}
//end::save-handleErrors[]
player.save flush: true
request.withFormat {
form multipartForm { redirect player }
'*' { respond player, status: HttpStatus.CREATED }
}
//tag::save[]
}
输出如下:
Calling save <com.itersdesktop.javatechs.grails.Player@1c25113d name=Alexis Barnett game=Pandemic region=EAST wins=96 losses=30 id=1 version=4 org_grails_datastore_mapping_dirty_checking_DirtyCheckable_
_$changedProperties=[name:HUE THI MY NGO] org_grails_datastore_gorm_GormValidateable__errors=org.grails.datastore.mapping.validation.ValidationErrors: 0 errors org_grails_datastore_gorm_GormValidateable
__skipValidate=false>
RegisterCmd: <com.itersdesktop.javatechs.grails.RegisterCmd@7ee6bb8c email=alexis.barnett@gmail.com grails_validation_Validateable__beforeValidateHelper=org.grails.datastore.gorm.support.BeforeValidateH
elper@3409999e grails_validation_Validateable__errors=grails.validation.ValidationErrors: 0 errors>
如果您对该项目感兴趣,请咨询https://bitbucket.org/itersdesktop/command-objects/src/master/
对于 Grails 3。3.X 似乎不支持具有多个命令对象。
我有一个带有方法的控制器,我也想绑定多个命令对象。当我通过 GET 调用方法时,效果很好,两个对象都被绑定了。问题是如果我通过 POST 调用方法,只有第一个命令对象被绑定而第二个被完全忽略。
简单示例:
def register(MembershipCommand cmd1, RegisterCommand cmd2) {
println(cmd1.email);
println(cmd2.pass);
respond([:]);
}
如果我调用 /register?email=test&pass=test
然后 cmd1
和 cmd2
得到填充
如果我用 POST 数据 {email:test,pass:test}
调用 /register
,cmd1
会被填充,而 cmd2.pass
是 null
。
有没有办法使用 POST 使此数据绑定工作?我不能真正使用 GET,因为涉及文件上传,而且表格相当大。
我知道另一种选择是将每个对象的方法分成 2 个和 1 个,然后让我的表单分别提交给每个对象,但我想尽可能避免这种情况。
有什么想法吗?
使 POST 正常工作的解决方案是将我的表单数据重组为对象样式格式。
所以不是 {email:test,pass:test}
我会 {cmd1:{email:test}, cmd2:{pass:test}}
我创建了一个最小的工作项目来测试你的想法。它就像一个魅力。以下是片段。
RegisterCmd.groovy
class RegisterCmd {
String email
}
edit.gsp
<g:form resource="${this.player}" method="POST" action="customisedUpdate">
<g:hiddenField name="version" value="${this.player?.version}" />
<fieldset class="form">
<f:field bean="player" property="name" />
<f:field bean="player" property="game" />
<f:field bean="player" property="region" />
<label>Email</label><g:field type="text" name="email"/>
</fieldset>
<fieldset class="buttons">
<input class="save" type="submit" value="${message(code: 'default.button.update.label', default: 'Update')}" />
</fieldset>
</g:form>
PlayerController.groovy
@Transactional
def customisedUpdate(Player player, RegisterCmd registerCmd) {
println "Calling save ${player.dump()}"
println "RegisterCmd: ${registerCmd.dump()}"
//end::save[]
//tag::save-handleErrors[]
if (player == null) {
render status: HttpStatus.NOT_FOUND
return
}
if (player.hasErrors()) {
respond player.errors, view: 'create'
return
}
//end::save-handleErrors[]
player.save flush: true
request.withFormat {
form multipartForm { redirect player }
'*' { respond player, status: HttpStatus.CREATED }
}
//tag::save[]
}
输出如下:
Calling save <com.itersdesktop.javatechs.grails.Player@1c25113d name=Alexis Barnett game=Pandemic region=EAST wins=96 losses=30 id=1 version=4 org_grails_datastore_mapping_dirty_checking_DirtyCheckable_
_$changedProperties=[name:HUE THI MY NGO] org_grails_datastore_gorm_GormValidateable__errors=org.grails.datastore.mapping.validation.ValidationErrors: 0 errors org_grails_datastore_gorm_GormValidateable
__skipValidate=false>
RegisterCmd: <com.itersdesktop.javatechs.grails.RegisterCmd@7ee6bb8c email=alexis.barnett@gmail.com grails_validation_Validateable__beforeValidateHelper=org.grails.datastore.gorm.support.BeforeValidateH
elper@3409999e grails_validation_Validateable__errors=grails.validation.ValidationErrors: 0 errors>
如果您对该项目感兴趣,请咨询https://bitbucket.org/itersdesktop/command-objects/src/master/
对于 Grails 3。3.X 似乎不支持具有多个命令对象。