-
-
Notifications
You must be signed in to change notification settings - Fork 161
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #1162 from SignalK/dynamic-component-loading
feature: embeddable plugin/webapp components
- Loading branch information
Showing
40 changed files
with
1,814 additions
and
350 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,25 +1,73 @@ | ||
## Webapps | ||
# Webapps and components | ||
|
||
### 1. Introduction | ||
## Introduction | ||
|
||
A Signal K webapp is an HTML5 application that runs on a web browser and is typically loaded from a Signal K server, using its HTTP interface. | ||
There are four ways to add web-based user interfaces to the Signal K Server: | ||
|
||
### 2. Sample webapps | ||
- Standalone Webapps | ||
- Embedded Webapps | ||
- Embedded Plugin Configuration Forms | ||
- Embedded Compontents | ||
|
||
There are several prototype level [browser based webapps that are installed during the server installation](http://localhost:3000/apps) such as: | ||
- [Instrumentpanel](https://github.com/SignalK/instrumentpanel) - Show your boat's live data with customizable gauges you can hide or move around. | ||
- [freeboard-sk](https://github.com/signalk/freeboard-sk) - Chart plotting webpage | ||
- [sailgauge](https://github.com/signalk/sailgauge) - html page to show sailing-related data on one screen. | ||
**Standalone Webapps** are Webapps that can be installed on the server via Appstore. The server provides a list of installed webapps. Once you navigate to them the server admin UI disappears and the webapp controls the whole page (browser window / tab). | ||
|
||
Other apps are published on npm, and appear in the App Store (http://localhost:3000/appstore) | ||
**Embedded Webapps** are installed and listed like Standalone Webapps, but they open **embedded in the server admin UI**, leaving the header and footer available so that the user can perform login, restart server and use the admin UI's sidebar to navigate to a different part of the admin UI. | ||
|
||
### 3. Getting started with your own webapps | ||
![vesselpositions](img/vesselpositions.png?raw=true "Vesselpositions Embedded Webapp") | ||
|
||
The simplest form of a Signal K client is a process or web page that receives and displays the data in some way. The server contains a very simple web page that connects to the server with a WebSocket connection and updates the display as new json messages are received from the server. If you have installed the server and started it with one of the file-based startup scripts in the bin/ directory you can access it at [http://localhost:3000/examples/consumer-example.html](http://localhost:3000/examples/consumer-example.html). The [html code of the web page](https://github.com/SignalK/signalk-server-node/blob/master/public/examples/consumer-example.html) is in the examples directory. | ||
**Embedded Plugin Configuration Forms** are related to server plugins. A plugin provides a schema for the configuration data it uses. The server uses the schema - a description of the structure of the data used to configure the plugin - to generate configuration forms for the installed plugins. The generated form is often lackin in usability due to it being totally generic. To address this a plugin can provide its own **Configuration Form** that the server embeds within the Plugin Configuration of the server admin UI. | ||
|
||
If you want to integrate your webapp in the main structure of http://localhost:3000/apps, you should make it installable via npm and add the keyword `signalk-webapp` to your package.json file. | ||
![calibration](img/calibration.png?raw=true "Calibration plugin configuration form") | ||
|
||
**Embedded Components** are individual UI components provided by a plugin or a webapp. They are currently available at the bottom of the Webapps page of the admin UI. The idea with embedding components would be to allow a plugin to add individual components to different parts of the server, but this is more an idea than a fully implemented feature at this stage. | ||
|
||
### 4. Going further | ||
## Webapp/Component Structure | ||
|
||
All different webapps (and server plugins) are installed with npm, from npm registry or for example from your own Github repository. Private plugins need not be published to npm - see the documentation for [npm install](https://docs.npmjs.com/cli/v6/commands/npm-install) for the exact details. Only webapps that are relevant for all users should be published in npm to be available in App store of everybody's server. | ||
|
||
The basic structure of a webapp is | ||
- directory named `public` that contains the actual webapp: html, JavaScript and resources such as images and css files. This directory is automatically mounted by the server so that the webapp is available via http once installed | ||
- package.json with special keywords that classifies the webapp | ||
- `signalk-webapp` - standalone webapp | ||
- `signalk-embeddable-webapp` - embeddable webapp | ||
- `signalk-plugin-configurator` - plugin configuration form | ||
|
||
This structure is all that is needed for a standalone webapp. | ||
|
||
There is no keyword for a module that provides only embedded components, use `signalk-webapp` instead. | ||
|
||
The embedded components are implemented using [Webpack Federated Modules](https://webpack.js.org/concepts/module-federation/) and [React Code Splitting](https://reactjs.org/docs/code-splitting.html). | ||
|
||
You need to configured Webpack to create the necessary code for federation using *ModuleFederationPlugin* and expose the component with fixed names: | ||
- embeddable webapp: `./AppPanel` | ||
- plugin configuration form: `./PluginConfigurationPanel` | ||
- embedded component: `./AddonPanel` | ||
|
||
The ModuleFederationPlugin library name must match the package name and be a "safe" name for a webpack module like in `library: { type: 'var', name: packageJson.name.replace(/[-@/]/g, '_') },` | ||
|
||
The exposed modules need to `export default` a React component - both class based components and stateless functional components can be used. The server dependencies like `reactstrap` can and should be used. Add `@signalk/server-admin-ui-dependencies` as a dependency to the webapp, it defines the depedencies used by the server admin UI. | ||
|
||
See the vesselpositions embedded webapp/component and Calibration plugin for examples of each. It is probably easier to start with either one and modify them to suit your needs. Don't forget to change the module id and name in package.json! | ||
|
||
|
||
## Webapp / Component and Admin UI / Server interfaces | ||
|
||
Standalone Webapps can use the server's APIs (Standard Signal K http and WebSocket APIS as well as the server's endpoints) but they need to implement everything themselves. | ||
|
||
Embedded Webapps, Components and Plugin Configuration Forms work inside the Admin UI and they can interact with the Admin UI and the server with APIs exposed by the Admin UI as component properties. | ||
|
||
This documentation is rudimentary on purpose, as the details need to be worked out. | ||
|
||
Embedded webapp properties | ||
- access to the login status of the browser user | ||
- ability to render Login form instead of the webapp content | ||
- getting and setting application data | ||
- opening an automatically reconnecting WebSocket connection to the server | ||
- getting Signal K data via `get` | ||
- [Embedded](packages/server-admin-ui/src/views/Webapps/Embedded.js) | ||
|
||
PluginConfigurationForm properties | ||
- `configuration` : the configuration data of the plugin | ||
- `save`: function to save the configuration data | ||
- [EmbeddedPluginConfigurationForm](packages/server-admin-ui/src/views/Configuration/EmbeddedPluginConfigurationForm.js) | ||
|
||
Have an idea for a great way to visualise Signal K data? Something bugging you about the sample consumers? [Read the documentation](https://signalk.org/specification/master/), share your input, write some code, [join the mailing list](https://groups.google.com/forum/#!forum/signalk) and our [Slack chat](http://slack-invite.signalk.org/)! |
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,30 @@ | ||
|
||
require('validate-peer-dependencies')(__dirname, { | ||
resolvePeerDependenciesFrom: './', | ||
handleFailure(result) { | ||
let { missingPeerDependencies, incompatibleRanges } = result; | ||
|
||
let missingWithVersions = (missingPeerDependencies || []).reduce( | ||
(message, metadata) => { | ||
return `${message}${metadata.name}@${metadata.specifiedPeerDependencyRange} `; | ||
}, | ||
'' | ||
); | ||
|
||
let incompatiblePeerDependenciesMessage = (incompatibleRanges || []).reduce( | ||
(message, metadata) => { | ||
return `${message}\n\t* ${metadata.name}: \`${metadata.specifiedPeerDependencyRange}\`; it was resolved to \`${metadata.version}\``; | ||
}, | ||
'' | ||
); | ||
|
||
if (missingWithVersions.length >0) { | ||
console.error((`Please INSTALL MISSING PEERDEPENDENCIES with | ||
npm install --save-dev ${missingWithVersions}`)) | ||
process.exit(-1) | ||
} | ||
|
||
console.error((`Incompatible peerDependencies: ${incompatiblePeerDependenciesMessage}`)) | ||
process.exit(-1) | ||
}, | ||
}) |
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,24 @@ | ||
{ | ||
"name": "@signalk/server-admin-ui-dependencies", | ||
"version": "1.0.0", | ||
"description": "UI dependencies of the admin UI exposed to plugins that expose federated components", | ||
"main": "index.js", | ||
"author": "[email protected]", | ||
"license": "Apache-2.0", | ||
"optionalDependencies": { | ||
"validate-peer-dependencies": "^1.1.0" | ||
}, | ||
"peerDependencies": { | ||
"@fortawesome/fontawesome-free": "^5.15.1", | ||
"bootstrap": "^4.5.3", | ||
"font-awesome": "^4.7.0", | ||
"react": "^16.13.1", | ||
"react-dom": "^16.13.1", | ||
"react-redux": "^5.1.2", | ||
"react-select": "^3.1.0", | ||
"reactstrap": "^5.0.0", | ||
"redux": "^3.7.2", | ||
"redux-thunk": "^2.3.0", | ||
"simple-line-icons": "^2.5.5" | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.