tridactyl/src/keydown_content.ts

93 lines
2.6 KiB
TypeScript
Raw Normal View History

/** Shim for the keyboard API because it won't hit in FF57. */
import * as Messaging from "./messaging"
import * as msgsafe from "./msgsafe"
import { isTextEditable, getAllDocumentFrames } from "./dom"
import { isSimpleKey } from "./keyseq"
function keyeventHandler(ke: KeyboardEvent) {
// Ignore JS-generated events for security reasons.
if (!ke.isTrusted) return
2018-04-15 22:32:26 +01:00
// Mode is changed based on ke target in the bg.
if (state.mode === "input" || !isTextEditable(ke.target as Node)) {
modeSpecificSuppression(ke)
}
Messaging.message("keydown_background", "recvEvent", [
msgsafe.KeyboardEvent(ke),
])
}
2018-04-15 22:32:26 +01:00
// {{{ Bad key suppression system
2017-11-19 02:41:01 +00:00
// This is all awful and will go away when we move the parsers and stuff to content properly.
import state from "./state"
2017-11-19 02:41:01 +00:00
import * as normalmode from "./parsers/normalmode"
let keys = []
2018-04-15 22:32:26 +01:00
/** Choose to suppress a key or not */
function modeSpecificSuppression(ke: KeyboardEvent) {
2017-11-19 02:41:01 +00:00
switch (state.mode) {
case "normal":
keys.push(ke)
const response = normalmode.parser(keys)
// Suppress if there's a match.
if (response.isMatch) {
2017-11-19 02:41:01 +00:00
ke.preventDefault()
ke.stopImmediatePropagation()
2017-11-19 02:41:01 +00:00
}
// Update keys array.
keys = response.keys || []
2017-11-19 02:41:01 +00:00
break
// Hintmode can't clean up after itself yet, so it needs to block more FF shortcuts.
2017-11-19 02:41:01 +00:00
case "hint":
case "find":
2018-04-15 22:32:26 +01:00
if (isSimpleKey(ke)) {
ke.preventDefault()
ke.stopImmediatePropagation()
}
break
case "gobble":
if (isSimpleKey(ke) || ke.key === "Escape") {
ke.preventDefault()
ke.stopImmediatePropagation()
}
break
case "input":
if (ke.key === "Tab") {
ke.preventDefault()
ke.stopImmediatePropagation()
}
break
case "ignore":
break
case "insert":
break
2017-11-19 02:41:01 +00:00
}
}
// }}}
// Add listeners
window.addEventListener("keydown", keyeventHandler, true)
document.addEventListener("readystatechange", ev =>
getAllDocumentFrames().map(frame => {
frame.contentWindow.removeEventListener(
"keydown",
keyeventHandler,
true,
)
frame.contentWindow.addEventListener("keydown", keyeventHandler, true)
}),
)
import * as SELF from "./keydown_content"
Messaging.addListener("keydown_content", Messaging.attributeCaller(SELF))
// Dummy export so that TS treats this as a module.
export {}