Support for dynamic keybindings linked to actions/scripts
This commit is contained in:
parent
f098c924fb
commit
d0f9be35c6
9 changed files with 590 additions and 150 deletions
173
package-lock.json
generated
173
package-lock.json
generated
|
@ -1664,7 +1664,6 @@
|
|||
"version": "1.0.10",
|
||||
"resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz",
|
||||
"integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"sprintf-js": "~1.0.2"
|
||||
}
|
||||
|
@ -1915,6 +1914,27 @@
|
|||
"object.assign": "^4.1.0"
|
||||
}
|
||||
},
|
||||
"babel-runtime": {
|
||||
"version": "6.26.0",
|
||||
"resolved": "https://registry.npmjs.org/babel-runtime/-/babel-runtime-6.26.0.tgz",
|
||||
"integrity": "sha1-llxwWGaOgrVde/4E/yM3vItWR/4=",
|
||||
"requires": {
|
||||
"core-js": "^2.4.0",
|
||||
"regenerator-runtime": "^0.11.0"
|
||||
},
|
||||
"dependencies": {
|
||||
"core-js": {
|
||||
"version": "2.6.11",
|
||||
"resolved": "https://registry.npmjs.org/core-js/-/core-js-2.6.11.tgz",
|
||||
"integrity": "sha512-5wjnpaT/3dV+XB4borEsnAYQchn00XSgTAWKDkEqv+K8KevjbzmofK6hfJ9TZIlpj2N0xQpazy7PiRQiWHqzWg=="
|
||||
},
|
||||
"regenerator-runtime": {
|
||||
"version": "0.11.1",
|
||||
"resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.11.1.tgz",
|
||||
"integrity": "sha512-MguG95oij0fC3QV3URf4V2SDYGJhJnJGqvIIgdECeODCT98wSWDAJ94SSuVpYQUoTcGUIL6L4yNB7j1DFFHSBg=="
|
||||
}
|
||||
}
|
||||
},
|
||||
"balanced-match": {
|
||||
"version": "1.0.0",
|
||||
"resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz",
|
||||
|
@ -2566,6 +2586,11 @@
|
|||
}
|
||||
}
|
||||
},
|
||||
"clone": {
|
||||
"version": "2.1.2",
|
||||
"resolved": "https://registry.npmjs.org/clone/-/clone-2.1.2.tgz",
|
||||
"integrity": "sha1-G39Ln1kfHo+DZwQBYANFoCiHQ18="
|
||||
},
|
||||
"clone-deep": {
|
||||
"version": "4.0.1",
|
||||
"resolved": "https://registry.npmjs.org/clone-deep/-/clone-deep-4.0.1.tgz",
|
||||
|
@ -4822,6 +4847,11 @@
|
|||
"resolved": "https://registry.npmjs.org/heap/-/heap-0.2.6.tgz",
|
||||
"integrity": "sha1-CH4fELBGky/IWU3Z5tN4r8nR5aw="
|
||||
},
|
||||
"highlight.js": {
|
||||
"version": "9.18.1",
|
||||
"resolved": "https://registry.npmjs.org/highlight.js/-/highlight.js-9.18.1.tgz",
|
||||
"integrity": "sha512-OrVKYz70LHsnCgmbXctv/bfuvntIKDz177h0Co37DQ5jamGZLVmoCVMtjMtNZY3X9DrCcKfklHPNeA0uPZhSJg=="
|
||||
},
|
||||
"hmac-drbg": {
|
||||
"version": "1.0.1",
|
||||
"resolved": "https://registry.npmjs.org/hmac-drbg/-/hmac-drbg-1.0.1.tgz",
|
||||
|
@ -5643,6 +5673,14 @@
|
|||
"verror": "1.10.0"
|
||||
}
|
||||
},
|
||||
"katex": {
|
||||
"version": "0.6.0",
|
||||
"resolved": "https://registry.npmjs.org/katex/-/katex-0.6.0.tgz",
|
||||
"integrity": "sha1-EkGOCRIcBckgQbazuftrqyE8tvM=",
|
||||
"requires": {
|
||||
"match-at": "^0.1.0"
|
||||
}
|
||||
},
|
||||
"kind-of": {
|
||||
"version": "6.0.3",
|
||||
"resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz",
|
||||
|
@ -5719,6 +5757,14 @@
|
|||
"integrity": "sha1-HADHQ7QzzQpOgHWPe2SldEDZ/wA=",
|
||||
"dev": true
|
||||
},
|
||||
"linkify-it": {
|
||||
"version": "1.2.4",
|
||||
"resolved": "https://registry.npmjs.org/linkify-it/-/linkify-it-1.2.4.tgz",
|
||||
"integrity": "sha1-B3NSbDF8j9E71TTuHRgP+Iq/iBo=",
|
||||
"requires": {
|
||||
"uc.micro": "^1.0.1"
|
||||
}
|
||||
},
|
||||
"load-json-file": {
|
||||
"version": "2.0.0",
|
||||
"resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-2.0.0.tgz",
|
||||
|
@ -5984,6 +6030,85 @@
|
|||
"object-visit": "^1.0.0"
|
||||
}
|
||||
},
|
||||
"markdown-it": {
|
||||
"version": "6.1.1",
|
||||
"resolved": "https://registry.npmjs.org/markdown-it/-/markdown-it-6.1.1.tgz",
|
||||
"integrity": "sha1-ztA39Ec+6fUVOsQU933IPJG6knw=",
|
||||
"requires": {
|
||||
"argparse": "^1.0.7",
|
||||
"entities": "~1.1.1",
|
||||
"linkify-it": "~1.2.2",
|
||||
"mdurl": "~1.0.1",
|
||||
"uc.micro": "^1.0.1"
|
||||
}
|
||||
},
|
||||
"markdown-it-abbr": {
|
||||
"version": "1.0.4",
|
||||
"resolved": "https://registry.npmjs.org/markdown-it-abbr/-/markdown-it-abbr-1.0.4.tgz",
|
||||
"integrity": "sha1-1mtTZFIcuz3Yqlna37ovtoZcj9g="
|
||||
},
|
||||
"markdown-it-deflist": {
|
||||
"version": "2.0.3",
|
||||
"resolved": "https://registry.npmjs.org/markdown-it-deflist/-/markdown-it-deflist-2.0.3.tgz",
|
||||
"integrity": "sha512-/BNZ8ksW42bflm1qQLnRI09oqU2847Z7MVavrR0MORyKLtiUYOMpwtlAfMSZAQU9UCvaUZMpgVAqoS3vpToJxw=="
|
||||
},
|
||||
"markdown-it-emoji": {
|
||||
"version": "1.4.0",
|
||||
"resolved": "https://registry.npmjs.org/markdown-it-emoji/-/markdown-it-emoji-1.4.0.tgz",
|
||||
"integrity": "sha1-m+4OmpkKljupbfaYDE/dsF37Tcw="
|
||||
},
|
||||
"markdown-it-footnote": {
|
||||
"version": "2.0.0",
|
||||
"resolved": "https://registry.npmjs.org/markdown-it-footnote/-/markdown-it-footnote-2.0.0.tgz",
|
||||
"integrity": "sha1-FOnE9o/xLPNU+jZa43gnboEEypQ="
|
||||
},
|
||||
"markdown-it-ins": {
|
||||
"version": "2.0.0",
|
||||
"resolved": "https://registry.npmjs.org/markdown-it-ins/-/markdown-it-ins-2.0.0.tgz",
|
||||
"integrity": "sha1-papqMPHi9x6Ul1Z8/f9A8f3mdIM="
|
||||
},
|
||||
"markdown-it-katex": {
|
||||
"version": "2.0.3",
|
||||
"resolved": "https://registry.npmjs.org/markdown-it-katex/-/markdown-it-katex-2.0.3.tgz",
|
||||
"integrity": "sha1-17hqGuoLnWSW+rTnkZoY/e9YnDk=",
|
||||
"requires": {
|
||||
"katex": "^0.6.0"
|
||||
}
|
||||
},
|
||||
"markdown-it-mark": {
|
||||
"version": "2.0.0",
|
||||
"resolved": "https://registry.npmjs.org/markdown-it-mark/-/markdown-it-mark-2.0.0.tgz",
|
||||
"integrity": "sha1-RqGqlHEFrtgYiXjgoBYXnkBPQsc="
|
||||
},
|
||||
"markdown-it-sub": {
|
||||
"version": "1.0.0",
|
||||
"resolved": "https://registry.npmjs.org/markdown-it-sub/-/markdown-it-sub-1.0.0.tgz",
|
||||
"integrity": "sha1-N1/WAm6ufdywEkl/ZBEZXqHjr+g="
|
||||
},
|
||||
"markdown-it-sup": {
|
||||
"version": "1.0.0",
|
||||
"resolved": "https://registry.npmjs.org/markdown-it-sup/-/markdown-it-sup-1.0.0.tgz",
|
||||
"integrity": "sha1-y5yf+RpSVawI8/09YyhuFd8KH8M="
|
||||
},
|
||||
"markdown-it-task-lists": {
|
||||
"version": "2.1.1",
|
||||
"resolved": "https://registry.npmjs.org/markdown-it-task-lists/-/markdown-it-task-lists-2.1.1.tgz",
|
||||
"integrity": "sha512-TxFAc76Jnhb2OUu+n3yz9RMu4CwGfaT788br6HhEDlvWfdeJcLUsxk1Hgw2yJio0OXsxv7pyIPmvECY7bMbluA=="
|
||||
},
|
||||
"markdown-it-toc-and-anchor": {
|
||||
"version": "4.2.0",
|
||||
"resolved": "https://registry.npmjs.org/markdown-it-toc-and-anchor/-/markdown-it-toc-and-anchor-4.2.0.tgz",
|
||||
"integrity": "sha512-DusSbKtg8CwZ92ztN7bOojDpP4h0+w7BVOPuA3PHDIaabMsERYpwsazLYSP/UlKedoQjOz21mwlai36TQ04EpA==",
|
||||
"requires": {
|
||||
"clone": "^2.1.0",
|
||||
"uslug": "^1.0.4"
|
||||
}
|
||||
},
|
||||
"match-at": {
|
||||
"version": "0.1.1",
|
||||
"resolved": "https://registry.npmjs.org/match-at/-/match-at-0.1.1.tgz",
|
||||
"integrity": "sha512-h4Yd392z9mST+dzc+yjuybOGFNOZjmXIPKWjxBd1Bb23r4SmDOsk2NYCU2BMUBGbSpZqwVsZYNq26QS3xfaT3Q=="
|
||||
},
|
||||
"md5.js": {
|
||||
"version": "1.3.5",
|
||||
"resolved": "https://registry.npmjs.org/md5.js/-/md5.js-1.3.5.tgz",
|
||||
|
@ -5995,6 +6120,11 @@
|
|||
"safe-buffer": "^5.1.2"
|
||||
}
|
||||
},
|
||||
"mdurl": {
|
||||
"version": "1.0.1",
|
||||
"resolved": "https://registry.npmjs.org/mdurl/-/mdurl-1.0.1.tgz",
|
||||
"integrity": "sha1-/oWy7HWlkDfyrf7BAP1sYBdhFS4="
|
||||
},
|
||||
"mem": {
|
||||
"version": "4.3.0",
|
||||
"resolved": "https://registry.npmjs.org/mem/-/mem-4.3.0.tgz",
|
||||
|
@ -8345,8 +8475,7 @@
|
|||
"sprintf-js": {
|
||||
"version": "1.0.3",
|
||||
"resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz",
|
||||
"integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=",
|
||||
"dev": true
|
||||
"integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw="
|
||||
},
|
||||
"sshpk": {
|
||||
"version": "1.16.1",
|
||||
|
@ -9018,6 +9147,11 @@
|
|||
"integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=",
|
||||
"dev": true
|
||||
},
|
||||
"uc.micro": {
|
||||
"version": "1.0.6",
|
||||
"resolved": "https://registry.npmjs.org/uc.micro/-/uc.micro-1.0.6.tgz",
|
||||
"integrity": "sha512-8Y75pvTYkLJW2hWQHXxoqRgV7qb9B+9vFEtidML+7koHUFapnVJAZ6cKs+Qjz5Aw3aZWHMC6u0wJE3At+nSGwA=="
|
||||
},
|
||||
"unescape": {
|
||||
"version": "1.0.1",
|
||||
"resolved": "https://registry.npmjs.org/unescape/-/unescape-1.0.1.tgz",
|
||||
|
@ -9106,6 +9240,11 @@
|
|||
"integrity": "sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==",
|
||||
"dev": true
|
||||
},
|
||||
"unorm": {
|
||||
"version": "1.6.0",
|
||||
"resolved": "https://registry.npmjs.org/unorm/-/unorm-1.6.0.tgz",
|
||||
"integrity": "sha512-b2/KCUlYZUeA7JFUuRJZPUtr4gZvBh7tavtv4fvk4+KV9pfGiR6CQAQAWl49ZpR3ts2dk4FYkP7EIgDJoiOLDA=="
|
||||
},
|
||||
"unset-value": {
|
||||
"version": "1.0.0",
|
||||
"resolved": "https://registry.npmjs.org/unset-value/-/unset-value-1.0.0.tgz",
|
||||
|
@ -9217,6 +9356,14 @@
|
|||
}
|
||||
}
|
||||
},
|
||||
"uslug": {
|
||||
"version": "1.0.4",
|
||||
"resolved": "https://registry.npmjs.org/uslug/-/uslug-1.0.4.tgz",
|
||||
"integrity": "sha1-uaIvCRTgqGFAYz2swwLl9PpFBnc=",
|
||||
"requires": {
|
||||
"unorm": ">= 1.0.0"
|
||||
}
|
||||
},
|
||||
"util": {
|
||||
"version": "0.11.1",
|
||||
"resolved": "https://registry.npmjs.org/util/-/util-0.11.1.tgz",
|
||||
|
@ -9336,6 +9483,26 @@
|
|||
"vue-style-loader": "^4.1.0"
|
||||
}
|
||||
},
|
||||
"vue-markdown": {
|
||||
"version": "2.2.4",
|
||||
"resolved": "https://registry.npmjs.org/vue-markdown/-/vue-markdown-2.2.4.tgz",
|
||||
"integrity": "sha512-hoTX/W1UIdHZrp/b0vpHSsJXAEfWsafaQLgtE2VX4gY8O/C3L2Gabqu95gyG429rL4ML1SwGv+xsPABX7yfFIQ==",
|
||||
"requires": {
|
||||
"highlight.js": "^9.12.0",
|
||||
"markdown-it": "^6.0.1",
|
||||
"markdown-it-abbr": "^1.0.3",
|
||||
"markdown-it-deflist": "^2.0.1",
|
||||
"markdown-it-emoji": "^1.1.1",
|
||||
"markdown-it-footnote": "^2.0.0",
|
||||
"markdown-it-ins": "^2.0.0",
|
||||
"markdown-it-katex": "^2.0.3",
|
||||
"markdown-it-mark": "^2.0.0",
|
||||
"markdown-it-sub": "^1.0.0",
|
||||
"markdown-it-sup": "^1.0.0",
|
||||
"markdown-it-task-lists": "^2.0.1",
|
||||
"markdown-it-toc-and-anchor": "^4.1.2"
|
||||
}
|
||||
},
|
||||
"vue-prism-editor": {
|
||||
"version": "0.6.1",
|
||||
"resolved": "https://registry.npmjs.org/vue-prism-editor/-/vue-prism-editor-0.6.1.tgz",
|
||||
|
|
|
@ -26,8 +26,10 @@
|
|||
"@johmun/vue-tags-input": "^2.1.0",
|
||||
"@postlight/mercury-parser": "^2.2.0",
|
||||
"axios": "^0.19.0",
|
||||
"babel-runtime": "^6.26.0",
|
||||
"prismjs": "^1.20.0",
|
||||
"vue": "^2.6.10",
|
||||
"vue-markdown": "^2.2.4",
|
||||
"vue-prism-editor": "^0.6.1",
|
||||
"vuejs-auto-complete": "^0.9.0",
|
||||
"webextension-polyfill": "^0.3.1"
|
||||
|
|
|
@ -1,6 +1,8 @@
|
|||
import utils from './utils';
|
||||
import Message from './listeners/message';
|
||||
import Command from './listeners/command';
|
||||
import Connect from './listeners/connect';
|
||||
import Menu from './listeners/menu';
|
||||
import Message from './listeners/message';
|
||||
|
||||
global.browser = require('webextension-polyfill');
|
||||
|
||||
|
@ -35,7 +37,9 @@ const app = {
|
|||
}, {});
|
||||
},
|
||||
|
||||
prepareMenu() {
|
||||
async refreshMenu() {
|
||||
await browser.contextMenus.removeAll();
|
||||
|
||||
for (const [host] of Object.entries(this.hosts)) {
|
||||
const hostId = this.separator + host;
|
||||
browser.contextMenus.create({
|
||||
|
@ -64,19 +68,23 @@ const app = {
|
|||
}
|
||||
},
|
||||
|
||||
initListeners() {
|
||||
browser.contextMenus.onClicked.addListener(Menu.Listener);
|
||||
browser.runtime.onConnect.addListener(Connect.Listener);
|
||||
browser.runtime.onMessage.addListener(Message.Listener);
|
||||
browser.commands.onCommand.addListener(Command.Listener);
|
||||
},
|
||||
|
||||
async refresh() {
|
||||
this.hosts = await utils.methods.getHosts();
|
||||
this.actions = await utils.methods.getActions();
|
||||
this.scripts = await utils.methods.getScripts();
|
||||
await browser.contextMenus.removeAll();
|
||||
|
||||
this.prepareMenu();
|
||||
browser.contextMenus.onClicked.addListener(Menu.Listener);
|
||||
browser.runtime.onConnect.addListener(Message.Listener);
|
||||
await this.refreshMenu();
|
||||
},
|
||||
|
||||
async create() {
|
||||
await this.refresh();
|
||||
this.initListeners();
|
||||
},
|
||||
};
|
||||
|
||||
|
|
26
src/listeners/command.js
Normal file
26
src/listeners/command.js
Normal file
|
@ -0,0 +1,26 @@
|
|||
import utils from '../utils';
|
||||
|
||||
export default {
|
||||
async Listener(command) {
|
||||
const [commands, hosts, actions, scripts] = await Promise.all([utils.methods.getCommands(), utils.methods.getHosts(), utils.methods.getActions(), utils.methods.getScripts()]);
|
||||
|
||||
if (command in commands) {
|
||||
const actionName = commands[command];
|
||||
if (actionName in actions) {
|
||||
const action = actions[actionName];
|
||||
const host = hosts[Object.values(action.hosts)[0]];
|
||||
return await utils.methods.run(action, host);
|
||||
}
|
||||
|
||||
if (actionName in scripts) {
|
||||
const script = scripts[actionName];
|
||||
const host = hosts[Object.values(script.hosts)[0]];
|
||||
return await utils.methods.runScript(script.script, host);
|
||||
}
|
||||
|
||||
console.warn('No such action nor script', actionName);
|
||||
}
|
||||
},
|
||||
};
|
||||
|
||||
// vim:sw=2:ts=2:et:
|
139
src/listeners/connect.js
Normal file
139
src/listeners/connect.js
Normal file
|
@ -0,0 +1,139 @@
|
|||
import axios from 'axios';
|
||||
import Mercury from '@postlight/mercury-parser';
|
||||
import utils from '../utils';
|
||||
|
||||
const Service = (() => {
|
||||
const actionService = async port => {
|
||||
port.onMessage.addListener(async message => {
|
||||
let ret = null;
|
||||
switch (message.type) {
|
||||
case 'run':
|
||||
ret = await utils.methods.run(message.action, message.host);
|
||||
port.postMessage(ret);
|
||||
break;
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
const urlService = async port => {
|
||||
port.onMessage.addListener(async message => {
|
||||
const tab = await utils.methods.getCurrentTab();
|
||||
switch (message.type) {
|
||||
case 'get':
|
||||
port.postMessage(tab.url);
|
||||
break;
|
||||
|
||||
case 'set':
|
||||
await browser.tabs.sendMessage(tab.id, { type: 'setURL', url: message.url }, {});
|
||||
break;
|
||||
|
||||
case 'open':
|
||||
await browser.tabs.create({
|
||||
url: message.url,
|
||||
});
|
||||
break;
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
const domService = async port => {
|
||||
port.onMessage.addListener(async message => {
|
||||
const tab = await utils.methods.getCurrentTab();
|
||||
let dom = null;
|
||||
|
||||
switch (message.type) {
|
||||
case 'get':
|
||||
dom = await browser.tabs.sendMessage(tab.id, { type: 'getDOM' }, {});
|
||||
port.postMessage(dom);
|
||||
break;
|
||||
|
||||
case 'set':
|
||||
await browser.tabs.sendMessage(tab.id, { type: 'setDOM', html: message.html }, {});
|
||||
break;
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
const notifyService = async port => {
|
||||
port.onMessage.addListener(async message => {
|
||||
switch (message.type) {
|
||||
case 'run':
|
||||
await utils.methods.notify(message.message, message.title, message.error);
|
||||
break;
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
const commandService = async port => {
|
||||
port.onMessage.addListener(async message => {
|
||||
switch (message.type) {
|
||||
case 'get':
|
||||
const commands = await browser.commands.getAll();
|
||||
console.log('Available commands', commands);
|
||||
port.postMessage(commands);
|
||||
break;
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
const axiosService = async port => {
|
||||
port.onMessage.addListener(async message => {
|
||||
const method = axios[message.type.toLowerCase()];
|
||||
const response = await method(message.url, ...message.args);
|
||||
port.postMessage({
|
||||
config: {
|
||||
data: response.config.data,
|
||||
headers: response.config.headers,
|
||||
maxContentLength: response.config.maxContentLength,
|
||||
method: response.config.method,
|
||||
timeout: response.config.timeout,
|
||||
url: response.config.url,
|
||||
xsrfCookieName: response.config.xsrfCookieName,
|
||||
xsrfHeaderName: response.config.xsrfHeaderName,
|
||||
},
|
||||
headers: response.headers,
|
||||
data: response.data,
|
||||
status: response.status,
|
||||
statusText: response.statusText,
|
||||
});
|
||||
});
|
||||
};
|
||||
|
||||
const mercuryService = async port => {
|
||||
port.onMessage.addListener(async message => {
|
||||
let response = null;
|
||||
switch (message.type) {
|
||||
case 'parse':
|
||||
response = await Mercury.parse(message.url, {
|
||||
contentType: 'html',
|
||||
html: message.html,
|
||||
});
|
||||
|
||||
port.postMessage(response);
|
||||
break;
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
return {
|
||||
action: actionService,
|
||||
url: urlService,
|
||||
dom: domService,
|
||||
command: commandService,
|
||||
notify: notifyService,
|
||||
axios: axiosService,
|
||||
mercury: mercuryService,
|
||||
};
|
||||
})();
|
||||
|
||||
export default {
|
||||
async Listener(port) {
|
||||
if (port.name in Service) {
|
||||
await Service[port.name](port);
|
||||
} else {
|
||||
console.warn(`No such message service: {port.name}`);
|
||||
}
|
||||
},
|
||||
};
|
||||
|
||||
// vim:sw=2:ts=2:et:
|
|
@ -1,124 +1,20 @@
|
|||
import axios from 'axios';
|
||||
import Mercury from '@postlight/mercury-parser';
|
||||
import utils from '../utils';
|
||||
|
||||
const Service = (() => {
|
||||
const actionService = async port => {
|
||||
port.onMessage.addListener(async message => {
|
||||
let ret = null;
|
||||
switch (message.type) {
|
||||
case 'run':
|
||||
ret = await utils.methods.run(message.action, message.host);
|
||||
port.postMessage(ret);
|
||||
break;
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
const urlService = async port => {
|
||||
port.onMessage.addListener(async message => {
|
||||
const tab = await utils.methods.getCurrentTab();
|
||||
switch (message.type) {
|
||||
case 'get':
|
||||
port.postMessage(tab.url);
|
||||
break;
|
||||
|
||||
case 'set':
|
||||
await browser.tabs.sendMessage(tab.id, { type: 'setURL', url: message.url }, {});
|
||||
break;
|
||||
|
||||
case 'open':
|
||||
await browser.tabs.create({
|
||||
url: message.url,
|
||||
});
|
||||
break;
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
const domService = async port => {
|
||||
port.onMessage.addListener(async message => {
|
||||
const tab = await utils.methods.getCurrentTab();
|
||||
let dom = null;
|
||||
|
||||
switch (message.type) {
|
||||
case 'get':
|
||||
dom = await browser.tabs.sendMessage(tab.id, { type: 'getDOM' }, {});
|
||||
port.postMessage(dom);
|
||||
break;
|
||||
|
||||
case 'set':
|
||||
await browser.tabs.sendMessage(tab.id, { type: 'setDOM', html: message.html }, {});
|
||||
break;
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
const notifyService = async port => {
|
||||
port.onMessage.addListener(async message => {
|
||||
switch (message.type) {
|
||||
case 'run':
|
||||
await utils.methods.notify(message.message, message.title, message.error);
|
||||
break;
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
const axiosService = async port => {
|
||||
port.onMessage.addListener(async message => {
|
||||
const method = axios[message.type.toLowerCase()];
|
||||
const response = await method(message.url, ...message.args);
|
||||
port.postMessage({
|
||||
config: {
|
||||
data: response.config.data,
|
||||
headers: response.config.headers,
|
||||
maxContentLength: response.config.maxContentLength,
|
||||
method: response.config.method,
|
||||
timeout: response.config.timeout,
|
||||
url: response.config.url,
|
||||
xsrfCookieName: response.config.xsrfCookieName,
|
||||
xsrfHeaderName: response.config.xsrfHeaderName,
|
||||
},
|
||||
headers: response.headers,
|
||||
data: response.data,
|
||||
status: response.status,
|
||||
statusText: response.statusText,
|
||||
});
|
||||
});
|
||||
};
|
||||
|
||||
const mercuryService = async port => {
|
||||
port.onMessage.addListener(async message => {
|
||||
let response = null;
|
||||
switch (message.type) {
|
||||
case 'parse':
|
||||
response = await Mercury.parse(message.url, {
|
||||
contentType: 'html',
|
||||
html: message.html,
|
||||
});
|
||||
|
||||
port.postMessage(response);
|
||||
break;
|
||||
}
|
||||
});
|
||||
const getCommands = async (message, sender, sendResponse) => {
|
||||
const commands = await browser.commands.getAll();
|
||||
sendResponse(commands);
|
||||
};
|
||||
|
||||
return {
|
||||
action: actionService,
|
||||
url: urlService,
|
||||
dom: domService,
|
||||
notify: notifyService,
|
||||
axios: axiosService,
|
||||
mercury: mercuryService,
|
||||
getCommands: getCommands,
|
||||
};
|
||||
})();
|
||||
|
||||
export default {
|
||||
async Listener(port) {
|
||||
if (port.name in Service) {
|
||||
await Service[port.name](port);
|
||||
async Listener(message, sender, sendResponse) {
|
||||
if (message.type in Service) {
|
||||
await Service[message.type](message, sender, sendResponse);
|
||||
} else {
|
||||
console.warn(`No such message service: {port.name}`);
|
||||
console.warn('No such handled message type', message.type);
|
||||
}
|
||||
},
|
||||
};
|
||||
|
|
|
@ -30,5 +30,67 @@
|
|||
"matches": ["*://*/*"],
|
||||
"js": ["content.js"]
|
||||
}
|
||||
]
|
||||
],
|
||||
"commands": {
|
||||
"user-command-0": {
|
||||
"suggested_key": {
|
||||
"default": "Ctrl+Alt+0"
|
||||
},
|
||||
"description": "User command 0"
|
||||
},
|
||||
"user-command-1": {
|
||||
"suggested_key": {
|
||||
"default": "Ctrl+Alt+1"
|
||||
},
|
||||
"description": "User command 1"
|
||||
},
|
||||
"user-command-2": {
|
||||
"suggested_key": {
|
||||
"default": "Ctrl+Alt+2"
|
||||
},
|
||||
"description": "User command 2"
|
||||
},
|
||||
"user-command-3": {
|
||||
"suggested_key": {
|
||||
"default": "Ctrl+Alt+3"
|
||||
},
|
||||
"description": "User command 3"
|
||||
},
|
||||
"user-command-4": {
|
||||
"suggested_key": {
|
||||
"default": "Ctrl+Alt+4"
|
||||
},
|
||||
"description": "User command 4"
|
||||
},
|
||||
"user-command-5": {
|
||||
"suggested_key": {
|
||||
"default": "Ctrl+Alt+5"
|
||||
},
|
||||
"description": "User command 5"
|
||||
},
|
||||
"user-command-6": {
|
||||
"suggested_key": {
|
||||
"default": "Ctrl+Alt+6"
|
||||
},
|
||||
"description": "User command 6"
|
||||
},
|
||||
"user-command-7": {
|
||||
"suggested_key": {
|
||||
"default": "Ctrl+Alt+7"
|
||||
},
|
||||
"description": "User command 7"
|
||||
},
|
||||
"user-command-8": {
|
||||
"suggested_key": {
|
||||
"default": "Ctrl+Alt+8"
|
||||
},
|
||||
"description": "User command 8"
|
||||
},
|
||||
"user-command-9": {
|
||||
"suggested_key": {
|
||||
"default": "Ctrl+Alt+9"
|
||||
},
|
||||
"description": "User command 9"
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -16,8 +16,8 @@
|
|||
|
||||
<div v-if="actionMode === 'request'">
|
||||
<div class="help">
|
||||
<a href="https://platypush.readthedocs.io/en/latest/plugins.html" target="_blank">Plugins reference</a>. Use <tt>$URL$</tt> as argument value to denote the current
|
||||
URL. You can also call remotely stored procedure through <tt>procedure.<procedure_name></tt>.
|
||||
<a href="https://platypush.readthedocs.io/en/latest/plugins.html" target="_blank">Plugins reference</a>. Use <code>$URL$</code> as argument value to denote the
|
||||
current URL. You can also call remotely stored procedure through <code>procedure.<procedure_name></code>.
|
||||
</div>
|
||||
|
||||
<form ref="runForm" @submit.prevent="runAction">
|
||||
|
@ -28,12 +28,14 @@
|
|||
:source="actionsAutocomplete"
|
||||
:disableInput="loading"
|
||||
:name="action.name || ''"
|
||||
:initialValue="selectedAction ? selectedAction.name : null"
|
||||
:initialDisplay="selectedAction ? selectedAction.name : null"
|
||||
:initialValue="selectedAction ? selectedAction.action : null"
|
||||
:initialDisplay="selectedAction ? selectedAction.action : null"
|
||||
@input="onActionChange"
|
||||
/>
|
||||
</div>
|
||||
<div class="action-doc" v-text="actionTemplate.doc" v-if="actionTemplate.doc" />
|
||||
<div class="action-doc" v-if="actionTemplate.doc">
|
||||
<vue-markdown :source="actionTemplate.doc" v-if="actionTemplate.doc" />
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="row" v-for="(arg, name) in action.defaultArgs" :key="name">
|
||||
|
@ -96,7 +98,7 @@
|
|||
</form>
|
||||
</div>
|
||||
|
||||
<form class="save-form" @submit.prevent="save" v-if="saveMode">
|
||||
<form class="save-form" ref="saveForm" @submit.prevent="save" v-if="saveMode">
|
||||
<div class="row">
|
||||
<input type="text" name="displayName" v-model="saveParams.name" placeholder="Display name" />
|
||||
</div>
|
||||
|
@ -133,6 +135,14 @@
|
|||
<MultipleHostSelector :hosts="hosts" :selected="selectedHosts && selectedHosts.length ? selectedHosts : [host.name]" />
|
||||
</div>
|
||||
|
||||
<div class="row command-selector" v-if="Object.keys(commands).length">
|
||||
<label for="_command-selector">Key binding associated to this action</label>
|
||||
<select id="_command-selector" v-model="command">
|
||||
<option value="">-- No key binding</option>
|
||||
<option v-for="command in commands" :key="command.name" :value="command.name" v-text="command.shortcut" />
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<div class="row buttons">
|
||||
<button type="submit" :disabled="loading"><i class="fas fa-save" /> Save {{ actionMode === 'request' ? 'Action' : 'Script' }}</button>
|
||||
<button type="button" @click="toggleSaveMode" :disabled="loading"><i class="fas fa-times" /> Cancel</button>
|
||||
|
@ -148,6 +158,7 @@
|
|||
import 'prismjs';
|
||||
import 'prismjs/themes/prism.css';
|
||||
import PrismEditor from 'vue-prism-editor';
|
||||
import VueMarkdown from 'vue-markdown';
|
||||
|
||||
import mixins from '../utils';
|
||||
import Autocomplete from 'vuejs-auto-complete';
|
||||
|
@ -181,6 +192,7 @@ export default {
|
|||
VueTagsInput,
|
||||
MultipleHostSelector,
|
||||
PrismEditor,
|
||||
VueMarkdown,
|
||||
},
|
||||
|
||||
data() {
|
||||
|
@ -198,6 +210,8 @@ export default {
|
|||
selectedCategories: [],
|
||||
selectedHosts: null,
|
||||
actionMode: 'request',
|
||||
commands: {},
|
||||
command: '',
|
||||
action: {
|
||||
name: null,
|
||||
args: [],
|
||||
|
@ -290,6 +304,17 @@ export default {
|
|||
}, {});
|
||||
},
|
||||
|
||||
async updateCommands() {
|
||||
try {
|
||||
this.commands = (await browser.commands.getAll()).reduce((obj, command) => {
|
||||
obj[command.name] = command;
|
||||
return obj;
|
||||
}, {});
|
||||
} catch (e) {
|
||||
console.log('Could not get configured commands', e);
|
||||
}
|
||||
},
|
||||
|
||||
async runAction() {
|
||||
this.loading = true;
|
||||
|
||||
|
@ -362,7 +387,24 @@ export default {
|
|||
this.storedActions = await this.getActions();
|
||||
},
|
||||
|
||||
async getCommand() {
|
||||
const action = this.saveParams.name.trim();
|
||||
const commands = await this.getCommands();
|
||||
const [command] = Object.entries(commands).filter(([, act]) => act === action)[0] || [];
|
||||
return command;
|
||||
},
|
||||
|
||||
async save(event) {
|
||||
const action = event.target.displayName.value.trim();
|
||||
if (this.command && this.command.length) {
|
||||
await this.saveCommand(this.command, action);
|
||||
} else {
|
||||
const command = await this.getCommand();
|
||||
if (command) {
|
||||
await this.removeCommand(command);
|
||||
}
|
||||
}
|
||||
|
||||
return this.actionMode === 'request' ? await this.storeAction(event) : await this.storeScript(event);
|
||||
},
|
||||
|
||||
|
@ -374,12 +416,12 @@ export default {
|
|||
const hosts = [...saveForm.querySelectorAll('input[data-type="host"]:checked')].map(el => el.value);
|
||||
|
||||
if (!displayName.length) {
|
||||
this.notify('Please specify an action name', 'No action name provided');
|
||||
await this.notify('Please specify an action name', 'No action name provided');
|
||||
return;
|
||||
}
|
||||
|
||||
if (!hosts.length) {
|
||||
this.notify('Please specify at least one device where the action should run', 'No devices provided');
|
||||
await this.notify('Please specify at least one device where the action should run', 'No devices provided');
|
||||
return;
|
||||
}
|
||||
|
||||
|
@ -404,12 +446,12 @@ export default {
|
|||
const hosts = [...saveForm.querySelectorAll('input[data-type="host"]:checked')].map(el => el.value);
|
||||
|
||||
if (!displayName.length) {
|
||||
this.notify('Please specify an action name', 'No action name provided');
|
||||
await this.notify('Please specify an action name', 'No action name provided');
|
||||
return;
|
||||
}
|
||||
|
||||
if (!hosts.length) {
|
||||
this.notify('Please specify at least one device where the action should run', 'No devices provided');
|
||||
await this.notify('Please specify at least one device where the action should run', 'No devices provided');
|
||||
return;
|
||||
}
|
||||
|
||||
|
@ -453,7 +495,7 @@ export default {
|
|||
this.selectedCategories = tags;
|
||||
},
|
||||
|
||||
initAction() {
|
||||
async initAction() {
|
||||
const action = this.selectedAction || this.selectedScript;
|
||||
if (!action) {
|
||||
return;
|
||||
|
@ -465,10 +507,11 @@ export default {
|
|||
this.saveParams.iconClass = action.iconClass;
|
||||
this.selectedCategories = action.categories.map(cat => (typeof cat === 'string' ? { text: cat } : cat));
|
||||
this.selectedHosts = action.hosts;
|
||||
this.command = await this.getCommand();
|
||||
|
||||
if (this.selectedAction) {
|
||||
this.actionMode = 'request';
|
||||
this.action.name = action.name;
|
||||
this.action.name = action.action;
|
||||
this.action.defaultArgs = Object.entries(action.args).reduce((obj, [name, value]) => {
|
||||
obj[name] = { value: value };
|
||||
return obj;
|
||||
|
@ -486,6 +529,7 @@ export default {
|
|||
this.loadPlugins();
|
||||
this.loadActions();
|
||||
this.initAction();
|
||||
this.updateCommands();
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
@ -524,13 +568,15 @@ export default {
|
|||
.action-doc {
|
||||
width: 40em;
|
||||
margin: 0.5em auto auto 1em;
|
||||
white-space: pre;
|
||||
border: 1px solid rgba(0, 0, 0, 0.15);
|
||||
border-radius: 1em;
|
||||
padding: 1em;
|
||||
max-height: 10em;
|
||||
overflow: auto;
|
||||
background: rgba(250, 255, 240, 0.5);
|
||||
|
||||
* {
|
||||
margin: 1em;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -606,6 +652,12 @@ form {
|
|||
font-size: 2em;
|
||||
margin-left: 0.2em;
|
||||
}
|
||||
|
||||
.command-selector {
|
||||
label {
|
||||
margin-right: 1em;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
|
||||
<!-- vim:sw=2:ts=2:et: -->
|
||||
|
|
114
src/utils.js
114
src/utils.js
|
@ -10,6 +10,10 @@ export default {
|
|||
},
|
||||
|
||||
methods: {
|
||||
getExtensionId() {
|
||||
return browser.i18n.getMessage('@@extension_id');
|
||||
},
|
||||
|
||||
async notify(message, title = 'platypush', error = false) {
|
||||
let msg = '';
|
||||
if (title && title.length) {
|
||||
|
@ -39,10 +43,17 @@ export default {
|
|||
},
|
||||
|
||||
async getCurrentTab() {
|
||||
const tabs = await browser.tabs.query({
|
||||
currentWindow: true,
|
||||
active: true,
|
||||
});
|
||||
let tabs = [];
|
||||
|
||||
try {
|
||||
tabs = await browser.tabs.query({
|
||||
currentWindow: true,
|
||||
active: true,
|
||||
});
|
||||
} catch (e) {
|
||||
console.warn('Could not get active tab', e);
|
||||
return;
|
||||
}
|
||||
|
||||
if (!tabs.length) {
|
||||
await this.notify('No active tab', '', true);
|
||||
|
@ -54,22 +65,41 @@ export default {
|
|||
|
||||
async getURL() {
|
||||
const tab = await this.getCurrentTab();
|
||||
return await browser.tabs.sendMessage(tab.id, { type: 'getURL' });
|
||||
try {
|
||||
return await browser.tabs.sendMessage(tab.id, { type: 'getURL' });
|
||||
} catch (e) {
|
||||
console.warn('Could not get URL', e);
|
||||
}
|
||||
},
|
||||
|
||||
async getDOM() {
|
||||
const tab = await this.getCurrentTab();
|
||||
return await browser.tabs.sendMessage(tab.id, { type: 'getDOM' });
|
||||
try {
|
||||
const tab = await this.getCurrentTab();
|
||||
return await browser.tabs.sendMessage(tab.id, { type: 'getDOM' });
|
||||
} catch (e) {
|
||||
console.warn('Could not get DOM', e);
|
||||
}
|
||||
},
|
||||
|
||||
async setDOM(html) {
|
||||
const tab = await this.getCurrentTab();
|
||||
await browser.tabs.sendMessage(tab.id, { type: 'setDOM', html: html });
|
||||
try {
|
||||
await browser.tabs.sendMessage(tab.id, { type: 'setDOM', html: html });
|
||||
} catch (e) {
|
||||
console.warn('Could not set DOM', e);
|
||||
}
|
||||
},
|
||||
|
||||
async getTargetElement() {
|
||||
const tab = await this.getCurrentTab();
|
||||
const target = await browser.tabs.sendMessage(tab.id, { type: 'getTargetElement' });
|
||||
let target = null;
|
||||
try {
|
||||
target = await browser.tabs.sendMessage(tab.id, { type: 'getTargetElement' });
|
||||
} catch (e) {
|
||||
console.warn('Could not get current element', e);
|
||||
return;
|
||||
}
|
||||
|
||||
if (!target) {
|
||||
return;
|
||||
}
|
||||
|
@ -91,14 +121,14 @@ export default {
|
|||
|
||||
if (Array.isArray(action.args)) {
|
||||
args = action.args
|
||||
.filter(arg => arg.value != null && arg.value.length)
|
||||
.filter(arg => arg.value != null && (typeof arg.value !== 'string' || arg.value.length))
|
||||
.reduce((obj, arg) => {
|
||||
obj[arg.name] = arg.value;
|
||||
return obj;
|
||||
}, {});
|
||||
} else {
|
||||
args = Object.entries(args)
|
||||
.filter(([, value]) => value != null && value.length)
|
||||
.filter(([, value]) => value != null && (typeof value !== 'string' || value.length))
|
||||
.reduce((obj, [name, value]) => {
|
||||
obj[name] = value;
|
||||
return obj;
|
||||
|
@ -295,15 +325,72 @@ export default {
|
|||
await this.notify('You can find this script under the Local Actions menu', 'Script saved');
|
||||
},
|
||||
|
||||
async getCommands() {
|
||||
this.loading = true;
|
||||
|
||||
try {
|
||||
const response = await browser.storage.local.get('commands');
|
||||
if (!response.commands) {
|
||||
return {};
|
||||
}
|
||||
|
||||
return JSON.parse(response.commands);
|
||||
} finally {
|
||||
this.loading = false;
|
||||
}
|
||||
},
|
||||
|
||||
async saveCommands(commands) {
|
||||
this.loading = true;
|
||||
try {
|
||||
await browser.storage.local.set({ commands: JSON.stringify(commands) });
|
||||
} finally {
|
||||
this.loading = false;
|
||||
}
|
||||
},
|
||||
|
||||
async saveCommand(command, action) {
|
||||
const commands = await this.getCommands();
|
||||
if (command in commands) {
|
||||
if (action === commands[command]) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (!confirm(`The action ${commands[command]} is already linked to this key binding. Do you want to overwrite it?`)) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
Object.entries(commands).forEach(([cmd, act]) => {
|
||||
if (act === action) {
|
||||
delete commands[cmd];
|
||||
}
|
||||
});
|
||||
|
||||
commands[command] = action;
|
||||
await this.saveCommands(commands);
|
||||
},
|
||||
|
||||
async removeCommand(command, action) {
|
||||
const commands = await this.getCommands();
|
||||
if (!(command in commands) || !confirm('Are you sure that you want to remove this key binding?')) {
|
||||
return;
|
||||
}
|
||||
|
||||
delete commands[command];
|
||||
await this.saveCommands(commands);
|
||||
},
|
||||
|
||||
async loadConfig() {
|
||||
this.loading = true;
|
||||
|
||||
try {
|
||||
const [hosts, actions, scripts] = await Promise.all([this.getHosts(), this.getActions(), this.getScripts(false)]);
|
||||
const [hosts, actions, scripts, commands] = await Promise.all([this.getHosts(), this.getActions(), this.getScripts(false), this.getCommands()]);
|
||||
return {
|
||||
hosts: hosts,
|
||||
actions: actions,
|
||||
scripts: scripts,
|
||||
commands: commands,
|
||||
};
|
||||
} finally {
|
||||
this.loading = false;
|
||||
|
@ -315,9 +402,10 @@ export default {
|
|||
const hosts = config.hosts || {};
|
||||
const actions = config.actions || {};
|
||||
const scripts = config.scripts || {};
|
||||
const commands = config.commands || {};
|
||||
|
||||
try {
|
||||
await Promise.all([this.saveHosts(hosts), this.saveActions(actions), this.saveScripts(scripts)]);
|
||||
await Promise.all([this.saveHosts(hosts), this.saveActions(actions), this.saveScripts(scripts), this.saveCommands(commands)]);
|
||||
} finally {
|
||||
this.loading = false;
|
||||
}
|
||||
|
|
Loading…
Add table
Reference in a new issue