/* global localStorage, fetch */ import { PluginManager } from '@remixproject/engine' import { IframePlugin } from '@remixproject/engine-web' import { EventEmitter } from 'events' import QueryParams from './lib/query-params' import { PermissionHandler } from './app/ui/persmission-handler' const requiredModules = [ // services + layout views + system views 'manager', 'compilerArtefacts', 'compilerMetadata', 'contextualListener', 'editor', 'offsetToLineColumnConverter', 'network', 'theme', 'fileManager', 'contentImport', 'web3Provider', 'scriptRunner', 'fetchAndCompile', 'mainPanel', 'hiddenPanel', 'sidePanel', 'menuicons', 'fileExplorers', 'terminal', 'settings', 'pluginManager'] export function isNative (name) { const nativePlugins = ['vyper', 'workshops', 'debugger'] return nativePlugins.includes(name) || requiredModules.includes(name) } export function canActivate (name) { return ['ethdoc'].includes(name) || isNative(name) } export class RemixAppManager extends PluginManager { constructor () { super() this.event = new EventEmitter() this.pluginsDirectory = 'https://raw.githubusercontent.com/ethereum/remix-plugins-directory/master/build/metadata.json' this.pluginLoader = new PluginLoader() this.permissionHandler = new PermissionHandler() } async canActivatePlugin (from, to) { return canActivate(from.name) } async canDeactivatePlugin (from, to) { if (requiredModules.includes(to.name)) return false return from.name === 'manager' } async canCall (from, to, method, message) { // Make sure the caller of this methods is the target plugin if (to !== this.currentRequest.from) { return false } // skipping native plugins' requests if (isNative(from)) { return true } // ask the user for permission return await this.permissionHandler.askPermission(this.profiles[from], this.profiles[to], method, message) } onPluginActivated (plugin) { this.pluginLoader.set(plugin, this.actives) this.event.emit('activate', plugin) } getAll () { return Object.keys(this.profiles).map((p) => { return this.profiles[p] }) } getIds () { return Object.keys(this.profiles) } onPluginDeactivated (plugin) { this.pluginLoader.set(plugin, this.actives) this.event.emit('deactivate', plugin) } onRegistration () {} async ensureActivated (apiName) { await this.activatePlugin(apiName) this.event.emit('ensureActivated', apiName) } async ensureDeactivated (apiName) { await this.deactivatePlugin(apiName) this.event.emit('ensureDeactivated', apiName) } isRequired (name) { return requiredModules.includes(name) } async registeredPlugins () { const res = await fetch(this.pluginsDirectory) const plugins = await res.json() plugins.push({ name: 'walletconnect', kind: 'provider', displayName: 'Wallet Connect', events: [], methods: ['sendAsync'], url: 'ipfs://QmUD93rF9RKaDabCM5jFTHraBKd72HApbv7m9Vd5i5EHZe', description: 'Use an external wallet for transacting', icon: '', location: 'mainPanel' }) return plugins.map(plugin => { if (plugin.name === 'scriptRunner') { plugin.url = 'ipfs://QmYCUgBadbcHj3B8d8TsT4A2fKYvXGFWt1izxyXasjdWtF' } return new IframePlugin(plugin) }) } } /** @class Reference loaders. * A loader is a get,set based object which load a workspace from a defined sources. * (localStorage, queryParams) **/ class PluginLoader { get currentLoader () { return this.loaders[this.current] } constructor () { const queryParams = new QueryParams() this.donotAutoReload = ['remixd'] // that would be a bad practice to force loading some plugins at page load. this.loaders = {} this.loaders.localStorage = { set: (plugin, actives) => { if (!this.donotAutoReload.includes(plugin.name)) { localStorage.setItem('workspace', JSON.stringify(actives)) } }, get: () => { return JSON.parse(localStorage.getItem('workspace')) } } this.loaders.queryParams = { set: () => {}, get: () => { const { activate } = queryParams.get() if (!activate) return [] return activate.split(',') } } this.current = queryParams.get().activate ? 'queryParams' : 'localStorage' } set (plugin, actives) { this.currentLoader.set(plugin, actives) } get () { return this.currentLoader.get() } }