2017-10-28 05:11:10 +01:00
|
|
|
export type TabMessageType =
|
|
|
|
"excmd_content" |
|
|
|
|
"keydown_content" |
|
2017-10-28 13:42:54 +01:00
|
|
|
"commandline_content" |
|
2017-11-08 23:20:41 +00:00
|
|
|
"commandline_frame" |
|
|
|
|
"hinting_content"
|
2017-10-28 05:11:10 +01:00
|
|
|
export type NonTabMessageType =
|
|
|
|
"keydown_background" |
|
|
|
|
"commandline_background"
|
|
|
|
export type MessageType = TabMessageType | NonTabMessageType
|
|
|
|
|
|
|
|
export interface Message {
|
|
|
|
type: MessageType
|
|
|
|
// and other unknown attributes...
|
|
|
|
[key: string]: any
|
|
|
|
}
|
|
|
|
|
|
|
|
export type listener = (message: Message, sender?, sendResponse?) => void|Promise<any>
|
2017-10-23 09:42:50 +01:00
|
|
|
|
|
|
|
/** await a promise and console.error and rethrow if it errors
|
|
|
|
|
|
|
|
Errors from promises don't get logged unless you seek them out.
|
2017-10-28 05:11:10 +01:00
|
|
|
|
|
|
|
There's an event for catching these, but it's not implemented in firefox
|
|
|
|
yet: https://bugzilla.mozilla.org/show_bug.cgi?id=1269371
|
2017-10-23 09:42:50 +01:00
|
|
|
*/
|
|
|
|
async function l(promise) {
|
|
|
|
try {
|
|
|
|
return await promise
|
|
|
|
} catch (e) {
|
|
|
|
console.error(e)
|
|
|
|
throw e
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Calls methods on obj that match .command and sends responses back
|
|
|
|
export function attributeCaller(obj) {
|
|
|
|
function handler(message: Message, sender, sendResponse) {
|
|
|
|
console.log(message)
|
|
|
|
|
|
|
|
// Args may be undefined, but you can't spread undefined...
|
|
|
|
if (message.args === undefined) message.args = []
|
|
|
|
|
|
|
|
// Call command on obj
|
2017-10-28 05:11:10 +01:00
|
|
|
try {
|
|
|
|
let response = obj[message.command](...message.args)
|
|
|
|
|
|
|
|
// Return response to sender
|
|
|
|
if (response instanceof Promise) {
|
|
|
|
return response
|
|
|
|
} else {
|
|
|
|
sendResponse(response)
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
return new Promise((resolve, error)=>error(e))
|
2017-10-05 23:11:56 +01:00
|
|
|
}
|
|
|
|
}
|
2017-10-23 09:42:50 +01:00
|
|
|
return handler
|
|
|
|
}
|
|
|
|
|
2017-10-28 05:11:10 +01:00
|
|
|
|
2017-10-23 09:42:50 +01:00
|
|
|
/** Send a message to non-content scripts */
|
2017-10-28 05:11:10 +01:00
|
|
|
export async function message(type: NonTabMessageType, command, args?) {
|
2017-10-23 09:42:50 +01:00
|
|
|
// One day typescript will be smart enough to back propagate this cast.
|
2017-10-28 05:11:10 +01:00
|
|
|
return l(browser.runtime.sendMessage({type, command, args} as Message))
|
2017-10-05 23:11:56 +01:00
|
|
|
}
|
|
|
|
|
2017-10-23 09:42:50 +01:00
|
|
|
/** The first active tab in the currentWindow.
|
|
|
|
*
|
|
|
|
* TODO: Highlander theory: Can there ever be more than one?
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
//#background_helper
|
|
|
|
async function activeTab() {
|
|
|
|
return (await l(browser.tabs.query({active: true, currentWindow: true})))[0]
|
|
|
|
}
|
|
|
|
|
|
|
|
//#background_helper
|
|
|
|
async function activeTabID() {
|
|
|
|
return (await activeTab()).id
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Message the active tab of the currentWindow */
|
|
|
|
//#background_helper
|
2017-10-28 05:11:10 +01:00
|
|
|
export async function messageActiveTab(type: TabMessageType, command: string, args?: any[]) {
|
|
|
|
return messageTab(await activeTabID(), type, command, args)
|
2017-10-23 09:42:50 +01:00
|
|
|
}
|
|
|
|
|
2017-10-28 05:11:10 +01:00
|
|
|
export async function messageTab(tabId, type: TabMessageType, command, args?) {
|
2017-10-23 09:42:50 +01:00
|
|
|
let message: Message = {
|
|
|
|
type,
|
|
|
|
command,
|
|
|
|
args,
|
|
|
|
}
|
2017-10-28 05:11:10 +01:00
|
|
|
return l(browser.tabs.sendMessage(tabId, message))
|
2017-10-23 09:42:50 +01:00
|
|
|
}
|
|
|
|
|
2017-10-28 05:11:10 +01:00
|
|
|
export async function messageAllTabs(type: TabMessageType, command: string, args?: any[]) {
|
|
|
|
let responses = []
|
2017-10-23 09:42:50 +01:00
|
|
|
for (let tab of await browser.tabs.query({})) {
|
2017-10-28 05:11:10 +01:00
|
|
|
try { responses.push(await messageTab(tab.id, type, command, args)) }
|
|
|
|
catch (e) { console.error(e) }
|
|
|
|
}
|
|
|
|
return responses
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const listeners = new Map<string, Set<listener>>()
|
|
|
|
|
|
|
|
/** Register a listener to be called for each message with type */
|
|
|
|
export function addListener(type: MessageType, callback: listener) {
|
|
|
|
if (!listeners.get(type)) {
|
|
|
|
listeners.set(type, new Set())
|
|
|
|
}
|
|
|
|
listeners.get(type).add(callback)
|
|
|
|
return () => { listeners.get(type).delete(callback) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Recv a message from runtime.onMessage and send to all listeners */
|
|
|
|
function onMessage(message, sender, sendResponse) {
|
|
|
|
if (listeners.get(message.type)) {
|
|
|
|
for (let listener of listeners.get(message.type)) {
|
|
|
|
listener(message, sender, sendResponse)
|
|
|
|
}
|
2017-10-23 09:42:50 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-10-05 23:11:56 +01:00
|
|
|
browser.runtime.onMessage.addListener(onMessage)
|