在html中,一个包含多部分数据的表单:
<form action="@routes.Files.upload" method="post" enctype="multipart/form-data">
<input type="hidden" name="groupId" value="1" />
<input type="hidden" name="tagId" value="2" />
<input type="file" name="file"/>
<input type="submit" value="upload it"/>
</form>
如何编写动作Files upload
?
我知道如何获取上传的文件:
request.body.file("file") map {
filepart => filepart.ref.moveTo(newFile);
}
如何获得提交的输入:
Form(tuple("groupId" -> text, "tagId" -> text)).bindFromRequest.fold(
errors => ...,
params => ....
)
但如何将它们组合在一起?
我找不到合适的file
类型可以在Form(tuple(...))
中使用,也不能在request.body
中获取输入值。
答案 0 :(得分:27)
这个答案适用于Java,但你应该能够很容易地将它适应Scala。
您需要做的是为表单中的所有字段定义模型,但文件除外。然后像往常一样使用文件上传API来检索文件。
例如,这就是我所做的:
表格(在upload.scala.html中):
@form(action = routes.UploadResourceController.doUpload(), 'enctype -> "multipart/form-data") {
@inputText(uploadForm("lang"))
@inputText(uploadForm("country"))
@inputFile(uploadForm("resourceFile"))
<p>
<input type="submit">
</p>
}
模型(models / UploadResource.java):
public class UploadResource {
@Required
public String lang;
@Required
public String country;
/* notice a field for the file is missing */
}
控制器(controllers / UploadResourceController.java):
public static Result doUpload() {
Form<UploadResource> filledForm = uploadForm.bindFromRequest();
if (filledForm.hasErrors()) {
return badRequest(views.html.upload.render(filledForm));
} else {
UploadResource resource = filledForm.get();
MultipartFormData body = request().body().asMultipartFormData();
FilePart resourceFile = body.getFile("resourceFile");
/* Check resourceFile for null, then extract the File object and process it */
}
}
我希望这会有所帮助。
答案 1 :(得分:13)
Scala中需要表单字段的示例:
<强>型号:强>
case class Specs (userid: String)
<强>控制器:强>
object Upload extends Controller {
val uploadForm = Form(
mapping(
"userid" -> nonEmptyText
)(Specs.apply)(Specs.unapply)
)
def upload = Action(parse.multipartFormData) { implicit request =>
val sp : Option[Specs] = uploadForm.bindFromRequest().fold (
errFrm => None,
spec => Some(spec)
)
request.body.file("file").map { f =>
sp.map { spec =>
val filePath = ... // incorporate userid
// XXX: file read to memory b4 writing to disk. bad for large files
f.ref.moveTo(new File(filePath), replace=true)
Ok("File uploaded")
}.getOrElse{
BadRequest("Form binding error.")
}
}.getOrElse {
BadRequest("File not attached.")
}
}
}
答案 2 :(得分:3)
另一个例子如何做到这一点:
<强>型号:强>
case class Specs(userId: String)
<强>控制器强>
def upload = Action(parse.multipartFormData) { implicit request =>
uploadForm.bindFromRequest().fold(
hasErrors => Ok(ourFormHTML(hasErrors),
specs => {
request.body.file("inputFileFieldName") match {
case Some(file) => {
import java.io.File
val filename = file.filename
val contetType = file.contentType
file.ref.moveTo(new File(Play.application().path().getAbsolutePath + file.filename))
Ok("congratz you did it")
}
case _ => Ok(ourHTML if we dont send file but want the form anyway)
}
}
)
别忘了给文件命名,因为你可能最终想知道出了什么问题。
答案 3 :(得分:2)
我使用angular和其他表单参数上传文件。我在下面创建了它,它起作用了。
角度功能
Upload.upload({
url: '/api/upload',
method:'POST',
data: {
"file": user.profilePic, //file object
"username": user.username
}
}).then(function (resp) {
//console.log('Success ' + resp.config.data.file.name + 'uploaded. Response: ' + resp.data);
}, function (resp) {
console.log('Error status: ' + resp.status);
}, function (evt) {
var progressPercentage = parseInt(100.0 * evt.loaded / evt.total);
//console.log('progress: ' + progressPercentage + '% ' + evt.config.data.file.name);
});
播放2.1控制器
/**
*
*Upload user profile
**/
public static Result upload() {
Logger.info("Uploading images##");
Http.MultipartFormData body = request().body().asMultipartFormData();
Http.MultipartFormData.FilePart profile = body.getFile("file");
if (profile != null) {
File file = profile.getFile();
//upload file to a directory
//todo
//get the username from form
Map<String,String[]> dataPart = request().body().asMultipartFormData().asFormUrlEncoded();
String username = dataPart.get("username")[0];
//save/update the details with ebean
return ok("File uploaded");
} else {
return status(400, "Missing file");
}
}