- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在使用 Turbolinks 5 和 Rails,试图弄清楚它如何改变游戏规则。
传统上,我的页面通过一组 JavaScript 来工作,其中包含库(jQuery、Knockout 等)和应用程序特定代码,例如 View 模型 - 所有以某种有组织的结构附加到窗口对象。应用程序某个区域中的所有 Controller ,例如前端将共享相同的 JavaScript(和 CSS)包。
在每个使用 JavaScript 的页面上,都会有一个初始化程序来启动。例如:
$.ready(function() { window.users.update.init(#{@user_edit_view_model.javascript_configuration.to_json.html_safe}) } )
这个 init 函数的职责是设置一个淘汰 View 模型并将其绑定(bind)到某个节点。 javascript_configuration 可能包含当前用户的初始化状态。
无论如何,这种方法似乎不太适合 Turbolinks。
我知道只有当用户访问users#edit
作为第一页(或进行硬刷新)时它才会触发,所以$.ready
显然已经出局了图片的内容。
如果我附加到 turbolinks:load
事件,则上述代码不仅会在用户输入 users#edit
时触发,还会在用户导航到的任何页面上触发随后(直到他/她在某个时刻进行完全刷新)。
我已经能够通过定义一个在第一次执行后删除回调的函数来解决这个问题。
# For running code after the current page is ready
window.turbolinks_in = (callback) ->
event_listener = ->
callback()
window.removeEventListener("turbolinks:load", event_listener)
window.addEventListener "turbolinks:load", event_listener
此外,我还设计了:
# For running code when leaving the current page
window.turbolinks_out = (callback) ->
event_listener = ->
callback()
window.removeEventListener("turbolinks:before-cache", event_listener)
window.addEventListener "turbolinks:before-cache", event_listener
这两个函数似乎允许我在页面加载和“卸载”时运行代码。
我的问题是:
turbolinks:before-cache
,那么如何在缓存页面短暂显示后进行清理?我确实理解,当(已经)缓存的页面被替换时,不应触发 turbolinks:before-cache
,这在语义上是有意义的,但是它的位置是什么?像 turbolinks:unload
这样的东西不存在。最佳答案
- Seeing as I had to come up with my own wrapper functions, I suspect Turbolinks has been deliberately developed to discourage the initialization flow that I use. Is that true?
- If it's true, what is the idiomatic way to go about? How should I set up a knockout view model or for that matter, run any code that pertains to the particular page?
就在特定页面上运行代码而言,首先,我通常会避免为该页面添加内联脚本
。这通常在非 Turbolinks 应用程序中工作正常,但在启用 Turbolinks 的应用程序中,状态和事件处理程序在页面加载之间维护,事情可能会变得有点困惑。理想情况下,您应该将所有 JS 包含在一个文件中。 (如果您想知道如何注入(inject)服务器生成的内容,我稍后会介绍。)
在设置和拆卸方面,您的做法是正确的,但我想知道您是否会考虑使用 turbolinks_in
和 turbolinks_out
函数在 turbolinks:load
上实现单个初始化函数,然后在 turbolinks:before-cache
和 turbolinks:before-render
上实现单个拆卸函数code> (这是您要执行的“卸载”事件)。您的设置/拆卸代码可能如下所示:
document.addEventListener('turbolinks:load', window.MyApp.init)
document.addEventListener('turbolinks:before-cache', window.MyApp.destroy)
document.addEventListener('turbolinks:before-render', window.MyApp.destroy)
因此,window.MyApp.init
不是直接在内联脚本中调用对象的 init
函数,而是决定要初始化哪些对象。
它是如何决定的?!
看起来您正在为 JS 对象镜像 Controller 名称和操作名称。这是一个很好的起点,我们只需要将这些名称输入客户端即可。一种常见的方法(我相信 Basecamp 使用过)是在 head
中使用 meta
元素来输出一些服务器生成的属性:
<meta name="controller_name" content="<%= controller_name %>">
<meta name="action_name" content="<%= action_name %>">
只要您的对象遵循 Controller /操作命名模式,您的应用程序的初始化函数可能类似于:
;(function () {
window.MyApp = {
init: function () {
var controller = window[getControllerName()]
var action
if (controller) action = controller[getActionName()]
if (action && typeof action.init === 'function') action.init()
}
}
function getControllerName () {
return getMeta('controller_name')
}
function getActionName () {
return getMeta('action_name')
}
function getMeta (name) {
var meta = document.querySelector('[name=' + name + ']')
if (meta) return meta.getAttribute('content')
}
})()
您可以按照此方法包含您的 JavaScript 配置:
<meta name="javascript_config" content="<%= @view_model.javascript_configuration.to_json %>">
然后您的应用程序的 init 函数可以使用如下配置调用相关的初始化程序:
window.MyApp = {
init: function () {
var controller = window[getControllerName()]
var action
var config = getConfig()
if (controller) action = controller[getActionName()]
if (action && typeof action.init === 'function') action.init(config)
}
}
// …
function getConfig () {
return JSON.parse(getMeta('javascript_config') || null)
}
最后,我们需要拆解初始化的结果。为此,我们将存储对象的 init
的返回值,如果它包含 destroy
函数,我们将调用它:
;(function () {
var result
window.MyApp = {
init: function () {
var controller = window[getControllerName()]
var action
var config = getConfig()
if (controller) action = controller[getActionName()]
if (action && typeof action.init === 'function') {
result = action.init(config)
}
},
// …
destroy: function () {
if (result && typeof result.destroy === 'function') result.destroy()
result = undefined
}
}
})()
把它们放在一起:
;(function () {
var result
window.MyApp = {
init: function () {
var controller = window[getControllerName()]
var action
var config = getConfig()
if (controller) action = controller[getActionName()]
if (action && typeof action.init === 'function') {
result = action.init(config)
}
},
destroy: function () {
if (result && typeof result.destroy === 'function') result.destroy()
result = undefined
}
}
function getControllerName () {
return getMeta('controller_name')
}
function getActionName () {
return getMeta('action_name')
}
function getConfig () {
return JSON.parse(getMeta('javascript_config') || null)
}
function getMeta (name) {
var meta = document.querySelector('[name=' + name + ']')
if (meta) return meta.getAttribute('content')
}
})()
显然,这种方法仅适用于每页一个 init,因此您可能需要对其进行调整以适应更多情况。实现这一点超出了本答案的范围,但您可能希望查看 T3 ,它会为您处理这一切!
关于javascript - rails Turbolinks 5 : How do I run JS code on a particular page enter and exit?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44183674/
我已经禁用了回车键,以便可以执行 ajax 提交,但我想确保如果用户在服务器响应返回之前按两次回车键,则表单不会提交两次。 这里我禁用了回车键并为其分配了一个名为 submitForm() 的函数:
我正在尝试将 $animate 服务合并到我自己的指令中。我无法进入并离开以实际设置动画。 奇怪的是,使用$animate.enter,元素附加到DOM,回调函数触发。但似乎从未添加 ng-anima
如果 ,我如何检查 Bash输入 键被按下了? 我正在使用读取命令: read -p "Please press ENTER" var 最佳答案 首先检查退出状态是否正常($?应该为0)。 其次,检查
在我的 PreviewKeyDown() handler 如何区分数字键盘上的 ENTER 键和主板上的 ENTER 键? 两个键返回相同的值 Key.Enter为 KeyEventArgs.Key
这个问题已经有答案了: How do I detect "shift+enter" and generate a new line in Textarea? (19 个回答) 已关闭10 年前。 我正
我的问题与这个 question 有点相关.但是,我没有奢侈去监听 submit 事件,所以这个问题的答案对我来说不起作用。 我的问题:是否可以在 Javascript 中检测日文/中文建议菜单何时打
我希望能够在不需要实现新类的情况下修改 wpf 文本框的行为。 我想要一个类似 Enter/Alt+Enter 行为的 Excel,当用户点击“Enter”时,文本框被验证(movefocus ...
在 Chrome 中的 contenteditable 中按 Enter 时,会插入一个 div。这会干扰我的标记,我需要它成为 br。 我知道shift-enter是一个br。解决问题的最佳方法是什
我正在使用$('.inputs').keydown(function (e) { if (e.which === 13) {到达下一个输入字段。但是每次我按 enter/return 转到下一个输入字
我有一个 textarea ,其中当您单独按 Enter 时,jquery ajax 会启动,但我想添加一个功能,即当您按 Enter + Shift 时仅转到新行并且不启动ajax。你能指导一下吗?
场景如下: 有一个密码字段,我可以在其中输入文本(这是使用 sendKeys 完成的) 现在,没有提交/输入按钮 所以我必须按 Enter 键并登录应用程序 以下是我搜索并尝试但没有成功的几种方法:
这是我尝试过的,但显然失败了: ed.on('keyup', function(e){ console.log(e.keyCode) if(e.keyCode == 13 && !e.
所以我有一个带有 jQuery 的表单,可以阻止用户使用 Enter 键提交表单。 这是代码: $(document).ready(function() { // code prevent
有人知道如何在Javascript中检测按键代码“+1 Enter”和“-1 Enter”吗?我想检测何时一一按下键码而不是一次按下键码。 但是当我编写以下代码时,没有给出任何效果。 // +1 en
我想添加 shift + enter 键转到下一行,只需 enter 键 即可使用 JAVASCRIPT 触发提交按钮仅。 如何实现? 我也想提交多行文本......提前致谢 function typ
int DisplaySchedule() { int nDisplaySchedule_Choice; system("cls"); printf("----- DISPLA
在 textarea 中,当用户按下 Shift+Enter 时,它应该在下一个新行继续,当他简单地按下 Enter 时,它应该提交表单而不使用提交按钮。 这是 Fiddle!! 我浏览了很多但对我没
我有以下简单的 我还有以下 jQuery/JavaScript 代码块: $('textarea#streamWriter').keydown(function (e) { if (e.k
我有许多将转换为HTML的Word文档。要求单词文档中的段落应转换为元素。 在使用Microsoft Office API的SaveAs方法进行了一些测试以将文档转换为HTML之后,我意识到带有手动换
终端是否可以检测 ⇧ Shift+Enter↵ 或 Ctrl+Enter↵ 按键? 我正在尝试配置 vim 来执行使用这些序列的键映射,虽然它们在 gvim 中工作正常,但它们似乎在任何终端控制台中都
我是一名优秀的程序员,十分优秀!