根据我的理解,所有的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的方式,对将来遇到这个问题的任何人都有帮助。

其他回答

资产管道文档建议如何做特定于控制器的JS:

例如,如果生成了一个ProjectsController,则会在app/assets/javascripts/projects.js中生成一个新文件。Coffee和另一个在app/assets/stylesheets/projects.css.scss。你应该在它们各自的资产文件中放入一个控制器唯一的JavaScript或CSS,因为这些文件可以仅为这些控制器加载诸如<%= javascript_include_tag params[:controller] %>或<%= stylesheet_link_tag params[:controller] %>这样的行。

链接到:asset_pipeline

这是我如何解决样式问题:(原谅Haml)

%div{:id => "#{params[:controller].parameterize} #{params[:view]}"}
    = yield

这样我开始所有特定的页面。css。Sass文件:

#post
  /* Controller specific code here */
  &#index
    /* View specific code here */
  &#new
  &#edit
  &#show

这样可以很容易地避免任何冲突。 当涉及到.js时。你可以初始化Coffee文件中的元素,比如;

$('#post > #edit') ->
  $('form > h1').css('float', 'right')

希望这能有所帮助。

步骤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/

另一种选择:要创建特定于页面或模型的文件,您可以在assets/javascripts/文件夹中创建目录。

assets/javascripts/global/
assets/javascripts/cupcakes
assets/javascripts/something_else_specific

你的主application.js清单文件可以配置为从global/加载它的文件。特定的页面或页面组可以有它们自己的清单,这些清单从它们自己的特定目录加载文件。Sprockets会自动将application.js加载的文件与特定于页面的文件结合起来,这使得这个解决方案可以工作。

这种技术也可以用于style_sheets/。

你可以在你的布局文件(例如application.html.erb)中添加这一行来自动加载特定于控制器的javascript文件(当你生成控制器时创建的那个):

<%= javascript_include_tag params[:controller] %>

您还可以添加一行来在每个操作的基础上自动加载脚本文件。

<%= javascript_include_tag params[:controller] + "/" + params[:action] %>

只需将页面脚本放到以控制器名称命名的子目录中。在这些文件中,您可以使用=require包含其他脚本。 最好创建一个助手,只在文件存在的情况下包含该文件,以避免浏览器出现404错误。