Skip to content

Scripting Support 2.0

James Baicoianu edited this page Feb 20, 2021 · 8 revisions

As of JanusWeb 1.1, we're experimenting with a new syntax for defining custom elements in Janus room scripts. This allows developers to extend built-in functionality or create all-new functionality from scratch, opening the community up to build, share, and use libraries of custom-defined object types.

This scripting engine also adds more object-oriented functionality to our scripting API, as well as some convenience functions for translating between object-relative and world-relative coordinates.

Custom Elements

A key part of being able to build reusable components for VR and the web is to be able to encapsulate code into reusable, extendable, and shareable chunks. We've built our implementation based on the HTML5 Custom Elements spec, and we now expose an API to room scripts which makes it easy for room developers to define the objects they need to add interactivity and other advanced functionality to their sites. We've also started a few side projects to build up some libraries of components we see as being useful to the creation of VR experiences as a whole:

  • janus-script-ui: a collection of 3D/VR UI controls - buttons, toggles, sliders, etc
  • janus-script-layout: a collection of components for quickly laying out groups of objects in 3D
  • janus-script-game: a collection of custom components inspired by gaming mechanics (doors, teleporters, etc)
  • janus-script-video: a collection of components for building advanced players for various media types
  • janus-script-audio: a collection of components for wiring up positional audio and audio processing for your room

So what is a Custom Element? Well, first let's look at a super-basic Janus room. Following the markup reference at janusvr.com/guide/markuplanguage/index.html we start with a basic room:

<FireBoxRoom>
  <Room use_local_asset="room1">
    <Object js_id="mycube" id="cube" col="1 0 0" pos="0 0 5" />
  </Room>
</FireBoxRoom>

Nothing fancy, just a room with a red cube. But let's say we want something to happen to the cube - let's say we want a button, and when we push the button it'll change to a green cube. With custom elements and the janus-ui library, this is easy:

<FireBoxRoom>
  <Assets>
    <AssetScript src="janus-ui.js" />
  </Assets>
  <Room use_local_asset="room1">
    <Object js_id="mycube" id="cube" col="1 0 0" pos="0 0 5" />
    <PushButton pos="0 0 4" onbuttonpush="room.objects['mycube'].col = V(0,1,0)" />
  </Room>
</FireBoxRoom>

What if we want each push of the button to toggle between red and green? No problem - there's a <ToggleButton> component. Let's get fancy with the colors, too - you can use any approved HTML color name:

<FireBoxRoom>
  <Assets>
    <AssetScript src="janus-ui.js" />
  </Assets>
  <Room use_local_asset="room1">
    <Object js_id="mycube" id="cube" col="1 0 0" pos="0 0 5" />
    <ToggleButton pos="0 0 4" 
                  onbuttonpush="room.objects['mycube'].col = 'forestgreen'" 
                  onbuttonrelease="room.objects['mycube'].col = 'maroon'" 
                  />
  </Room>
</FireBoxRoom>

What if we have a bunch of images, and we want to lay them out in a circle? Janus-layout and <CircularLayout> to the rescue:

<FireBoxRoom>
  <Assets>
    <AssetScript src="janus-layout.js" />
  </Assets>
  <Room use_local_asset="room1">
    <CircularLayout radius="2.5" pos="0 1 0">
      <Image id="http://i.imgur.com/ff3Wn06.png" />
      <Image id="http://i.imgur.com/rtCeOEs.png" />
      <Image id="http://i.imgur.com/ByExCaO.png" />
      <Image id="http://i.imgur.com/dYBH57H.png" />
      <Image id="http://i.imgur.com/2PQlhdB.png" />
      <Image id="http://i.imgur.com/xNEw11S.png" />
    </CircularLayout>
  </Room>
</FireBoxRoom>

Okay, cool. So how do I go about defining my own components? Well, let's take another example, and say we want an easy way to display an image with a caption. Let's define a <captioned-image> element which does this for us. We can inherit from the built-in <Image> class, and just add our own functionality without having to change anything else.

Our markup will look like this

<FireBoxRoom>
  <Assets>
    <AssetScript src="captioned-image.js" />
  </Assets>
  <Room use_local_asset="room1">
    <captioned-image image_id="http://i.imgur.com/ff3Wn06.png" caption="WebVR Fireworks Experiment" pos="-2 0 4" />
    <captioned-image image_id="http://i.imgur.com/ByExCaO.png" caption="Teleporters" pos="0 0 4" />
    <captioned-image image_id="http://i.imgur.com/2PQlhdB.png" caption="Sliders and Buttons" pos="2 0 4" />
  </Room>
