根据我的理解,所有的JavaScript都合并到一个文件中。Rails在添加//= require_tree时默认这样做。到你的application.js manifest文件的底部。

这听起来像是一个真正的救星,但我有点担心特定于页面的JavaScript代码。这段代码是否在每一页上执行?我想要的最后一件事是当我的所有对象只在1页上需要时,为每一页实例化它们。

此外,代码是否也有可能发生冲突?

还是在页面底部放置一个小的脚本标记,只调用一个执行页面javascript代码的方法?

你不再需要require.js了吗?

谢谢

编辑:我很感激所有的答案…我认为他们并没有真正解决问题。其中一些是关于样式的,似乎没有关联……其他人只是提到javascript_include_tag…我知道这是存在的(很明显…),但Rails 3.1的前进方式似乎是将所有JavaScript打包到一个文件中,而不是在每个页面的底部加载单个JavaScript。

我能想到的最好的解决方案是用id或类将某些特性包装在div标签中。在JavaScript代码中,只需检查id或类是否在页面上,如果在,则运行与之关联的JavaScript代码。这样,如果动态元素不在页面上,JavaScript代码就不会运行——即使它包含在由Sprockets打包的庞大application.js文件中。

我上面的解决方案的好处是,如果一个搜索框包含在100个页面中的8个页面上,那么它将只在这8个页面上运行。你也不需要在网站的8个页面上包含相同的代码。事实上,您再也不需要在站点的任何地方包含手动脚本标记。

我想这就是我问题的真正答案。


当前回答

我知道我来这个派对有点晚了,但我想提出一个我最近一直在用的解决方案。然而,让我首先提一下……

The Rails 3.1/3.2 Way(不,先生,我不喜欢它。)

见:http://guides.rubyonrails.org/asset_pipeline.html how-to-use-the-asset-pipeline

为了完整起见,我在这个回答中包括以下内容,因为这不是一个不可行的解决方案……虽然我不太喜欢。

“Rails方式”是一个面向控制器的解决方案,而不是像这个问题的原始作者所要求的那样面向视图。有一些特定于控制器的JS文件,以它们各自的控制器命名。所有这些文件都被放置在一个文件夹树中,默认情况下,任何application.js require指令都不包含这个文件夹树。

为了包含特定于控制器的代码,将以下内容添加到视图中。

<%= javascript_include_tag params[:controller] %>

我讨厌这个解决方案,但它就在那里,而且很快。假设你可以将这些文件命名为“people-index.js”和“people-show.js”,然后使用“#{params[:controller]}-index”来获得面向视图的解决方案。同样是权宜之计,但我不喜欢。

我的数据属性方式

叫我疯狂,但我想我所有的JS编译和缩小到application.js当我部署。我不想在所有地方都包含这些零散的文件。

我把我所有的JS加载到一个紧凑的,即将被浏览器缓存的文件中。如果我的application.js的某一部分需要在页面上被触发,我会让HTML来告诉我,而不是Rails。

我使用了一个名为data-jstags的自定义数据属性,而不是将JS锁定到特定的元素id或用标记类丢弃HTML。

<input name="search" data-jstag="auto-suggest hint" />

在每个页面上,我使用-此处插入首选JS库方法-在DOM完成加载时运行代码。这段引导代码执行以下操作:

遍历DOM中标记有data-jstag的所有元素 对于每个元素,在空格上分割属性值,创建一个标记字符串数组。 对于每个标记字符串,在哈希中对该标记执行查找。 如果找到匹配的键,则运行与之关联的函数,将元素作为参数传递。

所以说,我有以下定义在我的应用程序。js某处:

function my_autosuggest_init(element) {
  /* Add events to watch input and make suggestions... */
}

function my_hint_init(element) {
  /* Add events to show a hint on change/blur when blank... */
  /* Yes, I know HTML 5 can do this natively with attributes. */
}

var JSTags = {
  'auto-suggest': my_autosuggest_init,
  'hint': my_hint_init
};

引导事件将对搜索输入应用my_autosuggest_init和my_hint_init函数,在用户输入时将其转换为显示建议列表的输入,并在输入为空白和不集中时提供某种输入提示。

