]> git.gir.st - VimFx.git/blob - extension/lib/vim-frame.coffee
Add `g[`, `g]` and a jump list
[VimFx.git] / extension / lib / vim-frame.coffee
1 ###
2 # Copyright Anton Khodakivskiy 2012, 2013.
3 # Copyright Simon Lydell 2013, 2014, 2015, 2016.
4 #
5 # This file is part of VimFx.
6 #
7 # VimFx is free software: you can redistribute it and/or modify
8 # it under the terms of the GNU General Public License as published by
9 # the Free Software Foundation, either version 3 of the License, or
10 # (at your option) any later version.
11 #
12 # VimFx is distributed in the hope that it will be useful,
13 # but WITHOUT ANY WARRANTY; without even the implied warranty of
14 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 # GNU General Public License for more details.
16 #
17 # You should have received a copy of the GNU General Public License
18 # along with VimFx. If not, see <http://www.gnu.org/licenses/>.
19 ###
20
21 # This file is the equivalent to vim.coffee. `vim.window` is called
22 # `vim.content` to be consistent with Firefox’s frame script terminology and to
23 # avoid confusion about what it represents. There is one `VimFrame` instance for
24 # each tab. It mostly tries to mimic the `Vim` class in vim.coffee, but also
25 # keeps track of web page state. `VimFrame` is not part of the config file API.
26
27 messageManager = require('./message-manager')
28 ScrollableElements = require('./scrollable-elements')
29 utils = require('./utils')
30
31 class VimFrame
32 constructor: (@content) ->
33 @mode = 'normal'
34 @hintMatcher = null
35
36 @resetState()
37
38 messageManager.listen('modeChange', ({mode}) =>
39 @mode = mode
40 )
41
42 messageManager.listen(
43 'markPageInteraction', @markPageInteraction.bind(this)
44 )
45
46 messageManager.listen('clearHover', @clearHover.bind(this))
47
48 # If the target is the topmost document, reset everything. Otherwise filter
49 # out elements belonging to the target frame. On some sites, such as Gmail,
50 # some elements might be dead at this point.
51 resetState: (target = @content.document) ->
52 if target == @content.document
53 @state = {
54 hasInteraction: false
55 shouldRefocus: false
56 marks: {}
57 jumpList: []
58 jumpListIndex: -1
59 explicitBodyFocus: false
60 hasFocusedTextInput: false
61 lastFocusedTextInput: null
62 lastHover: {
63 element: null
64 browserOffset: {x: 0, y: 0}
65 }
66 scrollableElements: new ScrollableElements(@content)
67 markerElements: []
68 inputs: null
69 }
70
71 else
72 isDead = (element) ->
73 return Cu.isDeadWrapper(element) or element.ownerDocument == target
74 check = (obj, prop) ->
75 obj[prop] = null if obj[prop] and isDead(obj[prop])
76
77 check(@state, 'lastFocusedTextInput')
78 check(@state.lastHover, 'element')
79 @state.scrollableElements.reject(isDead)
80 # `markerElements` and `inputs` could theoretically need to be filtered
81 # too at this point. YAGNI until an issue arises from it.
82
83 options: (prefs) -> messageManager.get('options', {prefs})
84
85 _enterMode: (@mode, args...) ->
86 messageManager.send('vimMethod', {
87 method: '_enterMode'
88 args: [@mode, args...]
89 })
90
91 notify: (args...) ->
92 messageManager.send('vimMethod', {method: 'notify', args})
93
94 hideNotification: ->
95 messageManager.send('vimMethod', {method: 'hideNotification'})
96
97 markPageInteraction: (value = true) -> @state.hasInteraction = value
98
99 setHover: (element, browserOffset) ->
100 utils.setHover(element, true)
101 utils.simulateMouseEvents(element, 'hover-start', browserOffset)
102 @state.lastHover.element = element
103 @state.lastHover.browserOffset = browserOffset
104
105 clearHover: ->
106 if @state.lastHover.element
107 {element, browserOffset} = @state.lastHover
108 utils.setHover(element, false)
109 utils.simulateMouseEvents(element, 'hover-end', browserOffset)
110 @state.lastHover.element = null
111
112 addToJumpList: ->
113 [newX, newY] = position = @state.scrollableElements.getPageScrollPosition()
114 jumpList = @state.jumpList[..@state.jumpListIndex]
115 .filter(([x, y]) -> not (x == newX and y == newY))
116 .concat([position])
117 @state.jumpList = jumpList
118 @state.jumpListIndex = jumpList.length - 1
119
120 module.exports = VimFrame
Imprint / Impressum