WAMS is a Web API that makes creating Multi-Screen applications easy. Multi-screen applications are ones where multiple devices (and their screens) can be used together in flexible ways allowing objects to be easily moved between screens or interactions, like gestures, to span multiple screens.
WAMS abstracts away socket connections with clients as well as the model, interaction, and coordination logic, providing a simple, unified framework that allows focus to be placed on creating and defining new interactions rather than on pre-requisite communication code. It comes packaged with default server logic and HTML files so you can easily build canvas-first shared workspace applications, or you can provide your own server routines, allowing you to integrate WAMS into your existing web application!
We use browser windows to represent screens because browsers are extremely flexible - they are cross-platform, already available on multiple devices and can fill screens (or be used to subdivide screens).
You will need to install node.js and npm.
Then you can install this repo directly as a node module. For example, to install the stable hello-world-test
release:
npm install wams-js/wams#hello-world-test
The easiest way to get started is to follow the Walkthrough tutorial below. More advanced users might want to check the code documentation and the examples. For a taste on how WAMS works, check the live demo section. (The live demo is currently broken).
To try out the examples, go to examples/
and run as follows:
node examples/[EXAMPLE_FILENAME]
For example:
node examples/polygons.js
This walkthrough is a friendly guide on how to use most WAMS features. For more details, see code documentation.
Note The examples on this page use ES2015 (ES6) JavaScript syntax like
const
variables and object destructuring. If you are not familiar with ES2015 features, you can read about them first.
First, let's set up a new directory for our demo application, and install WAMS into it:
mkdir demo
cd demo
npm install wams-js/wams#hello-world-test
Now, create an app.js file. In this file, include WAMS and initialize the application:
const WAMS = require("wams");
const app = new WAMS.Application();
app.listen(9000); // this starts the app on port 9000, you can use any port
Now, you can run your first WAMS application by executing:
node app.js
And you can connect to the app using the address in the output.
- Many terminal emulators let you control+click on an address to visit it in your browser.
Let's now make your first WAMS app do something. Add the following code just before the last line:
// the line below is equivalent to `const square = WAMS.predefined.items.square;`
const { square } = WAMS.predefined.items;
// spawn a square on the screen
app.spawn(square(100, "green", { x: 200, y: 200 }));
The first arguments describe the length and colour of the square. The last argument is an object that describes how the square should be oriented within the WAMS workspace. This code creates a green square on the canvas centered at coordinates { x: 200, y: 200 }
and a length of 100
.
Here is a simple example to show how several screens work with WAMS.
This example will spawn a draggable square and position connected screens in a line.
Put this code in your app.js file:
const WAMS = require("wams");
const app = new WAMS.Application();
const { actions, items, layouts } = WAMS.predefined;
function spawnSquare() {
const greenSquare = app.spawn(items.square(100, "green", { x: 200, y: 200 }));
greenSquare.on('drag', actions.drag);
}
const line = new layouts.LineLayout(300); // 300px overlap betweens views
function handleConnect({ view, device }) {
view.on('click', spawnSquare);
line.layout(view, device);
}
app.on('connect', handleConnect);
app.listen(9000);
Don't worry if the code doesn't make sense to you yet. The walkthrough will explain all the features used in it.
The square can now be moved around and seen by multiple screens with less than 20 lines of code 🎉
To test this on a single computer you could:
- open one browser window covering half your screen and position it on the left
- open another browser to the same address and position it on the right
- now click on an empty area to create a square
- click and hold to drag the square towards the other browser window
- you have your first multiscreen app!
To try a more complex multi-screen gestures example (gestures that span multiple screens), check out
examples/shared-polygons.js
.
The application can be configured through some options.
Below is the full list of possible options with example values.
const app = new WAMS.Application({
clientLimit: 2, // maximum number of devices that can connect to the app
shadows: true, // show shadows of other devices
status: true, // show information on current view, useful for debugging
// The following options can be accomplished directly through HTML and CSS,
// if you provide your own static files and server routine.
backgroundImage: "./monaLisa", // background image of the app
color: "white", // background color of the app
clientScripts: ["script.js"], // javascript scripts (relative paths or URLs) to include by the browser
stylesheets: ["styles.css"], // css styles to include by the browser
title: "Awesome App", // page title
});
You can substitute const app = new Wams.Application();
in your code with the code above to play with different options.
A WAMS app is made of items. There are several predefined items (see in the code documentation):
rectangle
square
circle
oval
line
polygon
image
html
Most of the items (except html
) are rendered onto an HTML canvas, which is the core of WAMS (i.e., in WAMS everything is drawn on HTML canvas, although for the most part, you do not need to know about this).
You have already seen square
used in the Hello world example above. Now let's look at some other items.
// application setup omitted here
// and in following examples
const { polygon } = WAMS.predefined.items;
const points = [
{ x: 0, y: 0 },
{ x: 50, y: 0 },
{ x: 25, y: 50 },
];
app.spawn(
polygon(points, "green", {
x: 500,
y: 100,
})
);
Polygons are built using an array of relative points. For a random set of points, you can use randomPoints
method from Wams.predefined.utilities
(see in code documentation).
For example:
const { items, utilities } = WAMS.predefined;
const points = utilities.randomPoints(4);
app.spawn(
items.polygon(points, "green", {
x: 500,
y: 100,
})
);
To use images, if you are using the default server provided by WAMS, you first need to set up a path to the static directory.
For this example, create an images
directory in the app folder and use it as your static directory.
Put monaLisa.jpg
from examples/img
to the images folder.
const path = require("node:path");
const { image } = WAMS.predefined.items;
app.addStaticDirectory(path.join(__dirname, "images"));
app.spawn(
image("monaLisa.jpg", {
width: 200,
height: 350,
x: 300,
y: 300,
})
);
Make sure to include width and height.
Example To see a great example of using images, check out
examples/card-table.js
If you need more control over styling than a canvas provides, or you would like to use iframe
, audio
, video
or other browser elements, WAMS also supports spawning HTML items.
const { html } = WAMS.predefined.items;
app.spawn(
html("<h1>Hello world!</h1>", 200, 100, {
x: 300,
y: 100,
})
);
The code above will spawn a wrapped h1
element with width of 200
and height of 100
, positioned at { x: 300, y: 100 }
.
You can set initial scale and rotation of an item:
app.spawn(
polygon(points, "green", {
x: 500,
y: 100,
scale: 2,
rotation: Math.PI,
})
);
Note Rotation is done around the (x, y) point provided and is defined in radians (Pi = 180 degrees)
To make an item draggable, it's enough to attach the predefined drag action to the drag event:
const item = app.spawn(items.square(100, 'green', { x: 200, y: 200 }));
item.on('drag', actions.drag);
This looks much better. Now let's remove the square when you click on it. To remove an item, use WAMS' removeItem
method.
item.on('click', () => app.removeItem(item));
Another cool interactive feature is rotation. To rotate an item, first add a rotate
listener, (the predefined action will do the trick), and then grab the item with your mouse and hold Control key.
item.on('rotate', actions.rotate);
Rotation using mouse + CTRL will pivot around whichever point on the item your cursor was at when you pressed the CTRL key.
You can also listen to swipe events on items (hold the item, quickly move it and release). To do that, add a swipe
handler.
function handleSwipe(event) {
console.log(`Swipe registered!`);
console.log(`Velocity: ${event.velocity}`);
console.log(`Direction: ${event.direction}`);
console.log(`X, Y: ${event.x}, ${event.y}`);
}
item.on('swipe', handleSwipe);
To move an item, you can use moveBy
and moveTo
item methods:
// do this on a different item than the one that uses removeItem
item.on('click', () => item.moveBy(100, -50));
Both methods accept x
and y
numbers that represent a vector (for moveBy
) or the final position (for moveTo
).
You can add event handlers to all WAMS items.
Often times, you want to use images, run custom code in the browser, or add CSS stylesheets. If you're using your own server routine, you can ignore this section and serve static files the way you normally would.
If you want to do this with the default server implementation provided by WAMS, you first need to set up a path to the static directory:
const path = require("node:path");
const app = new WAMS.Application();
app.addStaticDirectory(path.join(__dirname, "assets"));
This makes files under the specified path available at the root URL of the application. For example, if you have the same configuration as above, and there is an image.png
file in the assets
folder, it will be available at http(s)://<app-url>/image.png
- To run code in the browsers that use your app, create a .js file in your app static directory and include it in the application config:
const app = new WAMS.Application({
clientScripts: ["js/awesome-script.js"],
});
- To add CSS stylesheets:
const app = new WAMS.Application({
stylesheets: ["css/amazing-styles.css"],
});
WAMS will load these scripts and stylesheets in the browser window.
WAMS manages all connections under the hood, and provides helpful ways to react on connection-related events:
connect
– emitted each time a screen connects to a WAMS applicationdisconnect
– emitted when a screen disconnects
You can listen for both events on the Application instance. The handler function gets an event object with these properties:
view
device
group
View
is an object that stores the state of the connected screen, including:
index
topLeft
,topRight
,bottomRight
andbottomLeft
positionsscale
rotation
width
height
It also provides methods to transform the current screen's view:
moveBy
moveTo
rotateBy
scaleBy
And you can set up event listeners for the view itself, such as:
drag
rotate
pinch
click
swipe
disconnected
pointerdown
pointermove
pointerup
Device
stores dimensions of the screen and its original position when connected.
Group
is a collection of views that will have their inputs combined together into multi-device gestures! It is a good idea to use a non-overlapping layout for the devices and views in a group, and make sure that the devices and views are laid out similarly.
By default, every connected screen is positioned in the same location and can see the same objects. However, you can build more complex layouts by using view
, device
and group
objects' methods and state, or use one of the out-of-box predefined layouts.
There are currently two predefined layouts: TableLayout
and LineLayout
.
TableLayout
Places users around a table, with the given amount of overlap. The first user will be the "table", and their position when they join is stamped as the outline of the table. The next four users are positioned, facing inwards, around the four sides of the table.
const { table } = WAMS.predefined.layouts;
const overlap = 200; // 200px overlap between screens
const setTableLayout = table(overlap);
function handleConnect({ view, device }) { // note the {} brackets to destructure the event object
setTableLayout(view, device);
}
app.on('connect', handleConnect);
To see this layout in action, check out the card-table.js
example.
LineLayout
Places users in a line, with the given amount of overlap. Best used with either multi-screen gestures or when users are unable to manipulate their views.
const { LineLayout } = WAMS.predefined.layouts;
const overlap = 200; // 200px overlap between screens
// If using multi-device gestures, it's usually best to use an overlap of 0.
const line = new LineLayout(overlap);
function handleConnect({ view, device }) { // note the {} brackets to destructure the event object
line.layout(view, device);
}
app.on('connect', handleConnect);
To see this layout in action with multi-screen gestures, check out the shared-polygons.js
example.
When building more complex applications, sometimes you might want to have more flexibility than predefined items and behaviors provide.
The following sections show how to go beyond that.
To spawn a custom item, use CanvasSequence
. It allows to create a custom sequence of canvas actions and you can use most of the HTML Canvas methods as if you are writing regular canvas code.
The following sequence draws a smiling face item:
function smileFace(args) {
const sequence = new WAMS.CanvasSequence();
sequence.beginPath();
sequence.arc(75, 75, 50, 0, Math.PI * 2, true); // Outer circle
sequence.moveTo(110, 75);
sequence.arc(75, 75, 35, 0, Math.PI, false); // Mouth (clockwise)
sequence.moveTo(65, 65);
sequence.arc(60, 65, 5, 0, Math.PI * 2, true); // Left eye
sequence.moveTo(95, 65);
sequence.arc(90, 65, 5, 0, Math.PI * 2, true); // Right eye
sequence.stroke();
return { ...args, sequence };
}
app.spawn(smileFace({ x: 400, y: 300 }));
You can add interactivity to a custom item the same way as with predefined items. However, you first need to add a hitbox to the item. This can be a bit confusing, since the hitbox will always be given (x, y) values as if its item is located at (0, 0). Put another way, the hitbox doesn't need to know anything about how the item is positioned or oriented in the WAMS workspace:
function interactableSmileFace(args) {
const hitbox = new WAMS.Circle(
50, // 50 is the radius of the outer circle of the smiley
75, // the smiley is centered at (75, 75)
75,
);
return smileFace({ ...args, hitbox });
}
// The Circle doesn't need to know that we're creating the smiley at (400, 200) in the workspace
const item = app.spawn(interactableSmileFace({ x: 400, y: 200 }));
item.on('drag', actions.drag);
A hitbox can be made from WAMS.Rectangle
, WAMS.Polygon2D
, WAMS.Circle
, WAMS.Oval
, or WAMS.RoundedLine
.
WAMS.Polygon2D
accepts an array of points – vertices of the resulting polygon.
Sometimes, you would like to tell devices to execute client-side code at a specific time. Or you would like to communicate some client-side event to the server. To allow that, WAMS provides custom events.
Let's say we would like to send a message from the client to the server. WAMS methods are exposed to the client via the global WAMS
object.
To dispatch an event to the server, use WAMS.dispatch()
method:
// client.js
WAMS.dispatch("my-message", { foo: "bar" });
This dispatches a custom event to the server called my-message
and sends a payload object.
To listen to this event on the server, use app.on()
method:
// app.js
app.on("my-message", handleMyMessage);
function handleMyMessage(data) {
console.log(data.foo); // logs 'bar' to the server terminal
}
To dispatch an event to the client from the server, use app.dispatch()
method.
// app.js
app.dispatch("my-other-message", { bar: "foo" });
To listen to this event on the client, use WAMS.on()
method:
// client.js
WAMS.on("my-other-message", handleMyOtherMessage);
function handleMyOtherMessage(data) {
console.log(data.bar); // logs 'foo' to the browser console
}
Under the hood, client-side events are implemented with the DOM's CustomEvent. If you want to trigger a WAMS client event on the client, you can dispatch a custom event on the document element.
To give different views different rights for interacting with items, use view.index
to differentiate between connected devices.
A view is assigned the lowest free
index
, starting with 0. When a view with lower index disconnects, other connected views' indices stay the same.
For example, let's say we are making a card game and would like to only allow a card owner to flip it.
To do that, first we'll add an index to the card item to show who its owner is.
// during creation
const card = app.spawn(
image(url, {
/* ... */
owner: 1,
})
);
// or later
card.owner = 1;
The owner
property does not have a special meaning. You can use any property of any type.
Now, we will only flip the card if the event comes from the card owner:
function flipCard(event) {
if (event.view.index !== event.target.owner) return;
const card = event.target;
// assume we've attach 'back' and 'face' properties to the card with paths to images
const imgsrc = card.isFaceUp ? card.back : card.face;
card.setImage(imgsrc);
card.isFaceUp = !card.isFaceUp;
}
Sometimes you would like to spawn several items and then move or drag them together. To do that easily, you can use the createItemGroup
method (see in the code documentation):
const items = [];
items.push(
app.spawn(
html("<h1>hello world</h1>", 300, 100, {
x: 300,
y: 300,
})
)
);
items.push(app.spawn(square(200, "yellow", { x: 100, y: 100 })));
items.push(app.spawn(square(200, "blue", { x: 150, y: 150 })));
const group = app.createItemGroup({ items });
group.on('drag', actions.drag);
group.moveTo(500, 300);
Groups can only be moved together- rotation and scaling are not supported.
We welcome contribution to WAMS, please find details on how to setup your local development environment.
WAMS is a Node.js projects, and with that requires that you have Node installed. We recommend using ASDF or NVM as your Node Version Manager to install the correct version of Node.
Step #1: Install Node.js
# Using ASDF
asdf current # This will list the version of Node for this project
asdf install nodejs <version> # Install the correct version of Node
# Using NVM
nvm install
Step #2: Install dependencies
npm install
The package.json file includes scripts to run the linter and formatter:
# Check for errors
npm run lint
# Fix linting errors that can be auto-fixed:
npm run lint:fix
# Auto-format code:
npm run format
To enable automatic linting and saving on save, you can download the ESLint VS Code extension. All other required settings are included in the .vscode/settings.json
file.
The documentation is served from the branch gh-pages
. This branch is built using a github action, which is triggered by pushes to the deploy-docs
branch.
When you want to have changes in the code reflected in the documentation:
- Test the documentation changes locally using
npm run build
then openingdocs/index.html
and browsing. There is no need to commit the docs folder. - Open a PR and merge your changes into main.
- Merge main into
deploy-docs
and push the changes to github.