5

在过去的两周里,我一直在学习主干和相关工具以及编写应用程序。我遇到了一个设计问题,想知道有哪些可用的解决方案,以及 Backbone 专家是否认为这是一个问题。

问题:我最终不得不将我所有的视图依赖项放在我的 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 包括LandingViewDashboardView视图,它们依次获取相应的模板。初始路由加载具有登录模板的 LandingView。登录后,它调用 router.js 的 goTo 方法生成一个 DashboardView()。虽然这可行,但我觉得它有点难看。但是我不知道如何从 LandingView 生成一个新的 DashboardView,而不直接从 LandingView() 内部或从路由器引用 DashboardView()。

如果我继续通过 router.js 执行此操作,我最终会直接或间接地从路由器中提取我的所有视图 js 文件。听起来有点丑!

我查看了 Derick Baileys 的事件聚合器模式,但面临的问题是,如果 DashboardView 的实例还不存在,DashboardView 如何订阅由 LandingView 生成的事件?必须有人创建并初始化它才能订阅事件聚合器,对吗?如果那个人是路由器,我是否需要在路由器中预先实例化所有视图?那没有意义。

4

2 回答 2

11

我通过仅在首次命中路线时导入视图来解决此问题:

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;
});

它并没有解决最终必须将所有视图导入路由器的问题,但惰性requirejs调用不会强制加载和预先评估所有脚本和模板。

事实是,某个地方的某个人必须导入模块。路由器是一个合理的位置,因为通常它是用户导航到某个页面(视图)时命中的第一段代码。如果您觉得一个路由器负责太多,您应该考虑将您的路由器分成多个路由器,每个路由器负责您应用程序的不同“部分”。作为一个很好的类比,想想典型 MVC 场景中的控制器。

多路由器示例

userrouter.js处理所有与用户相关的视图('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处理所有与 Post 相关的视图('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,以便延迟加载的模块包含在构建中。

编辑:这是一个要点,它解释了 RequireJS 中的延迟加载和立即加载模块

于 2013-01-06T21:08:13.657 回答
1

我们为此使用了一个工厂,它只返回一个视图实例,它还可以缓存实例:

define(function() {
  // Classes are defined like this { key1: Class1, key2: Class2 }
  // not cachedObjects are defined like this { notCached : { key3: Class3 }}
  return function(Classes) {
    var objectCache = {};

    return {
      get: function(key, options) {
        var cachedObject = objectCache[key];
        if (cachedObject){
          return cachedObject;
        }

        var Class = Classes[key];
        if (Class) {
          cachedObject = new Class(options);
          objectCache[key] = cachedObject;
          return cachedObject;
        }

        Class = Classes.notCached[key];
        if (Class) {
          return new Class(options);
        }
      }
    };
  };
});

然后我们有一个创建工厂的模块:

define([
  'common/factory',
  'views/view1',
  'views/view2',
  'views/view3',
  ], function(
    viewCache,
    View1,
    View2,
    View3
  ) {

  var views = {
    route1: View1,
    route2: View2,
    notCached: {
      route3: View3,
    }
  };

  return viewCache(views);
});

在路由器中,您可以通过调用 viewCache.get(route) 轻松获取视图。好处是解耦视图的创建/缓存,现在可以单独测试。

同样,当我们使用 Marionette 时,我们不会在路由器中使用 viewCache,而是在 RegionManager 中使用,这更适合创建视图。我们的路由器只是用应用程序的实际状态和路由触发事件。

于 2013-01-06T22:00:18.747 回答