</FireBoxRoom>

And captioned-image.js looks like this:

room.registerElement('captioned-image', {
  image_id: '',
  caption: '',
  create() {
    // Called when each instance of this object type is created
    this.image = this.createObject('image', {
      id: this.image_id
    });
    this.captionobject = this.createObject('text', {
      pos: V(0,-.5,0),
      text: this.caption,
      col: 'white',
    });

    // Set up some event listeners to add interactivity.  Note that these mouse events are also emulated for VR, gamepads, and touchscreen devices as well
    this.addEventListener('click', (ev) => this.handleClick(ev));
    this.addEventListener('mouseover', (ev) => this.handleMouseOver(ev));
    this.addEventListener('mouseout', (ev) => this.handleMouseOut(ev));
  },
  update(dt) {
    // Perform some per-frame functionality here
  },
  handleClick(ev) {
    // Do something when this event is selected with mouseclick, touchscreen tap, gamepad, or motion controller
  },
  handleMouseOver(ev) {
    this.image.scale.set(1.1, 1.1, 1.1);
    this.captionobject.col = 'green';
  },
  handleMouseOut(ev) {
    this.image.scale.set(1, 1, 1);
    this.captionobject.col = 'white';
  },
});

Now how about something more advanced. Let's say we want to make a slideshow to show off some number of images. We want our slideshow to have fancy bouncy physics animations, and we want the selected image to be large and centered, while the other images fade out to either side. Clicking on an image will slide the whole set over to make the selected image active.

Our markup looks a lot like the <CircularLayout> example above, but here we use <SlideShow> instead:

<FireBoxRoom>
  <Assets>
    <AssetScript src="janus-layout.js" />
  </Assets>
  <Room use_local_asset="room2">
    <SlideShow pos="0 1.8 5">
      <Image id="http://i.imgur.com/ff3Wn06.png" />
      <Image id="http://i.imgur.com/rtCeOEs.png" />
      <Image id="http://i.imgur.com/ByExCaO.png" />
      <Image id="http://i.imgur.com/dYBH57H.png" />
      <Image id="http://i.imgur.com/2PQlhdB.png" />
      <Image id="http://i.imgur.com/xNEw11S.png" />
    </SlideShow>
  </Room>
</FireBoxRoom>

janusbase

All scripting objects inherit from the janusbase type. This class, and all those that inherit from it, expose the following API:

Properties

      object.parent         // Returns a reference to this object's parent
      object.children       // Returns an array of references to any children this object might have
      object.js_id          // the object's unique ID
      object.pos:           // object's position, relative to its parent
      object.vel:           // object's velocity
      object.accel:         // object's acceleration
      object.mass:          // object's physical mass
      object.scale:         // scaling factor for this object and its children
      object.col:           // color override
      object.xdir           // X-axis component of rotation matrix
      object.ydir           // Y-axis component of rotation matrix
      object.zdir           // Z-axis component of rotation matrix
      object.fwd            // alias for zdir
      object.rotation       // Object rotation in euler angles
      object.rotation_order // euler angle order
      object.sync           // Sync object over network when changed
      object.visible        // Show or hide object

Event callbacks

      // Mouse events
      object.onmouseover 
      object.onmouseout
      object.onmousemove
      object.onmousedown
      object.onmouseup
      object.onclick

      // Touch events
      object.ontouchstart
      object.ontouchmove
      object.ontouchend

      // Drag and drop events
      object.ondragover
      object.ondrag
      object.ondragenter
      object.ondragleave
      object.ondragstart
      object.ondragend
      object.ondrop

      // Physics events
      object.oncollision // called when this object collides with another object

Functions

      createObject(type, args) // Spawn a new object as a child of this one
      appendChild(child) // Attach the specified child object as a child of this one
      addEventListener(type, callback) // Register callbacks to be fired when this object throws the specified event
      removeEventListener(type, callback) // Unregister event callbacks
      localToWorld(localpoint) // Transform a vector from object's local space to world space
      worldToLocal(worldpoint) // Transform a vector from world space to object's local space
      distanceTo(other) // Calculate the scalar distance from this object to another
      addForce(type, args) // Attach a physical force to this object (supported types: 'static', 'gravity', 'spring', 'buoyancy', 'friction', 'drag', 'aero', 'aerocontrol', 'magnet', 'electrostatic')
      removeForce(force) // Remove the specified force from this body
      die() // Destroy this object and remove it from the scene
      executeCallback:     ['function', 'executeCallback']

}