Beyond Web-Apps
React, Javascript and WebAssembly to Port Legacy Native Apps
@FlorianRival
gdevelop-app.com
2008
Β open-source, cross-platform game creator designed to be used by everyone
data:image/s3,"s3://crabby-images/726b9/726b9220c2c3e09755bad854e5adf7d34f11ef8f" alt=""
- An editor for games
- Visual programming
- Output HTML5 games
data:image/s3,"s3://crabby-images/10eb2/10eb2758110951f84075233d052b402b5c633ff7" alt=""
data:image/s3,"s3://crabby-images/75add/75add2f0d05405869cb7354ee7e1359897c101a3" alt=""
data:image/s3,"s3://crabby-images/ab374/ab3744ca0a98552b24745da2ea631c94019f75cf" alt=""
data:image/s3,"s3://crabby-images/817c8/817c8123dc6273c2371896c3d67eb300e82e2300" alt=""
data:image/s3,"s3://crabby-images/dc672/dc67297fa268c7e355a9a1402a941266fa6b8930" alt=""
data:image/s3,"s3://crabby-images/a5cad/a5cad636247ab35bf8cfb9335454cde1c3aa63bb" alt=""
data:image/s3,"s3://crabby-images/101ef/101ef05b29def4412ed33dc185a07cc3b735c35f" alt=""
data:image/s3,"s3://crabby-images/3f695/3f695155aa353c7ba9b8e5d4a8387c6bbeaf2fe7" alt=""
data:image/s3,"s3://crabby-images/7c954/7c954b7c66e209c30e0eb964fd5b6942674aa35b" alt=""
2016
The editor is getting old
- Lots of cross-platforms issues with the UI toolkit, support for macOS and Linux is bad.
- Iterating on the software is slow (C++ π¬)
- Building UI is slow...
- ...and limited to old UI components...
- ...and UX would need some enhancements π
- The entry barrier for new contributors is high (C++ π¬)
It's time to react!
What do we want?
- A new editor!
- Cross-platform, better UI building paradigm, faster development, lower entry barrier
- No full rewrite!
- Full compatibility with the existing projects.
- Nobody got time to wait for a rewrite.
Could web technologies help?
data:image/s3,"s3://crabby-images/39a58/39a585d4a32c9c961049cbad7e66ec82d7377fc6" alt=""
GDevelop C++ codebase
New GDevelop editor
GUI (windows, dialogs...)
Filesystem
Core classes (game, scene, object, textures...)
Game exporter
libGDevelop.js
React.js powered interface
Node.js/browser adapters
data:image/s3,"s3://crabby-images/0ae47/0ae4797600ed2d1f5dd6ce6ab674cef8793194ef" alt=""
data:image/s3,"s3://crabby-images/09a15/09a158848408bd82e7efc1b2ab1547b8770a9501" alt=""
data:image/s3,"s3://crabby-images/8b9d4/8b9d490b4830a953bfc922d9dce736f349cc5b7f" alt=""
X-platform toolkit (wxWidgets)
data:image/s3,"s3://crabby-images/a15f2/a15f2e48170577b8aa1ba8243b18ba363e819711" alt=""
data:image/s3,"s3://crabby-images/042ae/042ae82df798b9e69eb2822d8b8ba3ebf2ffa3fc" alt=""
data:image/s3,"s3://crabby-images/c9c8c/c9c8c7626995e232d9c0e1bea3d951c2a780a6ab" alt=""
data:image/s3,"s3://crabby-images/a15f2/a15f2e48170577b8aa1ba8243b18ba363e819711" alt=""
data:image/s3,"s3://crabby-images/042ae/042ae82df798b9e69eb2822d8b8ba3ebf2ffa3fc" alt=""
data:image/s3,"s3://crabby-images/c9c8c/c9c8c7626995e232d9c0e1bea3d951c2a780a6ab" alt=""
data:image/s3,"s3://crabby-images/e3b3c/e3b3c30a8e07a655a6d8a46a741dc9d168a7a89f" alt=""
data:image/s3,"s3://crabby-images/085ad/085ad7afd614d67038744352323566a61b1c1fbf" alt=""
data:image/s3,"s3://crabby-images/39a58/39a585d4a32c9c961049cbad7e66ec82d7377fc6" alt=""
data:image/s3,"s3://crabby-images/0e69f/0e69f24873fcf745d9d377146139142708007ef2" alt=""
data:image/s3,"s3://crabby-images/cd607/cd607707b5e9bb723183337877f14a06a2993999" alt=""
data:image/s3,"s3://crabby-images/2f174/2f1747f2ac9c02d83913fc2e5e23eb7184f3139e" alt=""
data:image/s3,"s3://crabby-images/39a58/39a585d4a32c9c961049cbad7e66ec82d7377fc6" alt=""
data:image/s3,"s3://crabby-images/085ad/085ad7afd614d67038744352323566a61b1c1fbf" alt=""
LLVM IR (bitcode)
Using Emscripten
git clone https://github.com/juj/emsdk.git && cd emsdk
./emsdk install latest && ./emsdk activate latest
./emcc tests/hello_world.c
node a.out.js
For large projects, Emscripten provides replacements that swap GCC or Clang by Emscripten in build system (automake, CMake...)
Expose existing classes
class ObjectsContainer {
Object& InsertObject(const Object& object, std::size_t position) { ... };
Object& GetObject(const String& name) { ... };
}
class Layout : public ObjectsContainer {
public:
void setName(const gd::String & name_) { name = name_; };
const gd::String & getName() { return name; };
// ...
}
interface Layout {
void setName([Const] DOMString name);
[Const, Ref] DOMString getName();
// Inherited from ObjectsContainer
[Ref] Object insertObject([Ref] gdObject object, unsigned long pos);
[Ref] Object getObject([Const] DOMString name);
}
"WebIDL" bindings for Emscripten WebIDL binder:
Use Emscripten classes
Module().then(gd => {
const layout = new gd.Layout();
layout.setName("My game level");
const object = new gd.Object();
object.setName("My character");
layout.insertObject(object, 0);
console.log(layout.getName()); // "My game level"
console.log(object.getObject("My character")); // Returns a gd.Object
});
- Primitive types are converted, including JS strings (to char*)
- Objects references are converted to pointers (or references) by the webIDL generated glue code
- std::iostream is binded to console.log
data:image/s3,"s3://crabby-images/206f7/206f7ce576cd8e0535cb2f326bcc5fc92b81058b" alt=""
data:image/s3,"s3://crabby-images/d38c6/d38c6707fe3b15fffee6150dc6d661049726069f" alt=""
Things to know
const layout = new gd.Layout();
// ...
gd.destroy(layout); // Binded objects need manual destruction
- Memory management requires care!
- Output files can be large (~3mb for GDevelop), but gzip helps.
- For React:
componentDidMount() { this._project = new gd.Project(); }
componentWillUnmount() { gd.destroy(this._project); }
or use an Effect Hook π₯
Things to know
- A complete test set is invaluable for ensuring that no errors are in the bindings.
abort(16). Build with -s ASSERTIONS=1 for more info.
- Wrong types of parameters
- Parameter forgotten
- Using a deleted object
data:image/s3,"s3://crabby-images/31944/31944a9d2a5fc7613601f3c73eac42fec8cfa655" alt=""
Finding a component library
The goal is to get started with all the classical components of a user interface without rewriting everything
data:image/s3,"s3://crabby-images/bc6ea/bc6ea7d5f276db0ba563411a3b7eebeedbf99f48" alt=""
data:image/s3,"s3://crabby-images/beca2/beca24c1e69f499788fd7da5f1bacaaa4dfd2afe" alt=""
data:image/s3,"s3://crabby-images/a908a/a908a3705d1372c3ebe5e5a5256df4cc265bb5b3" alt=""
data:image/s3,"s3://crabby-images/399f9/399f9ff6fa044ca4ae69e0994a624af115338152" alt=""
data:image/s3,"s3://crabby-images/4ff01/4ff01f2bb75b45a017b539d53c5ee97530710cb7" alt=""
data:image/s3,"s3://crabby-images/387c0/387c0ea49c098c41f4c2069e4bf01e80295b9abc" alt=""
data:image/s3,"s3://crabby-images/c640a/c640a40ebb5f94729b380332be1ae40299fdd6d3" alt=""
data:image/s3,"s3://crabby-images/9ef9d/9ef9dff6fa745f8b0e8801a0a76a0937a2d05b68" alt=""
data:image/s3,"s3://crabby-images/d9b4b/d9b4bcf8060345e219192d7b7671f0fa35896377" alt=""
data:image/s3,"s3://crabby-images/39447/39447b5a9c8477573e8c72a4d904aace93a54f20" alt=""
data:image/s3,"s3://crabby-images/bdb42/bdb4284c985af236431edd3dc1fd6219148b0743" alt=""
data:image/s3,"s3://crabby-images/9804a/9804a05f53e3b9ff81fa9a9c23c3564d4f61d26d" alt=""
data:image/s3,"s3://crabby-images/9ab32/9ab3220837189ac796e128179e22d077c17aa54b" alt=""
data:image/s3,"s3://crabby-images/43d75/43d7506475a9fc63ae0b2cb96a3ecdce819cce12" alt=""
data:image/s3,"s3://crabby-images/83112/83112cca7875d98537a62121b928076d240e12f5" alt=""
data:image/s3,"s3://crabby-images/0b7db/0b7dbcc4c83571b18080e224353a2aa1162561df" alt=""
data:image/s3,"s3://crabby-images/918a3/918a33b5dfdfe6a9b1d81b5dc3672ccd988813b6" alt=""
- Extensive list of high quality components
- Good theming support
- Support for (nested) dialogs
- Flexibility of components
- Documentation quality
- Performance (blazing fast? β‘οΈ)
Large lists (with drag'n'drop)
We need to display lists from dozens to hundred of elements
data:image/s3,"s3://crabby-images/9628a/9628a5d9a9ea9c8fd3d6d8d5a4f802886a81092e" alt=""
data:image/s3,"s3://crabby-images/2c7e0/2c7e0332822a96a25a3541374e8aebc79fdfeaf5" alt=""
π Virtualized lists
data:image/s3,"s3://crabby-images/765cf/765cfeadeab1774b3bd0cb1b451c097003e64cdc" alt=""
const SortableObjectRow = SortableElement(props => (
<ObjectRow ... />
));
const ObjectsList = (props) => (
<List
rowRenderer={({ index, key, style }) => {
<SortableObjectRow />
})
/>
);
const SortableObjectsList = SortableContainer(ObjectsList);
<SortableObjectsList
onSortEnd={({ oldIndex, newIndex }) => ...}
...
/>
+ react-sortable-hoc
const ObjectRow = () => <div>...</div>;
const ObjectsList = (props) => (
props.map(group => <ObjectRow ... />)
)
Moveable editors
data:image/s3,"s3://crabby-images/65e1b/65e1bf717d6912ce9b48011500d69d668344630e" alt=""
data:image/s3,"s3://crabby-images/7882f/7882f79f997db5194f509db32ee4d337f3465953" alt=""
Game editors are traditionally composed of multiple editors.
react-mosaic
π Use React tiling window manager libraries
data:image/s3,"s3://crabby-images/b8701/b8701e23d5eead2fc48296976ff1d0f0e4dd67e1" alt=""
Keyboard handling
data:image/s3,"s3://crabby-images/f37d4/f37d4e58923860d4a4e6f6b28c29448d3edc06d4" alt=""
data:image/s3,"s3://crabby-images/0ae85/0ae855f5f9e8635c8e5522e3c5797c6bb71312ec" alt=""
Focus/tab browsing
Shortcuts
π Give focus() to your components
componentDidMount() {
if (this.props.focusOnMount) {
this._searchBar.focus();
}
}
or use a Hook π₯
https://blueprintjs.com/docs/#core/components/hotkeys
π Abstract keyboard shortcuts so that it's easy to add to any component
<Hotkeys>
<Hotkey
global={true}
combo="shift + a"
label="Be awesome all the time"
onKeyDown={() => console.log("Awesome!")}
/>
</Hotkeys>
Large trees (with drag'n'drop)
data:image/s3,"s3://crabby-images/39764/3976466e4229107bddc298a8df8b52832612389b" alt=""
data:image/s3,"s3://crabby-images/45b16/45b16ed767a4cbd7ecb76559457a2c6feaebb3c8" alt=""
π Virtualization again!
react-sortable-tree
data:image/s3,"s3://crabby-images/381b6/381b6003980edb112be409cc4934a247a87d49bb" alt=""
Levels rendering or visualizations
data:image/s3,"s3://crabby-images/93fe0/93fe0fcd511a304dfab8be1a27a7123244f169b2" alt=""
data:image/s3,"s3://crabby-images/efe4e/efe4e07191c5ebc23de49fa7b5b3695a25215dfc" alt=""
π The DOM is not powerful enough here
data:image/s3,"s3://crabby-images/1b5d5/1b5d536db8ada241d7e9d04f4c69c71a508d134e" alt=""
data:image/s3,"s3://crabby-images/e4412/e4412b2e99258b64f5527166b17bcec494e553dd" alt=""
data:image/s3,"s3://crabby-images/77177/77177fff10c110406991e7bc19cd700d25004474" alt=""
data:image/s3,"s3://crabby-images/848ad/848adf1fe585f0cf9e79d62c3ae9097812217f22" alt=""
π The DOM can still be good enough
data:image/s3,"s3://crabby-images/03568/035688c7ceb4c44f3636a7dc75538ea7ef5c596a" alt=""
<svg
onMouseMove={...}
onMouseUp={...}
width={this.props.imageWidth}
height={this.props.imageHeight}
>
<polygon
fill="rgba(255,0,0,0.2)"
stroke="rgba(255,0,0,0.5)"
fileRule="evenodd"
points={vertices.map(vertex =>
`${vertex.getX()},${vertex.getY()}`
).join(' ')}
/>
{vertices.map((vertex, j) => (
<circle
onMouseDown={...}
key={`vertex-${j}`}
fill="rgba(255,0,0,0.75)"
cx={vertex.getX()}
cy={vertex.getY()}
r={5}
/>
))}
</svg>
When things aren't native enough
π Build abstractions over Electron APIs
export default class ElectronMenuImplementation {
buildFromTemplate(template) {
this.menuTemplate = template;
return undefined;
}
showMenu(dimensions) {
if (!electron) return;
const { Menu } = electron.remote;
const browserWindow = electron.remote.getCurrentWindow();
this.menu = Menu.buildFromTemplate(this.menuTemplate);
this.menu.popup({
window: browserWindow,
x: Math.round(dimensions.left),
y: Math.round(dimensions.top + dimensions.height),
async: true, // Ensure the UI is not blocked on macOS.
});
}
}
class MaterialUIContextMenu extends React.Component { ... }
class ElectronContextMenu extends React.Component { ... }
export default class MaterialUIMenuImplementation {
buildFromTemplate(template) {
return template.map((item, id) => {
if (item.type === 'separator') {
return <Divider key={'separator' + id} />;
} else if (item.type === 'checkbox') {
return (
<MenuItem
...
/>
} else {
return (
<MenuItem
...
/>
}
}
showMenu() { ... }
}
data:image/s3,"s3://crabby-images/d56b4/d56b42d9c7088b5059483dfdbbb35c396e8164eb" alt=""
data:image/s3,"s3://crabby-images/bcda3/bcda399569511871e9eab9bed42fb28b3bdc7d9b" alt=""
When things aren't fast enough
π Double check rendering of your components
- why-did-you-update
Don't forget to measure performance in production (React development build is way slower)
- React 16.5 profiler
- shouldComponentUpdate is often the answer
data:image/s3,"s3://crabby-images/5f322/5f32238d01e70f9dfc25bb3e73fe414232d80f86" alt=""
π Inspect calls to Emscripten objects
- Beware of overhead in binding code/wasm call when calling functions, in particular if renderings
objects.filter(object => object.getType() !== "") // n wasm calls + string conversion
.map(object => object.getVariables(project)); // n wasm calls + string conversion
Won't be noticeable in most UI situations, but can be if used in renderings or long lists/trees
- It might be useful to store strings/values on the JS side if you know they won't change.
2017
data:image/s3,"s3://crabby-images/230d8/230d8e415ff794c2239b717205891121186a5014" alt=""
Ultra fast iteration and testing
Feedback loop is a few seconds while developing
data:image/s3,"s3://crabby-images/dc3bf/dc3bf99fd7d38ae096e1c5b567b9d8f56c23c43d" alt=""
data:image/s3,"s3://crabby-images/b2839/b2839dde252db96d7f2e26bcc9a6958f64c02529" alt=""
vs
- Storybook
- React Styleguidist
Modular development of UI components
data:image/s3,"s3://crabby-images/bd93e/bd93ee7bcd1413cbbf07dfa406447e40a0bffbcb" alt=""
- electron-builder
Auto-updates and packaging...
data:image/s3,"s3://crabby-images/3d573/3d5733a65b7c284360601d49afe884bff960aee0" alt=""
...and code signing, and upload, and creating installers for Windows, .DMG for macOS, .tar.gz/AppImage for Linux.
Since day one, GDevelop 5 is auto-updated.
Time between releases moved from months to weeks or days.
- More contributors than before
- Startup time is similar (even faster as a web-app, ~3seconds)
- People can try the software from their browsers in seconds without downloading anything
- Documentation can link to the editor and examples can be opened directly in browser
- Can be later ported to iOS/Android tablets, even phones
- Integration with Dropbox, Google Drive...
And more
Plot twist
It's (still) JavaScript!
data:image/s3,"s3://crabby-images/8386f/8386f9ba8837708cdd6385f1e13be4552ae6ae33" alt=""
WebAssembly is on average 33.7% faster than asm.js, with validation taking only 3% of the time it does for asm.js
- Bringing the web up to speed with WebAssemblyΒ Haas et al., PLDI 2017
It's good enough
but WebAssembly will make it even faster
Epilogue
What are users saying?
Wow, what a difference a year can makeΒ
but I have to admit, once you worked a bit with it, it could be really more productive on many aspects than GD4
Amazing how much easier and streamlined the engine has become
Thanks!
@FlorianRival
gdevelop-app.com
github.com/4ian/GDevelop
Beyond Web-Apps: React and WebAssembly to Port Legacy Native Apps
By Florian Rival
Beyond Web-Apps: React and WebAssembly to Port Legacy Native Apps
Talk at React Conf 2018, React Boston 2018 and React Next 2018
- 4,311