草庐IT

javascript - 在 Ext JS 4 MVC 应用程序中,什么应该负责加载商店?

coder 2024-07-28 原文

考虑一个复杂的应用程序,其中有自定义过滤逻辑和不同的加载模式;隐藏时延迟加载,隐藏时不加载,但在显示时加载并在修改自定义过滤器时重新加载等。

mvc 应用程序的哪个部分应该负责加载,以及如何连接它?

最佳答案

当我开始使用 MVC 时,我从未从 Sencha 那里找到明确的答案,但我可以告诉你我成功地为一些应用程序做了什么。

我根据它们的使用方式创建和加载我的商店。对我来说,这似乎分为三个不同的类别:

  • 适用于整个应用程序的商店
  • 适用于 View 所有实例的存储
  • 绑定(bind)到单个 View 实例的存储

  • 1.适用于整个应用程序的商店

    我通常有一个“主” Controller 来处理我的应用程序的框架,比如键盘导航、主菜单等。

    属于上述第一类的商店进入此“主” Controller 的 stores大批。如果这些商店不依赖于另一家商店,那么我只是让它们autoLoad: true和他们做完了。但是在它们依赖于另一个存储数据的情况下,然后我向父存储添加一个监听器以加载父存储中的依赖存储 onLoad事件。

    我发现属于第一类的一些商店的一个例子是我在整个应用程序的组合框中使用的引用商店,通常在许多不同类型的 View 中。这些通常配置为 autoLoad: true然后我再也没有为该用户的 session 加载它们。每当我需要一个使用引用存储的组合框时,我都可以这样设置:
    {
        xtype: 'combobox',
        allowBlank: false,
        forceSelection: true,
        store: Ext.getStore('SomeReferenceStore'),
        queryMode: 'local', // this makes sure the store doesn't reload
        valueField: 'id',
        displayField: 'name'
    }
    

    举一个两个商店的例子,它们都属于第一类,一个依赖于另一个:

    在我的一个应用程序中,我拥有动态数量的用户权限,当用户登录该应用程序时,我需要获取不同的权限并修改 User模型为每个不同的权限包含一个 bool 字段:
    Ext.define('MyApp.controller.Main', {
        extend: 'Ext.app.Controller',
    
        models: [
            'User', 
            'Reference',
            // etc...
        ],
    
        stores: [
            'CurrentUser', 
            'PermissionRef', // this is "autoLoad: true"
            // etc...
        ],
    
        init: function() {
            var me = this;       
    
            // update the user model and load the user
            me.getPermissionRefStore().on('load', function(store, records) {
                var model = me.getUserModel(),
                    fields = model.prototype.fields.getRange();
    
                // append the permissions onto the User model fields
                Ext.each(records, function(permission) {
                    fields.push({
                        name: permission.get('name'), 
                        type: 'bool',
                    });
                });
    
                // update the user model with the permission fields
                model.setFields(fields);
    
                // load the current user
                me.getCurrentUserStore().load();
    
                // other stuff...
    
            });
    
            // other stuff...
    
        }
    
    });
    

    有了这个,每当我需要对用户的引用以及他有哪些可用权限时,我只需调用:
    var user = Ext.getStore('CurrentUser').first();
    

    有时 View 也将依赖于正在加载的商店。例如,我的主菜单项由数据库表确定,在这种情况下,我将包含一个 onLoad以同样的方式监听(在 Controller 的 init 函数中):
    // create the menu once we know what menu items are available
    me.getMenuStore().on('load', function(store) {
        me.getViewport().add(Ext.widget('mainpanel'));            
    });
    
    MyApp.store.Menu本身将配置为 autoLoad: true .

    2. 适用于 View 所有实例的存储

    我创建和加载这些就像第一类一样,只是我将它们放在特定 View Controller 的 stores 中。数组而不是“主” Controller stores大批。

    然后是同样的基本概念,有些是autoLoad: true如果它们依赖于另一家商店的数据,则有些则不然。

    对于那些不是 autoLoad: true它们被加载到 Controller init 内的某处函数或作为某些事件被触发的结果。

    3. 绑定(bind)到单个 View 实例的存储

    在这里,我可能会反对 MVC 模式,但是对于应用到 View 的单个实例然后在 View 本身内部应用的存储来说,真的没有比这更好的地方了。

    在大多数情况下,我什至不会将这些存储放在 View Controller 的 stores 中。大批。我只是在 View 的 initComponent 中创建并加载它们功能。因此,当 View 被销毁时,不再有对商店的引用,因此商店也将被销毁。这有助于减少可以多次创建的商店的资源。

    例如,假设您有一个 MyApp.view.UnicornWeightGrid扩展 gridpanel并显示随着时间的推移独 Angular 兽的渐进权重。用户可以打开 UnicornWeightGrid对于该领域中的所有独 Angular 兽,用于比较和交叉引用。 UnicornWeightStore 将有许多不同的实例。因为有 UnicornWeightGrid 的实例.

    View 可以这样定义:
    Ext.define('MyApp.view.UnicornWeightGrid', {
        extend: 'Ext.grid.Panel',
        alias: 'widget.unicornweight',
        requires: [
            'MyApp.model.UnicornWeight'
        ],
        closable: true,    
        initComponent: function() {
            var me = this;
    
            me.store = Ext.create('Ext.data.Store', {
                model: 'MyApp.model.UnicornWeight',
                proxy: {
                    type: 'ajax',
                    url: 'unicorns/weight',
                    extraParams: {
                        name: me.unicornName
                    }
                }
            });
    
            me.store.load();
        });
    });
    

    然后每当我们想要获得不同的 UnicornWeightGrid我们可以简单地调用:
    var grid = Ext.widget('unicornweight', {
        unicornName: someUnicornNameVariable
    });
    
    myTabPanel.add(grid);
    

    任意 onLoad我在 View 中定义的商店上需要的监听器,我也附加在 View 内。所以我真的不需要其他任何地方的引用。

    尽管如此,这绝不是设置商店的唯一方法。

    我发现这是在创建几个不同的 ExtJS“MVC”应用程序时始终如一地处理存储的最可行的方法,但有时我在“特殊”情况下也会以不同的方式执行此操作。

    您应该记住,“MVC”是一种非常松散的设计模式,它在我使用过的几乎每个框架中的定义和实现方式都不同。所以你几乎可以做任何最适合你的事情。

    关于javascript - 在 Ext JS 4 MVC 应用程序中,什么应该负责加载商店?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12106055/

    有关javascript - 在 Ext JS 4 MVC 应用程序中,什么应该负责加载商店?的更多相关文章

    1. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

      类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

    2. ruby-on-rails - Rails - 子类化模型的设计模式是什么? - 2

      我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co

    3. ruby - 什么是填充的 Base64 编码字符串以及如何在 ruby​​ 中生成它们? - 2

      我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%

    4. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

      我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

    5. ruby - 为什么 4.1%2 使用 Ruby 返回 0.0999999999999996?但是 4.2%2==0.2 - 2

      为什么4.1%2返回0.0999999999999996?但是4.2%2==0.2。 最佳答案 参见此处:WhatEveryProgrammerShouldKnowAboutFloating-PointArithmetic实数是无限的。计算机使用的位数有限(今天是32位、64位)。因此计算机进行的浮点运算不能代表所有的实数。0.1是这些数字之一。请注意,这不是与Ruby相关的问题,而是与所有编程语言相关的问题,因为它来自计算机表示实数的方式。 关于ruby-为什么4.1%2使用Ruby返

    6. ruby-on-rails - Rails 应用程序之间的通信 - 2

      我构建了两个需要相互通信和发送文件的Rails应用程序。例如,一个Rails应用程序会发送请求以查看其他应用程序数据库中的表。然后另一个应用程序将呈现该表的json并将其发回。我还希望一个应用程序将存储在其公共(public)目录中的文本文件发送到另一个应用程序的公共(public)目录。我从来没有做过这样的事情,所以我什至不知道从哪里开始。任何帮助,将不胜感激。谢谢! 最佳答案 无论Rails是什么,几乎所有Web应用程序都有您的要求,大多数现代Web应用程序都需要相互通信。但是有一个小小的理解需要你坚持下去,网站不应直接访问彼此

    7. ruby - 无法运行 Rails 2.x 应用程序 - 2

      我尝试运行2.x应用程序。我使用rvm并为此应用程序设置其他版本的ruby​​:$rvmuseree-1.8.7-head我尝试运行服务器,然后出现很多错误:$script/serverNOTE:Gem.source_indexisdeprecated,useSpecification.Itwillberemovedonorafter2011-11-01.Gem.source_indexcalledfrom/Users/serg/rails_projects_terminal/work_proj/spohelp/config/../vendor/rails/railties/lib/r

    8. ruby - 如何在续集中重新加载表模式? - 2

      鉴于我有以下迁移:Sequel.migrationdoupdoalter_table:usersdoadd_column:is_admin,:default=>falseend#SequelrunsaDESCRIBEtablestatement,whenthemodelisloaded.#Atthispoint,itdoesnotknowthatusershaveais_adminflag.#Soitfails.@user=User.find(:email=>"admin@fancy-startup.example")@user.is_admin=true@user.save!ende

    9. ruby-on-rails - Rails 应用程序中的 Rails : How are you using application_controller. rb 是新手吗? - 2

      刚入门rails,开始慢慢理解。有人可以解释或给我一些关于在application_controller中编码的好处或时间和原因的想法吗?有哪些用例。您如何为Rails应用程序使用应用程序Controller?我不想在那里放太多代码,因为据我了解,每个请求都会调用此Controller。这是真的? 最佳答案 ApplicationController实际上是您应用程序中的每个其他Controller都将从中继承的类(尽管这不是强制性的)。我同意不要用太多代码弄乱它并保持干净整洁的态度,尽管在某些情况下ApplicationContr

    10. ruby - ruby 中的 TOPLEVEL_BINDING 是什么? - 2

      它不等于主线程的binding,这个toplevel作用域是什么?此作用域与主线程中的binding有何不同?>ruby-e'putsTOPLEVEL_BINDING===binding'false 最佳答案 事实是,TOPLEVEL_BINDING始终引用Binding的预定义全局实例,而Kernel#binding创建的新实例>Binding每次封装当前执行上下文。在顶层,它们都包含相同的绑定(bind),但它们不是同一个对象,您无法使用==或===测试它们的绑定(bind)相等性。putsTOPLEVEL_BINDINGput

    随机推荐