在过去的 2 周里,我一直在学习 backbone 和相关工具以及编写应用程序。我遇到了设计问题,想知道有哪些可用的解决方案,以及 Backbone 专家是否认为这是一个问题。
问题:我最终不得不将我所有的 View 依赖项放在我的 router.js 中,并且无法确定它们是否是解决该问题的方法。下面是我的 router.js 中的代码:
// router.js
define([
'jquery',
'underscore',
'backbone',
'text',
'views/landing',
'views/dashboard',
],
function($, _, Backbone, t,LandingView,DashboardView){
var AppRouter = Backbone.Router.extend({
routes: {
// Define some URL routes
'': 'showLanding',
'projects': 'showProjects',
// Default
'*actions': 'defaultAction'
},
navigate_to: function(model){
alert("navigate_to");
},
showProjects: function() {},
showLanding: function() {},
});
var initialize = function() {
var app_router = new AppRouter;
Backbone.View.prototype.event_aggregator = _.extend({}, Backbone.Events);
// Extend the View class to include a navigation method goTo
Backbone.View.prototype.goTo = function (loc) {
app_router.navigate(loc, true);
};
app_router.on('route:showLanding', function(){
var landing = new LandingView();
});
app_router.on('route:showProjects', function(){
var dashboard=new DashboardView();
});
app_router.on('defaultAction', function(actions){
alert("No routes");
// We have no matching route, lets just log what the URL was
console.log('No route:', actions);
});
Backbone.history.start({pushState: true});
};
return {
initialize: initialize
};
});
router.js 包括 LandingView 和 DashboardView View ,它们依次获取相应的模板。初始路由加载具有登录模板的 LandingView。登录后,它调用 router.js 的 goTo 方法来生成 DashboardView()。虽然这行得通,但我觉得它有点难看。但是我不知道如何从 LandingView 生成一个新的 DashboardView,而不直接从 LandingView() 内部或路由器引用 DashboardView()。
如果我继续通过 router.js 执行此操作,我最终将直接或间接地从路由器中提取我所有的 View js 文件。听起来有点难听!
我查看了 Derick Baileys 的事件聚合器模式,但面临这样的问题:如果 DashboardView 的实例尚不存在,DashboardView 如何订阅 LandingView 生成的事件?必须有人创建并初始化它才能订阅事件聚合器,对吗?如果那个人是路由器,我是否需要在路由器中预先实例化所有 View ?这没有意义。
最佳答案
我通过仅在首次访问路由时导入 View 来解决此问题:
define(['backbone'], function(Backbone) {
var AppRouter = Backbone.Router.extend({
routes: {
'': 'home',
'users': 'users'
},
home: function() {
requirejs(["views/home/mainview"], function(HomeView) {
//..initialize and render view
});
},
users: function() {
requirejs(["views/users/mainview"], function(UsersView) {
//..initialize and render view
});
}
});
return AppRouter;
});
它没有解决必须最终将所有 View 导入路由器的问题,但是惰性 requirejs 调用不会强制加载和评估所有脚本和模板。
事实是某人必须在某处导入模块。路由器是一个明智的位置,因为它通常是用户导航到特定页面( View )时命中的第一段代码。如果您觉得一个路由器负责太多,您应该考虑将您的路由器拆分为多个路由器,每个路由器负责应用程序的不同“部分”。一个很好的类比是在典型的 MVC 场景中考虑 Controller。
userrouter.js 处理所有与用户相关的 View (“users/”下的路由):
define(['backbone'], function(Backbone) {
var UserRouter = Backbone.Router.extend({
routes: {
'users', 'allUsers',
'users/:id', 'userById'
},
allUsers: function() {
requirejs(["views/users/listview"], function(UserListView) {
//..initialize and render view
});
},
userById: function(id) {
requirejs(["views/users/detailview"], function(UserDetailView) {
//..initialize and render view
});
}
});
return UserRouter;
});
postrouter.js 处理所有与帖子相关的 View (“posts/”下的路由):
define(['backbone'], function(Backbone) {
var PostRouter = Backbone.Router.extend({
routes: {
'posts', 'allPosts',
'posts/:id', 'postById'
},
allPosts: function() {
requirejs(["views/posts/listview"], function(PostListView) {
//..initialize and render view
});
},
postById: function(id) {
requirejs(["views/posts/detailview"], function(PostDetailView) {
//..initialize and render view
});
}
});
return PostRouter;
});
approuter.js 是主路由器,它在应用程序启动时启动并初始化所有其他路由。
define(['backbone', 'routers/userrouter', 'routers/postrouter'],
function(Backbone, UserRouter, PostRouter) {
var AppRouter = Backbone.Router.extend({
routes: {
'', 'home',
},
initialize: function() {
//create all other routers
this._subRouters = {
'users' : new UserRouter(),
'posts' : new PostRouter()
};
},
start: function() {
Backbone.history.start();
},
home: function() {
requirejs(["views/home/mainview"], function(HomeView) {
//..initialize and render view
});
}
});
return UserRouter;
});
最后,您的应用程序的 main.js,它会启动应用程序路由器:
new AppRouter().start();
通过这种方式,您可以保持每个单独的路由器精简,并避免在实际需要之前解决依赖关系树。
旁注:如果您使用嵌套的 requirejs 调用并且您正在使用 r.js 进行构建,请记住设置构建选项findNestedDependencies:true,因此延迟加载的模块会包含在构建中。
编辑:这是一个gist that explains lazy vs. immediate module loading in RequireJS .
关于javascript - Backbone.js "fat router"设计难题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14185909/
我正在尝试测试是否存在表单。我是Rails新手。我的new.html.erb_spec.rb文件的内容是:require'spec_helper'describe"messages/new.html.erb"doit"shouldrendertheform"dorender'/messages/new.html.erb'reponse.shouldhave_form_putting_to(@message)with_submit_buttonendendView本身,new.html.erb,有代码:当我运行rspec时,它失败了:1)messages/new.html.erbshou
我在从html页面生成PDF时遇到问题。我正在使用PDFkit。在安装它的过程中,我注意到我需要wkhtmltopdf。所以我也安装了它。我做了PDFkit的文档所说的一切......现在我在尝试加载PDF时遇到了这个错误。这里是错误:commandfailed:"/usr/local/bin/wkhtmltopdf""--margin-right""0.75in""--page-size""Letter""--margin-top""0.75in""--margin-bottom""0.75in""--encoding""UTF-8""--margin-left""0.75in""-
我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co
为了将Cucumber用于命令行脚本,我按照提供的说明安装了arubagem。它在我的Gemfile中,我可以验证是否安装了正确的版本并且我已经包含了require'aruba/cucumber'在'features/env.rb'中为了确保它能正常工作,我写了以下场景:@announceScenario:Testingcucumber/arubaGivenablankslateThentheoutputfrom"ls-la"shouldcontain"drw"假设事情应该失败。它确实失败了,但失败的原因是错误的:@announceScenario:Testingcucumber/ar
我将应用程序升级到Rails4,一切正常。我可以登录并转到我的编辑页面。也更新了观点。使用标准View时,用户会更新。但是当我添加例如字段:name时,它不会在表单中更新。使用devise3.1.1和gem'protected_attributes'我需要在设备或数据库上运行某种更新命令吗?我也搜索过这个地方,找到了许多不同的解决方案,但没有一个会更新我的用户字段。我没有添加任何自定义字段。 最佳答案 如果您想允许额外的参数,您可以在ApplicationController中使用beforefilter,因为Rails4将参数
我遵循MichaelHartl的“RubyonRails教程:学习Web开发”,并创建了检查用户名和电子邮件长度有效性的测试(名称最多50个字符,电子邮件最多255个字符)。test/helpers/application_helper_test.rb的内容是:require'test_helper'classApplicationHelperTest在运行bundleexecraketest时,所有测试都通过了,但我看到以下消息在最后被标记为错误:ERROR["test_full_title_helper",ApplicationHelperTest,1.820016791]test
我正在尝试从Postgresql表(table1)中获取数据,该表由另一个相关表(property)的字段(table2)过滤。在纯SQL中,我会这样编写查询:SELECT*FROMtable1JOINtable2USING(table2_id)WHEREtable2.propertyLIKE'query%'这工作正常:scope:my_scope,->(query){includes(:table2).where("table2.property":query)}但我真正需要的是使用LIKE运算符进行过滤,而不是严格相等。然而,这是行不通的:scope:my_scope,->(que
我正在尝试编写一个将文件上传到AWS并公开该文件的Ruby脚本。我做了以下事情:s3=Aws::S3::Resource.new(credentials:Aws::Credentials.new(KEY,SECRET),region:'us-west-2')obj=s3.bucket('stg-db').object('key')obj.upload_file(filename)这似乎工作正常,除了该文件不是公开可用的,而且我无法获得它的公共(public)URL。但是当我登录到S3时,我可以正常查看我的文件。为了使其公开可用,我将最后一行更改为obj.upload_file(file
当我尝试安装Ruby时遇到此错误。我试过查看this和this但无济于事➜~brewinstallrubyWarning:YouareusingOSX10.12.Wedonotprovidesupportforthispre-releaseversion.Youmayencounterbuildfailuresorotherbreakages.Pleasecreatepull-requestsinsteadoffilingissues.==>Installingdependenciesforruby:readline,libyaml,makedepend==>Installingrub
我在新的Debian6VirtualBoxVM上安装RVM时遇到问题。我已经安装了所有需要的包并使用下载了安装脚本(curl-shttps://rvm.beginrescueend.com/install/rvm)>rvm,但以单个用户身份运行时bashrvm我收到以下错误消息:ERROR:Unabletocheckoutbranch.安装在这里停止,并且(据我所知)没有安装RVM的任何文件。如果我以root身份运行脚本(对于多用户安装),我会收到另一条消息:Successfullycheckedoutbranch''安装程序继续并指示成功,但未添加.rvm目录,甚至在修改我的.bas