经过一段令人沮丧的试错期后,我终于设法让我的代码工作了!它不漂亮,但它很实用,现在我可以开始考虑改进它了。希望其他人会发现此信息有用...
让我走上正轨的一件事是了解需要更改的不仅是主干视图代码(具有新项目形式的代码),还包括相应的 Rails 模型。
对于 rails 部分(Rails 3.2.2),我确保以下模型文件具有必要的信息:
项目.rb
class Project < ActiveRecord::Base
has_and_belongs_to_many :users
#attr_accessible :name, :description, :users_attributes
#has_many :projects_users, foreign_key: "project_id", dependent: :destroy
#has_many :users, through :projects_users
#accepts_nested_attributes_for :users
end
用户.rb
class User < ActiveRecord::Base
has_and_belongs_to_many :projects
end
项目_用户.rb
class ProjectsUsers < ActiveRecord::Base
belongs_to :project
belongs_to :user
end
我在许多不同的地方读到 has_and_belongs_to_many 并不是在 Rails 中设置多对多关系的最佳方式。尽管如此,我无法使用 has_many 定义获得相同的功能 - project.rb 中的注释部分是我尝试这种不同方法的方式。文件 user.rb 有一些相应的代码,为简单起见,我将其删除。
现在,我需要在主干表单视图中完成的是发送一个带有 Rails projects_controller.rb 可以识别的 JSON 对象的 POST 请求。最初,我尝试了几个 POST 请求但没有成功(并且没有错误来指导我)。但是后来,我记得以前为团队实现了一个表单,可以将用户添加到特定团队中(HABTM Checkboxes - 此功能有一个 railscast)。看了这个例子之后,我意识到我的 POST 请求需要什么。这是我想在 rails 服务器日志文件中看到的内容:
Started POST "/projects" for 127.0.0.1 at 2012-06-27 00:35:22 +0000
Processing by ProjectsController#create as JSON
Parameters: {"project"=>{"description"=>"with some description", "user_ids"=>["101", "1", "99"], "name"=>"some new project"}}
实现上述要求的Backbone相关文件:
项目.js
App.Models.Project = Backbone.Model.extend({
urlRoot: '/projects',
// Default attributes for the project.
defaults: {
description: "",
user_ids: []
},
/* getters */
});
用户.js
App.Models.User = Backbone.Model.extend({
/* getters */
});
表单.js
App.Views.Projects.Common.Form = Backbone.View.extend({
...
events: {
"submit #new_project_form" : "formSubmit"
},
formSubmit: function(event) {
this.submitted($(event.target));
return false;
},
submitted: function(formElement) {
var newData = this.serializeFormData(formElement);
this.model = new App.Models.Project({
name : newData.name,
description : newData.description
});
this.saveFormData(newData);
return false;
},
serializeFormData: function(formElement) {
var fields = formElement.serializeArray();
var serializedData = {};
$.each(fields, function(index, field) {
serializedData[field.name] = field.value;
});
return serializedData;
},
// THE IMPORTANT PART FOR THE POST REQUEST
saveFormData: function(newData) {
// preserve reference to view for callbacks
var self = this;
var project = this.model;
project.set({
// a list of user ids associated with a project
"user_ids" : this.view_variables.user_ids
});
var project_object = ({
"project" : _.clone(project.attributes)
});
$.ajax({
type: 'POST',
url: '/projects',
data: project_object,
dataType: "json",
success: function() {
self.$el.hide();
self.addNewModelToCollection();
}
});
},
...
});
该代码有点冗长,并且包含一些特定于我的项目的代码。不过,相关部分在 saveFormData 函数中,其中使用了 jQuery ajax 函数。
如果您对导轨或 Backbone 部分有任何建议,请告诉我。我很乐意学习如何改进这个解决方案。