除非某些元素被标记为data-jstag="auto-suggest",否则自动建议代码永远不会触发。然而,它总是在那里,缩小并最终缓存在我的application.js中,以便我在页面上需要它。

如果你需要将额外的参数传递给你标记的JS函数,你将不得不应用一些创造性。要么添加数据参数属性,提出某种参数语法,甚至使用混合方法。

Even if I have some complicated workflow that seems controller-specific, I will just create a file for it in my lib folder, pack it into application.js, and tag it with something like 'new-thing-wizard'. When my bootstrap hits that tag, my nice, fancy wizard will be instantiated and run. It runs for that controller's view(s) when needed, but is not otherwise coupled to the controller. In fact, if I code my wizard right, I might be able to provide all configuration data in the views and therefore be able to re-use my wizard later for any other controller that needs it.

不管怎样,这就是我现在实现特定页面JS的方式,它对简单的站点设计和更复杂/丰富的应用程序都很有帮助。希望我在这里介绍的两种解决方案之一,我的方式或Rails的方式,对将来遇到这个问题的任何人都有帮助。

其他回答

我很感激所有的答案……我认为他们并没有真正解决问题。其中一些是关于样式的,似乎没有关联……其他人只是提到javascript_include_tag…我知道这是存在的(很明显…),但Rails 3.1的前进方式似乎是将所有Javascript打包到一个文件中,而不是在每个页面的底部加载单个Javascript。

我能想到的最好的解决方案是用id或类将某些特性包装在div标签中。在javascript代码中。然后只需检查id或类是否在页面上,如果在,则运行与之关联的javascript代码。这样,如果动态元素不在页面上,javascript代码就不会运行——即使它包含在由Sprockets打包的庞大的application.js文件中。

我上面的解决方案的好处是,如果一个搜索框包含在100个页面中的8个页面上,那么它将只在这8个页面上运行。你也不需要在网站的8个页面上包含相同的代码。事实上,你再也不需要在你的站点的任何地方包含手动脚本标签了——除非可能是预加载数据。

我想这就是我问题的真正答案。

步骤1。移除require_tree。在你的application。js和application。css中。

步骤2。编辑你的application.html。Erb(由rails默认)在布局文件夹。在以下标签中添加“params[:controller]”。

<%= stylesheet_link_tag    'application', params[:controller], media: 'all', 'data-turbolinks-track' => true %>

<%= javascript_include_tag 'application', params[:controller], 'data-turbolinks-track' => true %>

步骤3。在config/initializers/assets.rb中添加一个文件

%w( controller_one controller_two controller_three ).each do |controller|
  Rails.application.config.assets.precompile += ["#{controller}.js", "#{controller}.js.coffee", "#{controller}.css", "#{controller}.scss"]
end

引用: http://theflyingdeveloper.com/controller-specific-assets-with-rails-4/

我有另一个解决方案,虽然原始的工作对我来说很好,不需要任何花哨的选择性加载策略。放入普通的文档准备函数,然后测试当前的窗口位置,看看它是否是你的javascript所针对的页面:

$(document).ready(function() {
   if(window.location.pathname.indexOf('/yourpage') != -1) {
          // the javascript you want to execute
   }
}

这仍然允许rails 3加载所有的js。X放在一个小包中,但不会产生太多开销,也不会与js不打算用于的页面产生任何冲突。

菲利普的回答很好。下面是让它工作的代码:

在application.html.erb:

<身体类= " < % = params[:控制器].parameterize % > " >

假设你的控制器叫做Projects,它会生成:

<身体类= "项目" >

然后在projects.js.coffee中:

jQuery ->
  if $('body.projects').length > 0  
     $('h1').click ->
       alert 'you clicked on an h1 in Projects'

我还没有尝试过这个方法,但看起来下面的方法是正确的:

如果你有一个content_for是javascript(例如与真正的javascript在里面),链轮将不知道它,因此这将以同样的方式工作,因为它现在。 如果你想从javascript包中排除一个文件,你可以进入config/sprockets。并相应地修改source_files文件。然后,您只需在需要的地方包括您排除的任何文件。