]> git.gir.st - VimFx.git/blob - extension/packages/events.coffee
Merge branch 'betterstacktrace' into huffman
[VimFx.git] / extension / packages / events.coffee
1 utils = require 'utils'
2 keyUtils = require 'key-utils'
3 { getCommand } = require 'commands'
4 { Vim } = require 'vim'
5 { getPref } = require 'prefs'
6 { setWindowBlacklisted } = require 'button'
7 { unload } = require 'unload'
8 { console } = require 'console'
9
10 { interfaces: Ci } = Components
11
12 vimBucket = new utils.Bucket(utils.getWindowId, (obj) -> new Vim(obj))
13
14 suppressEvent = (event) ->
15 event.preventDefault()
16 event.stopPropagation()
17
18 # *************************
19 # NB! TODO! All this shit needs to be redone!!
20 # *************************
21
22 keyStrFromEvent = (event) ->
23
24 { ctrlKey: ctrl, metaKey: meta, altKey: alt, shiftKey: shift } = event
25
26 if !meta and !alt
27 if keyChar = keyUtils.keyCharFromCode(event.keyCode, shift)
28 keyStr = keyUtils.applyModifiers(keyChar, ctrl, alt, meta)
29
30 return keyStr
31
32 # Passthrough mode is activated when VimFx should temporarily stop processking
33 # keyboard input. For example when a context menu is whown
34 passthrough = false
35
36 logError = (err, eventName) ->
37 console.log("#{ err } (in #{ eventName })\n#{ err.stack.replace(/@(?:.+->)+ /g, '@') }")
38
39 # The following listeners are installed on every top level Chrome window
40 windowsListener =
41 keydown: (event) ->
42
43 if passthrough or getPref('disabled')
44 return
45
46 try
47 isEditable = utils.isElementEditable(event.originalTarget)
48
49 keyStr = keyStrFromEvent(event)
50
51 # We only handle the key if it's recognized by `keyCharFromCode`
52 # and if there is no focused editable element # or if it's the *Esc* key,
53 # which will remove the focus from the currently focused element
54 if keyStr and (not isEditable or keyStr == 'Esc')
55 if window = utils.getCurrentTabWindow(event)
56 if vim = vimBucket.get(window)
57 # No action if blacklisted
58 if vim.blacklisted
59 return
60
61 if vim.handleKeyDown(event, keyStr) and keyStr != 'Esc'
62 suppressEvent event
63 catch err
64 logError(err, 'keydown')
65
66 keypress: (event) ->
67
68 if passthrough or getPref('disabled')
69 return
70
71 try
72 isEditable = utils.isElementEditable(event.originalTarget)
73
74 # Try to execute keys that were accumulated so far.
75 # Suppress event if there is a matching command.
76 if window = utils.getCurrentTabWindow(event)
77 if vim = vimBucket.get(window)
78
79 # No action on blacklisted locations
80 if vim.blacklisted
81 return
82
83 # Blur from any active element on Esc. Calling before `handleKeyPress`
84 # because `vim.keys` will be reset afterwards`
85 blur_on_esc = vim.lastKeyStr == 'Esc' and getPref('blur_on_esc')
86
87 # Process event if there is no editable element in focus
88 # Or last key was Esc key
89 if not isEditable or vim.lastKeyStr == 'Esc'
90 result = vim.handleKeyPress(event)
91
92 # If there was some processing done then suppress the eveng
93 # unless it's the Esc key
94 if result and vim.lastKeyStr != 'Esc'
95 suppressEvent(event)
96
97 # Calling after the command has been executed
98 if blur_on_esc
99 cb = -> event.originalTarget?.ownerDocument?.activeElement?.blur()
100 window.setTimeout(cb, 0)
101
102 catch err
103 logError(err, 'keypress')
104
105 keyup: (event) ->
106 if window = utils.getCurrentTabWindow event
107 if vim = vimBucket.get(window)
108 if vim.lastKeyStr and vim.lastKeyStr != 'Esc'
109 suppressEvent(event)
110
111 vim.lastKeyStr = null
112
113 popupshown: (event) ->
114 if event.target.tagName in [ 'menupopup', 'panel' ]
115 passthrough = true
116
117
118 popuphidden: (event) ->
119 if event.target.tagName in [ 'menupopup', 'panel' ]
120 passthrough = false
121
122 # When the top level window closes we should release all Vims that were
123 # associated with tabs in this window
124 DOMWindowClose: (event) ->
125 if gBrowser = event.originalTarget.gBrowser
126 for tab in gBrowser.tabs
127 if browser = gBrowser.getBrowserForTab(tab)
128 vimBucket.forget(browser.contentWindow)
129
130 TabClose: (event) ->
131 if gBrowser = utils.getEventTabBrowser(event)
132 if browser = gBrowser.getBrowserForTab(event.originalTarget)
133 vimBucket.forget(browser.contentWindow)
134
135 # Update the toolbar button icon to reflect the blacklisted state
136 TabSelect: (event) ->
137 if vim = vimBucket.get(event.originalTarget?.linkedBrowser?.contentDocument?.defaultView)
138 if rootWindow = utils.getRootWindow(vim.window)
139 setWindowBlacklisted(rootWindow, vim.blacklisted)
140
141 # This listener works on individual tabs within Chrome Window
142 # User for: listening for location changes and disabling the extension
143 # on black listed urls
144 tabsListener =
145 onLocationChange: (browser, webProgress, request, location) ->
146 blacklisted = utils.isBlacklisted(location.spec, getPref('black_list'))
147 if vim = vimBucket.get(browser.contentWindow)
148 vim.enterNormalMode()
149 vim.blacklisted = blacklisted
150 if rootWindow = utils.getRootWindow(vim.window)
151 setWindowBlacklisted(rootWindow, vim.blacklisted)
152
153 addEventListeners = (window) ->
154 for name, listener of windowsListener
155 window.addEventListener(name, listener, true)
156
157 # Install onLocationChange listener
158 window.gBrowser.addTabsProgressListener(tabsListener)
159
160 removeEventListeners = ->
161 for name, listener of windowsListener
162 window.removeEventListener(name, listener, true)
163
164 unload ->
165 removeEventListeners(window)
166 window.gBrowser.removeTabsProgressListener(tabsListener)
167
168 exports.addEventListeners = addEventListeners
Imprint / Impressum