diff --git a/package.json b/package.json
index 3b186abf8d..1975bef152 100644
--- a/package.json
+++ b/package.json
@@ -42,7 +42,7 @@
"remix-analyzer": "0.3.1",
"remix-debug": "0.3.1",
"remix-lib": "0.4.1",
- "remix-plugin": "0.0.1-alpha.2",
+ "remix-plugin": "0.0.1-alpha.4",
"remix-solidity": "0.3.1",
"remix-tests": "0.1.1",
"remixd": "0.1.8-alpha.6",
diff --git a/src/app.js b/src/app.js
index d08ae3fdaf..051330aea5 100644
--- a/src/app.js
+++ b/src/app.js
@@ -36,7 +36,6 @@ var toolTip = require('./app/ui/tooltip')
var TransactionReceiptResolver = require('./transactionReceiptResolver')
const PluginManagerComponent = require('./app/components/plugin-manager-component')
-const PluginManagerApi = require('./app/components/plugin-manager-api')
const VerticalIconsComponent = require('./app/components/vertical-icons-component')
const VerticalIconsApi = require('./app/components/vertical-icons-api')
@@ -44,7 +43,17 @@ const VerticalIconsApi = require('./app/components/vertical-icons-api')
const SwapPanelComponent = require('./app/components/swap-panel-component')
const SwapPanelApi = require('./app/components/swap-panel-api')
-const AppManager = require('remix-plugin').AppManager
+const CompileTab = require('./app/tabs/compile-tab')
+const SettingsTab = require('./app/tabs/settings-tab')
+const AnalysisTab = require('./app/tabs/analysis-tab')
+const DebuggerTab = require('./app/tabs/debugger-tab')
+const SupportTab = require('./app/tabs/support-tab')
+const TestTab = require('./app/tabs/test-tab')
+const RunTab = require('./app/tabs/run-tab')
+const FilePanel = require('./app/panels/file-panel')
+
+import { EntityStore } from './lib/store'
+import { RemixAppManager } from './remixAppManager'
var styleGuide = require('./app/ui/styles-guide/theme-chooser')
var styles = styleGuide.chooser()
@@ -208,7 +217,7 @@ class App {
profile () {
return {
- type: 'app',
+ name: 'App',
description: 'the app',
methods: ['getExecutionContextProvider', 'getProviderEndpoint', 'detectNetWork', 'addProvider', 'removeProvider']
}
@@ -407,7 +416,7 @@ Please make a backup of your contracts and start using http://remix.ethereum.org
event: new EventEmitter(),
profile () {
return {
- type: 'txListener',
+ name: 'TxListener',
events: ['newTransaction']
}
}
@@ -442,41 +451,74 @@ Please make a backup of your contracts and start using http://remix.ethereum.org
- the current API is not optimal. For instance methods of `app` only refers to `executionContext`, wich does not make really sense.
*/
- const appManager = new AppManager({modules: [], plugins: []})
+ // TODOs those are instanciated before hand. should be instanciated on demand
- const swapPanelComponent = new SwapPanelComponent()
- const pluginManagerComponent = new PluginManagerComponent(
- {
- app: this,
- udapp: udapp,
- fileManager: fileManager,
- sourceHighlighters: registry.get('editor').api.sourceHighlighters,
- config: self._components.filesProviders['config'],
- txListener: txListenerModuleProxy
- })
+ const pluginManagerComponent = new PluginManagerComponent()
registry.put({api: pluginManagerComponent.proxy(), name: 'pluginmanager'})
- self._components.editorpanel.init()
- self._components.fileManager.init()
+ let filePanel = new FilePanel()
+ registry.put({api: filePanel, name: 'filepanel'})
+
+ let compileTab = new CompileTab(self._components.registry)
+ let run = new RunTab(self._components.registry)
+ let settings = new SettingsTab(self._components.registry)
+ let analysis = new AnalysisTab(self._components.registry)
+ let debug = new DebuggerTab(self._components.registry)
+ let support = new SupportTab(self._components.registry)
+ let test = new TestTab(self._components.registry, compileTab)
+
+ let appStore = new EntityStore('module', { actives: [], ids: [], entities: {} }, 'name')
+
+ const appManager = new RemixAppManager(appStore)
+
+ pluginManagerComponent.setApp(appManager)
+ pluginManagerComponent.setStore(appStore)
+
+ let sourceHighlighters = registry.get('editor').api.sourceHighlighters
+ let configProvider = self._components.filesProviders['config']
+ appManager.init([
+ { profile: this.profile(), api: this },
+ { profile: udapp.profile(), api: udapp },
+ { profile: fileManager.profile(), api: fileManager },
+ { profile: sourceHighlighters.profile(), api: sourceHighlighters },
+ { profile: configProvider.profile(), api: configProvider },
+ { profile: txListenerModuleProxy.profile(), api: txListenerModuleProxy },
+ { profile: compileTab.profile(), api: compileTab },
+ { profile: filePanel.profile(), api: filePanel },
+ { profile: test.profile(), api: test },
+ { profile: support.profile(), api: support },
+ { profile: debug.profile(), api: debug },
+ { profile: analysis.profile(), api: analysis },
+ { profile: settings.profile(), api: settings },
+ { profile: run.profile(), api: run },
+ { profile: pluginManagerComponent.profile(), api: pluginManagerComponent }])
+ const swapPanelComponent = new SwapPanelComponent()
const verticalIconComponent = new VerticalIconsComponent()
+ const swapPanelApi = new SwapPanelApi(swapPanelComponent, verticalIconComponent, appManager)
+ const verticalIconsApi = new VerticalIconsApi(verticalIconComponent, appManager)
- const swapPanelApi = new SwapPanelApi(swapPanelComponent, verticalIconComponent, pluginManagerComponent)
- const verticalIconsApi = new VerticalIconsApi(verticalIconComponent, pluginManagerComponent)
- const pluginManagerAPI = new PluginManagerApi(pluginManagerComponent)
-
+ self._components.editorpanel.init()
+ self._components.fileManager.init()
+
self._view.mainpanel.appendChild(self._components.editorpanel.render())
self._view.iconpanel.appendChild(verticalIconComponent.render())
self._view.swappanel.appendChild(swapPanelComponent.render())
- pluginManagerComponent.initDefault()
- verticalIconComponent.select('FilePanel')
- // appManager.init(pluginManagerAPI)
+ appManager.doActivate('App')
+ appManager.doActivate('Udapp')
+ appManager.doActivate('FileManager')
+ appManager.doActivate('SourceHighlighters')
+ appManager.doActivate('config')
+ appManager.doActivate('TxListener')
+ appManager.doActivate('FilePanel')
+ appManager.doActivate('SolidityCompile')
+ appManager.doActivate('Run')
+ appManager.doActivate('PluginManager')
+ appManager.doActivate('Settings')
+ appManager.doActivate('Support')
- pluginManagerAPI.init({
- modules: [],
- plugins: []
- })
+ verticalIconComponent.select('FilePanel')
// The event listener needs to be registered as early as possible, because the
// parent will send the message upon the "load" event.
diff --git a/src/app/components/plugin-manager-api.js b/src/app/components/plugin-manager-api.js
deleted file mode 100644
index 66acb32242..0000000000
--- a/src/app/components/plugin-manager-api.js
+++ /dev/null
@@ -1,21 +0,0 @@
-// const remixLib = require('remix-lib')
-
-class PluginManagerApi {
- constructor (pluginManagerComponent) {
- this.component = pluginManagerComponent
- pluginManagerComponent.event.on('activation', (item) => this.event.emit('activation', item)) // listen by appManager
- pluginManagerComponent.event.on('deactivation', (item) => this.event.emit('deactivation', item)) // listen by appManager
- }
-
- init (data) {
- for (var m in data.modules) {
- this.renderItem(m)
- }
-
- for (var n in data.plugins) {
- this.renderItem(n)
- }
- }
-}
-
-module.exports = PluginManagerApi
diff --git a/src/app/components/plugin-manager-component.js b/src/app/components/plugin-manager-component.js
index 53963013f9..cc4be780c7 100644
--- a/src/app/components/plugin-manager-component.js
+++ b/src/app/components/plugin-manager-component.js
@@ -14,6 +14,7 @@ var registry = require('../../global/registry')
const styleguide = require('../ui/styles-guide/theme-chooser')
const styles = styleguide.chooser()
+import { EntityStore } from '../../lib/store.js'
const PluginManagerProxy = require('./plugin-manager-proxy')
@@ -21,147 +22,73 @@ const EventEmitter = require('events')
class PluginManagerComponent {
- constructor ({ app, udapp, fileManager, sourceHighlighters, config, txListener }) {
+ constructor () {
this.event = new EventEmitter()
- this.modulesDefinition = {
- // service module. They can be seen as daemon
- // they usually don't have UI and only represent the minimal API a plugins can access.
- 'App': { name: 'App', target: app },
- 'Udapp': { name: 'Udapp', target: udapp },
- 'FileManager': { name: 'FileManager', target: fileManager },
- 'SourceHighlighters': { name: 'SourceHighlighters', target: sourceHighlighters },
- 'Config': { name: 'Config', target: config },
- 'TxListener': { name: 'TxListener', target: txListener },
- // internal components. They are mostly views, they don't provide external API for plugins
- 'SolidityCompile': { name: 'SolidityCompile', description: 'Compile functionality for solidity', class: 'evm-compiler', Type: CompileTab, icon: '' },
- 'FilePanel': { name: 'FilePanel', Type: FilePanel, description: 'Load files from a variety of sources', icon: '' },
- 'Test': { name: 'Test', dep: 'SolidityCompile', Type: TestTab, description: 'Test it', icon: '' },
- 'Run': { name: 'Run', Type: RunTab, description: 'Deploy & run a contract with access to a contracts methods', icon: '' },
- 'SolidityStaticAnalysis': { name: 'SolidityStaticAnalysis', Type: AnalysisTab, description: 'To see the static analysis',icon: '' },
- 'Debugger': { name: 'Debugger', Type: DebuggerTab, description: 'Just make it work!', icon: '' },
- 'Support': { name: 'Support', Type: SupportTab, description: 'Go here for support', icon: '' },
- 'PluginManager': { name: 'PluginManager', target: this, description: 'Mangement of Plugins and Modules', icon: '' },
- 'Settings': { name: 'Settings', Type: SettingsTab, description: 'The module for settings', icon: '' }
+ this.data = {
+ proxy: new PluginManagerProxy()
+ }
+ this.views = {
+ root: null,
+ items: {}
}
- this.activated = {} // list all activated modules
- // this.activated = [] // list all activated modules
- this.plugins = []
- this.data = {}
- this.data.proxy = new PluginManagerProxy()
-
- // This load the state from localstorage first then from the second parameter is nothing is found in localstorage
- // this.store = Store.fromLocal('***', {}) // Or EntityStore.fromLocal('***', {}, 'id')
}
- proxy () {
- return this.data.proxy
+ profile () {
+ return {
+ name: 'PluginManager',
+ methods: [],
+ events: [],
+ icon: ''
+ }
+ }
+
+ setApp (appManager) {
+ this.appManager = appManager
+ }
+
+ setStore (store) {
+ this.store = store
+ this.store.event.on('activate', (name) => { this.views.items[name] ? this.views.items[name].querySelector('button').innerHTML = 'deactivate' : null })
+ this.store.event.on('deactivate', (name) => { this.views.items[name] ? this.views.items[name].querySelector('button').innerHTML = 'activate' : null })
}
- initDefault () {
- this.activateInternal('App')
- this.activateInternal('Udapp')
- this.activateInternal('FileManager')
- this.activateInternal('SourceHighlighters')
- this.activateInternal('Config')
- this.activateInternal('TxListener')
-
- this.activateInternal('FilePanel')
- this.activateInternal('SolidityCompile')
- this.activateInternal('Run')
- this.activateInternal('PluginManager')
- this.activateInternal('Settings')
- this.activateInternal('Support')
+ proxy () {
+ return this.data.proxy
}
render () {
- // var self = this
- // loop over all this.modules and this.plugins
- let pluginManagerDiv = yo`
+ this.views.root = yo`
Plugin Manager
`
- for (var mod in this.modulesDefinition) {
- if (this.modulesDefinition[mod].icon) {
- pluginManagerDiv.appendChild(this.renderItem(mod))
- }
- }
- console.log(this.activated)
- return pluginManagerDiv
+ var modules = this.store.getAll()
+ modules.forEach((mod) => {
+ this.views.root.appendChild(this.renderItem(mod.profile.name))
+ })
+ return this.views.root
}
renderItem (item) {
let ctrBtns
- let action = (event) => {
- if (this.activated.hasOwnProperty(item)) {
- this.deactivateInternal(item)
- event.target.innerHTML = 'activate'
- } else {
- this.activateInternal(item)
- event.target.innerHTML = 'deactivate'
- }
- }
+ const mod = this.store.getOne(item)
+ if (!mod) return
+ let action = () => { this.store.isActive(item) ? this.appManager.doDeactivate(item) : this.appManager.doActivate(item) }
ctrBtns = yo`
-
+
`
-
- this.plugins.push(item)
- // var self = this
- this.view = yo`
+
+ this.views.items[item] = yo`
-
${this.modulesDefinition[item].name}
- ${this.modulesDefinition[item].description}
+ ${mod.profile.name}
+ ${mod.profile.description}
${ctrBtns}
`
- return this.view
- }
-
- activatePlugin (jsonProfile, api) {
- // let profile = { json: jsonProfile, api }
- // let plugin = new Plugin(profile, api)
- // this.appManager.addPlugin(plugin)
- // this.event.emit('displayableModuleActivated', jsonProfile, plugin.render())
- // this.activated[jsonProfile.name] = plugin
- }
-
- deactivateInternal (name) {
- if (!this.activated[name]) return
- this.event.emit('removingItem', this.activated[name])
- delete this.activated[name]
- }
-
- activateInternal (name) {
- if (this.activated[name]) return
- const mod = this.modulesDefinition[name]
- let dep
- if (mod.dep) dep = this.activateInternal(mod.dep)
- let instance = mod.target
- if (!instance && mod.Type) instance = new mod.Type(registry, dep)
- if (!instance) return console.log(`PluginManagerComponent: no Type or instance to add: ${JSON.stringify(mod)}`)
- registry.put({api: instance, name: mod.name.toLocaleLowerCase()})
- if (instance.profile && typeof instance.profile === 'function') {
- this.event.emit('requestActivation', instance.profile(), instance)
- }
- if (mod.icon && instance.render && typeof instance.render === 'function') {
- this.event.emit('requestContainer', mod, instance.render())
- }
- // if of type evm-compiler, we forward to the internal components
- if (mod.class === 'evm-compiler') {
- this.data.proxy.register(mod, instance)
- }
- this.activated[mod.name] = mod
- return instance
- }
-
- _activate (item) {
- this.event.emit('activation', item)
- }
-
- _deactivate (item) {
- this.event.emit('deactivation', item)
+
+ return this.views.items[item]
}
}
diff --git a/src/app/components/swap-panel-api.js b/src/app/components/swap-panel-api.js
index eb128b17c6..bae2df24aa 100644
--- a/src/app/components/swap-panel-api.js
+++ b/src/app/components/swap-panel-api.js
@@ -1,15 +1,15 @@
// const EventEmmitter = require('events')
class SwapPanelApi {
- constructor (swapPanelComponent, verticalIconsComponent, pluginManagerComponent) {
+ constructor (swapPanelComponent, verticalIconsComponent, appManager) {
this.component = swapPanelComponent
verticalIconsComponent.event.on('showContent', (moduleName) => {
this.component.showContent(moduleName)
})
- pluginManagerComponent.event.on('requestContainer', (mod, content) => {
+ appManager.event.on('requestContainer', (mod, content) => {
this.add(mod.name, content)
})
- pluginManagerComponent.event.on('removingItem', (mod) => {
+ appManager.event.on('removingItem', (mod) => {
this.remove(mod.name)
})
}
diff --git a/src/app/components/vertical-icons-api.js b/src/app/components/vertical-icons-api.js
index a167dab0c0..6710e510c0 100644
--- a/src/app/components/vertical-icons-api.js
+++ b/src/app/components/vertical-icons-api.js
@@ -1,10 +1,10 @@
// API
class VerticalIconsApi {
- constructor (verticalIconsComponent, pluginManagerComponent) {
+ constructor (verticalIconsComponent, appManager) {
this.component = verticalIconsComponent
- pluginManagerComponent.event.on('requestContainer', (mod, content) => verticalIconsComponent.addIcon(mod))
- pluginManagerComponent.event.on('removingItem', (mod) => verticalIconsComponent.removeIcon(mod))
+ appManager.event.on('requestContainer', (mod, content) => verticalIconsComponent.addIcon(mod))
+ appManager.event.on('removingItem', (mod) => verticalIconsComponent.removeIcon(mod))
}
}
module.exports = VerticalIconsApi
diff --git a/src/app/editor/SourceHighlighters.js b/src/app/editor/SourceHighlighters.js
index 8902235fe5..7c53ac19d7 100644
--- a/src/app/editor/SourceHighlighters.js
+++ b/src/app/editor/SourceHighlighters.js
@@ -9,7 +9,7 @@ module.exports = class SourceHighlighters {
profile () {
return {
- type: 'sourcehighlighter',
+ name: 'SourceHighlighters',
methods: ['highlight', 'discardHighlight']
}
}
diff --git a/src/app/files/browser-files-tree.js b/src/app/files/browser-files-tree.js
index b527d4cd71..54678e5110 100644
--- a/src/app/files/browser-files-tree.js
+++ b/src/app/files/browser-files-tree.js
@@ -133,7 +133,7 @@ function FilesTree (name, storage) {
this.profile = function () {
return {
- type: this.type,
+ name: this.type,
methods: ['get', 'set', 'remove']
}
}
diff --git a/src/app/files/fileManager.js b/src/app/files/fileManager.js
index a32ef613c8..8f8e23d4bf 100644
--- a/src/app/files/fileManager.js
+++ b/src/app/files/fileManager.js
@@ -52,7 +52,7 @@ class FileManager {
profile () {
return {
- type: 'fileManager',
+ name: 'FileManager',
methods: ['getFilesFromPath', 'getCurrentFile', 'getFile', 'setFile'],
events: ['currentFileChanged']
}
diff --git a/src/app/panels/editor-panel.js b/src/app/panels/editor-panel.js
index 6a8fda6d07..87d6b45335 100644
--- a/src/app/panels/editor-panel.js
+++ b/src/app/panels/editor-panel.js
@@ -16,6 +16,7 @@ class EditorPanel {
constructor (localRegistry) {
var self = this
self.event = new EventManager()
+ self._view = {}
self._components = {}
self._components.registry = localRegistry || globalRegistry
self._components.editor = new Editor({})
@@ -39,7 +40,6 @@ class EditorPanel {
}
}
}
- self._view = {}
var contextualListener = new ContextualListener({editor: self._components.editor, pluginManager: self._deps.pluginManager})
var contextView = new ContextView({contextualListener, editor: self._components.editor})
diff --git a/src/app/panels/file-panel.js b/src/app/panels/file-panel.js
index 49433f1065..31296769b6 100644
--- a/src/app/panels/file-panel.js
+++ b/src/app/panels/file-panel.js
@@ -196,6 +196,15 @@ function filepanel (localRegistry) {
self.render = function render () { return element }
+ self.profile = function () {
+ return {
+ name: 'FilePanel',
+ methods: [],
+ events: [],
+ icon: ''
+ }
+ }
+
function uploadFile (event) {
// TODO The file explorer is merely a view on the current state of
// the files module. Please ask the user here if they want to overwrite
diff --git a/src/app/tabs/analysis-tab.js b/src/app/tabs/analysis-tab.js
index f2d3b8d6bd..5635dece63 100644
--- a/src/app/tabs/analysis-tab.js
+++ b/src/app/tabs/analysis-tab.js
@@ -15,6 +15,14 @@ module.exports = class AnalysisTab {
self._components.registry = localRegistry || globalRegistry
self._deps = {}
}
+ profile () {
+ return {
+ name: 'SolidityAnalysis',
+ methods: [],
+ events: [],
+ icon: ''
+ }
+ }
render () {
const self = this
var staticanalysis = new StaticAnalysis()
diff --git a/src/app/tabs/compile-tab.js b/src/app/tabs/compile-tab.js
index 262a78a4f1..bc16ee6f48 100644
--- a/src/app/tabs/compile-tab.js
+++ b/src/app/tabs/compile-tab.js
@@ -56,8 +56,7 @@ module.exports = class CompileTab {
renderer: self._components.registry.get('renderer').api,
swarmfileProvider: self._components.registry.get('fileproviders/swarm').api,
fileManager: self._components.registry.get('filemanager').api,
- fileProviders: self._components.registry.get('fileproviders').api,
- pluginManager: self._components.registry.get('pluginmanager').api
+ fileProviders: self._components.registry.get('fileproviders').api
}
self.data = {
hideWarnings: self._deps.config.get('hideWarnings') || false,
@@ -187,9 +186,10 @@ module.exports = class CompileTab {
}
profile () {
return {
- type: 'solidityCompile',
+ name: 'SolidityCompile',
methods: ['getCompilationResult'],
- events: ['compilationFinished']
+ events: ['compilationFinished'],
+ icon: ''
}
}
addWarning (msg, settings) {
diff --git a/src/app/tabs/debugger-tab.js b/src/app/tabs/debugger-tab.js
index 94737cec03..7754bb7808 100644
--- a/src/app/tabs/debugger-tab.js
+++ b/src/app/tabs/debugger-tab.js
@@ -27,6 +27,15 @@ class DebuggerTab {
self._components.registry = localRegistry || globalRegistry
}
+ profile () {
+ return {
+ name: 'Debugger',
+ methods: [],
+ events: [],
+ icon: ''
+ }
+ }
+
render () {
const self = this
if (self._view.el) return self._view.el
diff --git a/src/app/tabs/run-tab.js b/src/app/tabs/run-tab.js
index d624fc894d..3c6ede3f62 100644
--- a/src/app/tabs/run-tab.js
+++ b/src/app/tabs/run-tab.js
@@ -178,7 +178,17 @@ function runTab (opts, localRegistry) {
`
container.appendChild(el)
- return { render () { return container } }
+ return {
+ render () { return container },
+ profile () {
+ return {
+ name: 'Run',
+ methods: [],
+ events: [],
+ icon: ''
+ }
+ }
+ }
}
module.exports = runTab
diff --git a/src/app/tabs/settings-tab.js b/src/app/tabs/settings-tab.js
index 525cd33c39..e05b6fd565 100644
--- a/src/app/tabs/settings-tab.js
+++ b/src/app/tabs/settings-tab.js
@@ -38,6 +38,14 @@ module.exports = class SettingsTab {
self._components.themeStorage = new Storage('style:')
self.data.currentTheme = self._components.themeStorage.get('theme') || 'light'
}
+ profile () {
+ return {
+ name: 'Settings',
+ methods: [],
+ events: [],
+ icon: ''
+ }
+ }
render () {
const self = this
if (self._view.el) return self._view.el
diff --git a/src/app/tabs/support-tab.js b/src/app/tabs/support-tab.js
index ea7367d71a..eabf86866c 100644
--- a/src/app/tabs/support-tab.js
+++ b/src/app/tabs/support-tab.js
@@ -28,6 +28,14 @@ module.exports = class SupportTab {
self.data.gitterIsLoaded = true
})
}
+ profile () {
+ return {
+ name: 'Support',
+ methods: [],
+ events: [],
+ icon: ''
+ }
+ }
render () {
const self = this
if (self._view.el) return self._view.el
diff --git a/src/app/tabs/test-tab.js b/src/app/tabs/test-tab.js
index bb85ba84a2..12edc63358 100644
--- a/src/app/tabs/test-tab.js
+++ b/src/app/tabs/test-tab.js
@@ -23,6 +23,14 @@ module.exports = class TestTab {
self.data = {}
self.testList = yo``
}
+ profile () {
+ return {
+ name: 'Run',
+ methods: [],
+ events: [],
+ icon: ''
+ }
+ }
render () {
const self = this
var testsOutput = yo``
diff --git a/src/lib/store.js b/src/lib/store.js
new file mode 100644
index 0000000000..d0ba59722f
--- /dev/null
+++ b/src/lib/store.js
@@ -0,0 +1,250 @@
+import { EventEmitter } from 'events'
+
+export class Store {
+
+ /**
+ * Instanciate the store from `localStorage` first
+ * @template T
+ * @param {string} name The name of the store
+ * @param {T} initialState The initial state used if state is not available in `localStorage`
+ */
+ static fromLocal(name, initialState) {
+ const fromLocal = localStorage.getItem(name)
+ const intial = fromLocal ? JSON.parse(fromLocal) : initialState
+ return new Store(name, intial)
+ }
+
+ /**
+ * Create a Store that hold the state of the component
+ * @template T
+ * @param {string} name The name of the store
+ * @param {T} initialState The initial state of the store
+ */
+ constructor(name, initialState) {
+ this.event = new EventEmitter()
+ this.name = name
+ this.state = initialState
+ }
+
+ /** Listen on event from the store */
+ get on() {
+ return this.event.on;
+ }
+
+ /** Liste once on event from the store */
+ get once() {
+ return this.event.once;
+ }
+
+ /**
+ * Update one field of the state
+ * @param {Partial} state The part of the state updated
+ */
+ /*
+ update(state) {
+ this.state = { ...this.state, ...state }
+ }
+ */
+
+ /**
+ * Get one field of the state
+ * @template Key key of `this.state`
+ * @param {Key} key A key of the state
+ */
+ get(key) {
+ return this.state.entities[key]
+ }
+
+ /** Reset the state its initial value */
+ reset() {
+ this.state = this.initialState
+ }
+
+ /** Dispatch an event with the new state */
+ dispatch() {
+ this.event.emit('newState', this.state)
+ }
+}
+
+/**
+ * An entity inside a collection
+ * @typedef {Object} EntityState
+ * @property {string[]} ids The list of IDs of the entity in the state
+ * @property {string[]} actives The list of IDs of the activated entities
+ * @property {Object} entities A map of ids and entities
+ */
+
+
+
+export class EntityStore extends Store {
+
+ /**
+ * Instanciate the store from `localStorage` first
+ * @param {string} name The name of the store
+ * @param {EntityState} initialState The initial state used if state is not available in `localStorage`
+ */
+ static fromLocal(name, initialState) {
+ const fromLocal = localStorage.getItem(name)
+ const intial = fromLocal ? JSON.parse(fromLocal) : initialState
+ return new EntityStore(name, intial)
+ }
+
+ /**
+ * Create a entity Store that hold a map entity of the same model
+ * @param {string} name The name of the store
+ * @param {EntityState} initialState The initial state used if state is not available in `localStorage`
+ */
+ constructor(name, initialState) {
+ super(name, initialState)
+ }
+
+ ////////////
+ // GETTER //
+ ////////////
+
+ /** Tne entities as a Map */
+ get entities() {
+ return this.state.entities
+ }
+
+ /** List of all the ids */
+ get ids() {
+ return this.state.ids
+ }
+
+ /** List of all active ID */
+ get actives() {
+ return this.state.actives
+ }
+
+ /** Return the length of the entity collection */
+ get length() {
+ return this.state.ids.length
+ }
+
+ /////////////
+ // SETTERS //
+ /////////////
+
+ /**
+ * Add a new entity to the state
+ * @param {Object} entity
+ */
+ add(id, entity) {
+ this.state.entities[id] = entity
+ this.state.ids.push(id)
+ }
+
+ /**
+ * Remove an entity from the state
+ * @param {(string|number)} id The id of the entity to remove
+ */
+ remove(id) {
+ delete this.state.entities[id]
+ this.state.ids.splice(this.state.ids.indexOf(id), 1)
+ this.state.actives.splice(this.state.ids.indexOf(id), 1)
+ }
+
+ /**
+ * Update one entity of the state
+ * @param {(string|number)} id The id of the entity to update
+ * @param {Object} update The fields to update in the entity
+ */
+ /*
+ updateOne(id, update) {
+ this.state.entities[id] = {
+ ...this.state.entities[id],
+ ...update
+ }
+ }
+ */
+
+ /**
+ * Activate one or several entity from the state
+ * @param {((string|number))} ids An id or a list of id to activate
+ */
+ activate(id) {
+ this.state.actives.push(id)
+ this.event.emit('activate', id)
+ }
+
+ /**
+ * Deactivate one or several entity from the state
+ * @param {(string|number))} ids An id or a list of id to deactivate
+ */
+ deactivate(id) {
+ this.state.actives.splice(this.state.actives.indexOf(id), 1)
+ this.event.emit('deactivate', id)
+ }
+
+ ///////////
+ // QUERY //
+ ///////////
+
+ /**
+ * Get one entity
+ * @param {(string|number)} id The id of the entity to get
+ */
+ getOne(id) {
+ return this.state.entities[id]
+ }
+
+ /**
+ * Get many entities as an array
+ * @param {(string|number)[]} ids An array of id of entity to get
+ */
+ getMany(ids) {
+ return ids.map(id => this.state.entities[id])
+ }
+
+ /** Get all the entities as an array */
+ getAll() {
+ return this.state.ids.map(id => this.state.entities[id])
+ }
+
+ /** Get all active entities */
+ getActives() {
+ return this.state.actives.map(id => this.state.entities[id])
+ }
+
+ ////////////////
+ // CONDITIONS //
+ ////////////////
+
+ /**
+ * Is the entity active
+ * @param {(string|number)} id The id of the entity to check
+ */
+ isActive(id) {
+ return this.state.actives.includes(id)
+ }
+
+ /**
+ * Is this id inside the store
+ * @param {(string|number)} id The id of the entity to check
+ */
+ hasEntity(id) {
+ return this.state.ids.includes(id)
+ }
+
+ /**
+ * Is the state empty
+ * @param {(string|number)} id The id of the entity to check
+ */
+ isEmpty() {
+ return this.state.ids.length === 0
+ }
+}
+
+/**
+ * Store the state of the stores into LocalStorage
+ * @param {Store[]} stores The list of stores to store into `localStorage`
+ */
+function localState(stores) {
+ stores.forEach(store => {
+ const name = store.name
+ store.on('newState', (state) => {
+ localStorage.setItem(name, JSON.stringify(state))
+ })
+ })
+}
diff --git a/src/remixAppManager.js b/src/remixAppManager.js
new file mode 100644
index 0000000000..0c6f27b4d8
--- /dev/null
+++ b/src/remixAppManager.js
@@ -0,0 +1,45 @@
+import { AppManagerApi } from 'remix-plugin'
+import { EventEmitter } from 'events'
+
+export class RemixAppManager extends AppManagerApi {
+
+ constructor (store) {
+ super(null)
+ this.store = store
+ this.event = new EventEmitter()
+ }
+
+ doActivate (name) {
+ this.activateOne(name)
+ // temp
+ this.store.activate(name)
+ // promise ?
+ const entity = this.getEntity(name)
+ if (entity.profile.icon && entity.api.render && typeof entity.api.render === 'function') {
+ this.event.emit('requestContainer', entity.profile, entity.api.render())
+ }
+ }
+
+ doDeactivate (name) {
+ this.deactivateOne(name)
+ // temp
+ this.store.deactivate(name)
+ // promise ?
+ const entity = this.getEntity(name)
+ if (entity.profile.icon && entity.api.render && typeof entity.api.render === 'function') {
+ this.event.emit('removingItem', entity.profile)
+ }
+ }
+
+ setActive (name, isActive) {
+ isActive ? this.store.activate(name) : this.store.deactivate(name)
+ }
+
+ getEntity (entityName) {
+ return this.store.get(entityName)
+ }
+
+ addEntity (entity) {
+ this.store.add(entity.profile.name, entity)
+ }
+}
\ No newline at end of file
diff --git a/src/universal-dapp.js b/src/universal-dapp.js
index 2262b576ac..aa6afa5300 100644
--- a/src/universal-dapp.js
+++ b/src/universal-dapp.js
@@ -31,14 +31,7 @@ function UniversalDApp (registry) {
UniversalDApp.prototype.profile = function () {
return {
- type: 'udapp',
- methods: ['runTestTx', 'getAccounts', 'createVMAccount']
- }
-}
-
-UniversalDApp.prototype.profile = function () {
- return {
- type: 'udapp',
+ name: 'Udapp',
methods: ['runTestTx', 'getAccounts', 'createVMAccount']
}
}