Everything you need to know about this creating or modifying the custom RoboPaint interaction layer we call a "mode"!
A mode for RoboPaint shows up as a bubble on the home screen, and a tool bar tab over every other mode and the home screen. These modes are complete web applications with a base HTML file that is loaded into a controller sub-window Electron WebView by RoboPaint, and loads shared JS/CSS, or anything else you like.
The "core" modes are edit
and paint
, with more coming as we think of them.
Each mode is intended to interact with the API in its own unique way, and
therefore allows an infinite set of functionality to piggyback on the existing
featureset and connectivity within RoboPaint.
Modes have three minimum required files: package.json
, a root
.html
file configured within it, and a base translation file under _i18n/
.
That's it! Everything else that deals with how the page works is within the
package.json
and your html file.
|mymode/
|- package.json
|- mymode.html
|- mymode.css
|- mymode.js
|- _i18n/
|--- mymode.en-US.json
|--- mymode.fr.json
|-- ...
We recommend simply copying one of the existing modes that comes with RoboPaint, and renaming the folder and the other files. RoboPaint supports full translation, so we also highly recommend designing your app to take full advantage of this, even if you don't have translations of your text strings.
Though not exactly a node module, we take a cue from NPM and use the "open"
format for package.json
to hold all top level mode configuration. See the
comments below for explanation on the non-obvious keys. A mode is added via
RoboPaint package.json dependency or through direct npm install
.
{
"name": "robopaint-mode-example", // Universal namespace ID for NPM, should be in this format!
"version": "0.8.0",
"robopaint-type": "mode", // If not equal to this, this package will be ignored.
"main": "example.js", // Used to inject your JS into the page at the correct time during preload. No need to include it in the HTML!
"author": "techninja",
"license": "MIT",
"robopaint": { // All RoboPaint specific details should live here
"core": false, // If true, will be available by default. Otherwise it must be turned on manually.
"debug": true, // If true, the modes devtools window will open by default.
"i18n": "native", // The type of translation this mode supports. Either DOM or native, see below for details.
"index": "example.html", // The path to the HTML file to open relative to this folder
"name": "example", // The machine name of the mode. Lowercase, no spaces, same as folder.
"opensvg": false, // Set to true if this mode cares about opening SVGs.
"weight": 2, // How far left or right to place in the navigation
"dependencies": [ // RoboPaint/Node/DOM modules to preload into the mode (so you don't have to!)
"paper",
"jquery",
"underscore",
"mediasets"
],
"graphics": {
"icon": "images/example_icon.svg", // Relative path to the color icon for the mode. Can be SVG or any other web format.
"previews": [
"images/example_preview.jpg" // Optional list of as many preview images of the output of the mode or screenshots.
]
},
"persistentScripts": [
"example_persistent.js" // A list of scripts to be run persistently outside of the mode sandbox, in RoboPaint proper.
]
}
}
A requirement for RP modes is a bare minimum set of translation for its
informational strings, and at best, a full set of all strings shown to the user
in the default language of English, with support for other languages optional
but highly suggested. Just put your JSON translation files in your
mymode/_i18n
folder and RP will parse and apply translations according to the
following two methods:
The two ways a mode will be translated:
native
: Use i18next style attribute tags and translate strings in place of all user displayed translatable text, and these will all be dynamically replaced by their appropriate language translated string automatically whenever a language is selected globally. This is the best method when you control all the code for the mode.dom
: Using a jQuery DOM selector mapping file, when a language is selected, RP will iterate through the given DOM mapping and replace the text contents with the translation string given. This is the best method when the majority of code is third party and can't be directly edited.
These are the minimum entries required for a mode. (Comments are for clarity and are not actually allowed in valid JSON.)
{
"_meta": { // Meta information is for machine reference, not for translation.
"creator": "myname",
"target": "en-US",
"release": "0.9.5",
"basetype": "mode"
},
"info": {
"name": "My Example mode", // 2 to 5 words that make up the "name" of the mode
"use": "Make Example Art", // 2 to 4 words that describe what the mode does, shows up in home screen bubble preview.
"detail": "Lots of detail text" // Describe the mode in detail, as much text as you need.
},
"myi18nsection": { // Optional standard i18next translation key:value structure...
"mystring": "My string" // Available for translation at modes.[modename].myi18nsection.mystring
}
}
Only required if your translation mode is dom
instead of native
. Expects one
key outside of the _meta
key, an array named map
with the data pairs in the
format of key: jQuery selector, value: fully namespaced translation key. See the
excerpt of the Edit mode DOM map below for example.
{
"_meta": {
"creator": "rogerfachini",
"release": "0.9.5",
"basetype": "DOMmap"
},
"map": {
".menu_title:eq(1)": "modes.edit.menu.file.title",
".menu_title:eq(2)": "modes.edit.menu.edit.title",
...
"#tool_select": {"title": "modes.tools.select"} // Can also set attribute text directly with an object
}
}
Your JavaScript has full Node.JS access, and HTML DOM access at the same time.
Feel free to use node modules specific to your mode in your package.json
, or
use any of the modules available/used in RoboPaint or Electron. Also available
are the following window
global variables:
i18n
: The full i18next CommonJS client module, loaded with the modes full translation set, and the RP central translations for common strings.app
: The electron app object from the main process.rpRequire
: The helper function for adding RP "modules", external libraries, and any other commonly useful code. See rp.require.js for a full list of shortcutsipc
: The Inter Process Communication module for sending events and messages to/from the main window process. Most of this is managed here, but having this globally available makes custom comms possible.cncserver
: Only the clientside DOM API wrappers for cncserver.robopaint
: A limited version of the robopaint object. Contains:settings
(read only),utils
,canvas
,cncserver
, and the following:robopaint.pauseTillEmpty(init)
: Tell RoboPaint to pause the cncserver buffer until after pauseTillEmpty(false) has been called, and the local push buffer to the main buffer has been emptied. This ensures a perfectly smooth start to printing.robopaint.svg
: A set of utility functions for managing the "loaded" SVG data storage. This data storage is used directly by modes that open SVG and is cleared at startup if "load last image" is unchecked in settings. Includes the following methods:svg.save(data)
: Save the given SVG text data, will overwrite any existing data without warning, so use with caution.svg.wrap(inner)
: Wrap the given inner SVG XML text data in a correctly namespaced and canvas sized header and footer.svg.load()
: Load the currently saved SVG text data. If empty, will return empty valid SVG text from the wrap function above.svg.isEmpty()
: Returns booleantrue
if there is no data saved.
mode
: JSON Package of the current mode, withpath
, and utility functions:mode.run({mixed})
: Emulation IPC passthrough of original commander API shortcut. Allows immediate queuing of ~500 cmds/sec to CNCServer, no callbacks required. See cncserver.commander.js for a full list of supported shortcut commands and their arguments.mode.settings
: An object with setters/getters to manage storing this modes specific settings. Automatically stores inlocalStorage
per botType.mode.translate()
: Trigger re-translation of the mode given current language settings and markup.mode.settings.$manage(selectors)
: A full settings management system for form elements. List the selectors for each input element you want to track, and from one call this will load, track changes & save all values automagically keyed on each elements ID!mode.fullCancel(message)
: Standardized full cancel, buffer clear and park with status message.
The mode object is also where event callback functions should be defined, full supported list here: As always, look to the base modes for precise implementation examples and common use cases.
mode.translateComplete()
: Called whenever translate is done. Happens on init, and after every language change.mode.onPenUpdate(actualPen)
: Called when the bot actually moves, the the object will contain the full CNCServer pen object of where it should or will be after the "lastDuration" key value. Usually used to operate the live draw pen indicator.mode.onCallbackEvent(name)
: When a "callbackname" is run into the CNCserver buffer and eventually run, this will fire with thename
given. Usually sent at the end of long jobs like autoPaint.mode.bindControls()
: A handy function to store all your control button bindings, called when the page is fully loaded & translation is done, but before .mode.onClose(callback)
: Called whenever the user attempts to either change the mode, or close the application. If implemented, the user can only close or change the mode once "callback" has been called.mode.onFullyResumed()
&mode.onFullyPaused()
: A mode can run pause or resume, but doesn't know when this happens until these are called.mode.onMessage(channel, data)
: For any undefined gen events, wraps ipc.
Here's a quick example using mode.run()
for quick sequential moves and
actions. This will buffer the commands to pick a color, draw a square, wash the
brush, then park.
mode.run([
['tool', 'color3'],
'up',
['move', {x: 20, y: 20}],
'down',
['move', {x: 400, y: 20}],
['move', {x: 400, y: 400}],
['move', {x: 20, y: 400}],
['move', {x: 20, y: 20}],
'wash',
'park'
]);
And that's just the start! Experiment with the example modes, make it better, tweak it, or even just toss it all out. It's all up to you.
If you'd like to get your mode back into the RoboPaint project, just follow the instructions for contributing back with a pull request at the bottom of the main project readme. To be eligible for being added to RoboPaint, your mode must pass code quality and security standards, and be available via NPM.