User manual • Developer manual • Testimonials / Reviews • List of known nin projects
DISCLAIMER: This is a tool created for internal use by Ninjadev, and is open sourced to share ideas with and get feedback from the community. You are free to use it, according to the License, but we will not necessarily provide support and may at any time add, change or remove features as we require for our productions.
nin is ninjatool
nin is Ninjadev's internal demo tool. It is a tool for easing development of browser-based WebGL demos. Core features include:
- Node-based demotool, effortlessly reuse your effects, scenes, and create crazy transitions.
- Livereloading of shaders and scenes in the browser. No more manual recompilation!
- Tight THREE.js integration for all your WebGL needs.
- Compile and pack your WebGL demo to a .png.html file for easy compo delivery
The backend component is written in node.js, and keeps track of and recompiles changed files. The frontend is created using React, and communicates with the backend over websockets.
To install nin simply run the command:
npm install -g ninjadev-nin
This projects requires node version 7.9.0
or newer.
You can install node from packaging here or download zipped source from this page and verify signatures with the guide here;
You will also need git
installed for project generation to work.
New project created with nin get a default .eslintrc suitable for WebGL demos with THREE.js.
Running nin new <dirname>
will create the specified directory and initialize a new nin project inside.
Running nin run
inside the newly created project will make it accessible on http://localhost:8000.
nin creates a manifest file called nin.json
.
You should fill out this file with the title of your demo, the authors, a description, song metadata, and even a google analytics tracking code to collect statistics.
The metadata is then used to generate html meta-tags in the head of the demo,
as well as in the metadata segment of the .png.html
file.
The png metadata can be viewed with a command such as pngcheck -c -t -7 bin/demo.png.html
on linux.
Create a new node by clicking Generate -> THREE NODE
in the frontend menu.
The node will be placed in src/nodeName.js
and added to the graph in res/graph.json
.
You must connect the node yourself to the output node.
This is done by setting connected.screen
to nodeName.render
as in the example below,
where nodeName
is the id of the node you want to connect to the display.
{
"id": "root",
"type": "NIN.RootNode",
"connected": {
"screen": "nodeName.render"
}
}
}
Create a shader node by clicking Generate -> Shader Node
in the frontend menu.
It generates the file src/nameOfTheShaderNode.js
and the folder src/shaders/nameOfTheShaderNode/
.
To get livereload on shader change, you shader must be specified in the options object of your node in res/graph.json
,
the shader generator will do this for you.
If needed, you can access the shader through the global SHADERS
object, by writing SHADERS.nameOfTheShader
.
The nin compile
command will create a single file bin/demo.png.html
that contains all the code and resources of your demo.
Base64 and PNG compression magic is used to achieve this.
It will at the same time output a file without PNG compression, bin/demo.html
, which will be slightly larger, but compatible with a wider range of devices (especially smartphones).
For faster compiles, pass the flag --no-closure-compiler
. This will only concatenate js files, without any minifying.
Some of nin's settings can be overriden by placing a .ninrc file in your home directory. Currently, keyboard shortcuts is the only behavior which can be changed in the .ninrc. The list of canonical names for keybindings can be found in nin/frontend/app/scripts/directives/menubar.js.
An example .ninrc looks like the following:
[keybinds]
startRendering=left
stopRendering=right
nin run
.- Open nin in your browser, navigate to the frame you want to render from, and press R. This will start dumping single frames as numbered .PNGs in
bin/render/
in your project folder. - Refresh etc every time WebGL crashes.
- When you have rendered all the frames:
nin render
. You needffmpeg
(ffmpeg.org) installed for this. - The demo is now available at
bin/render/render.mp4
.
Each frame will take up to around 4MB on disk, and the finished .mp4 will be on the order of 1GB when rendered, so make sure you have enough disk space. Expect to render maybe a frame or two per second.
You will need to have node installed.
Running make
in the nin folder will build and compile the entire project.
Running npm link
will add nin to your node binaries path, making it available globally.
First, run nin run
inside your project.
If you wish to develop on the frontend, running make run
inside nin/frontend/
makes webpack rebuild the frontend on file change.
You only need to rerun nin run
if you change files in either nin/dasBoot
or nin/backend
.
Usually, your demo will be a series of connected scenes. This section covers how to work with them.
Adding a scene is quite straight forward, once you become good friends with graph.json!
This is what you need to to do get your first scene wired up and ready to go!
Here is an example of how to prolong the demo by adding a new scene at the end.
Sometimes you want to squeeze in a new scene between two other scenes that you already have in your demo. Often you will perhaps only shorten the preceding scene and leave the start of the following scene intact. But for completeness, here is an example where we both shorten the previous scene, and chop down the beginning of the following scene.
To ensure that random things happen consistently across runs it is recommended to use the Random
-class rather than Math.random()
.
Set up Random
by initializing it in your constructor like so:
constructor(id, options) {
...
this.random = new Random('seedString');
...
}
Then use the random generator you just created like this:
var randomNumber = this.random();
A lot of making a demo is syncing what's happening on the screen with the music.
A BEAN
can be described as the smallest possible resoluton of the beat.
Working with BEAN
s directly will usually not give you a smooth 60 FPS animation.
BEAN
s are incremented less often than 60 FPS. A common approach is using frame
with FRAME_FOR_BEAN
instead.
E.g. Instead of doing
var startBean = 1;
var endBean = 20;
var fractionIn = (BEAN - startBean) / (endBean - startBean);
do
var startFrame = FRAME_FOR_BEAN(1);
var endFrame = FRAME_FOR_BEAN(20);
var fractionIn = (frame - startFrame) / (endFrame - startFrame);
Counter | Description | Typical way to access |
---|---|---|
Bean | The smallest possible resoluton of the beat. | BEAN |
Frame | Monotonously counts upwards. Usually what you want to use in your update() -function. |
frame (if you need frame inside your render method, you can store it on this inside the update method) |
Beat | ToDo | ToDo |
Bar | ToDo | ToDo |
In your scene-function, you can define an update function.
update(frame) {
// The coolest of code
}
This will run for every frame.
You might have defined an object in your scene, such as a box, the camera, or a light source, which you want to move within your scene. To aid you in doing this smoothly, there are some predefined functions you can utilize.
The most important ones are:
- smoothstep
- lerp
They all have the same API:
lerp(startValue, endValue, t)
When t
is smaller than or equal to 0, startValue
is returned.
When t
is between 0 and 1, a value between startValue and endValue is returned, depending on which interpolation function you're using.
When t
is larger than 1, endValue
is returned.
For more details, check out http://en.wikipedia.org/wiki/Smoothstep .
In the nin.json
-file you can define the music
-section directly in the root.
Here you can specify
path
: relative path to the music filebpm
: the tempo of your music in beats per minute
Sample music configuration:
"music": {
"path": "res/music.mp3",
"bpm": 190,
"subdivision": 12,
"BEANOffset": 0
},
For details on how this section is processed further you can check out nin/dasBoot/BEATBEAN.js
.
How to write text, or letters in general, to your 2D canvas.
Out of the box you can write text to your canvas like this:
this.canvas.getContext('2d').fillStyle = '#ffffff'; // The color of the text you are going to be displaying
this.canvas.getContext('2d').font = 'bold ' + (24) + 'pt Arial'; // Pattern: weight [size]size_unit font_name
this.canvas.getContext('2d').textAlign = 'center';
this.canvas.getContext('2d').textBaseline = 'middle';
this.canvas.getContext('2d').fillText('Hello Nin!', x_position_on_canvas, y_position_on_canvas);
The frontend part of this project uses ESLint for linting.
See the .eslintrc.js
file in the frontend part of this project.
The demo itself and our own dasBoot
uses the Google Closure Linter, please see this link for installation information.
https://developers.google.com/closure/utilities/docs/linter_howto
Use the --nojsdoc
flag.
To publish nin, checkout a new branch, run make bump-version
, git push
, get
it merged, and Travis will automatically publish a new release after master has
finished building.
The prerequisites remain the same, you at least need Node.
To build and compile the entire project, for now, you need only run npm start
in the root of the nin-repo.
To run nin without linking up through npm you can replace the nin
-command with node path-to-ninrepo/nin/backend/nin
.
E.g. when you are in a project folder of a demo, and you want to run it with your freshly compiled nin directly, you can run node path-to-ninrepo/nin/backend/nin run
instead of nin run
.
If you want to run it from powershell regularly, you might want to make an alias in your profile akin to this:
function nin
{
param($argz)
node $ninRepoPath\nin\backend\nin $argz
}
Alternatively you can use the psNin.ps1
script from your demo, or call it from anywhere if you supply it to the optional $demoPath
parameter.
nice! - mrdoob
Oh man, I didn't know you guys released your tools. I'm a big fan of your stuff -- awesome to see such polished prods on the web. Happy to have helped enable some amazing work! - daeken