Skip to content

Commit

Permalink
Update painter example
Browse files Browse the repository at this point in the history
Battle hardened! This should stop certain people (you know who you are) from messing with the service :)
Also added support for touch pressure where available
  • Loading branch information
jnordberg committed May 2, 2017
1 parent 8a01ffd commit 7d1615f
Show file tree
Hide file tree
Showing 6 changed files with 128 additions and 37 deletions.
59 changes: 47 additions & 12 deletions examples/painter/client/contents/paint.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@

import {Client} from 'wsrpc'
import {Painter, PaintEvent, StatusEvent} from './../../protocol/service'
import {Painter, PaintEvent, StatusEvent, IPaintEvent} from './../../protocol/service'
import * as zlib from 'browserify-zlib-next'
import * as shared from './../../shared/paint'

Expand All @@ -13,6 +13,7 @@ interface Position {
interface DrawEvent {
pos: Position
lastPos?: Position
force?: number
color: number
}

Expand Down Expand Up @@ -85,7 +86,6 @@ window.addEventListener('DOMContentLoaded', async () => {
colorWells.forEach((el) => el.classList.remove('active'))
well.classList.add('active')
activeColor = color
console.log(activeColor)
})
colorWells.push(well)
colorPicker.appendChild(well)
Expand All @@ -104,7 +104,14 @@ window.addEventListener('DOMContentLoaded', async () => {
shared.paint(event, ctx)
})

const loadingEl = document.createElement('div')
loadingEl.className = 'loading'
loadingEl.innerHTML = 'Loading canvas...'
document.body.appendChild(loadingEl)

async function fetchCanvas() {
document.documentElement.classList.add('loading')

const request = {
width: Math.min(window.innerWidth, shared.canvasWidth),
height: Math.min(window.innerHeight, shared.canvasHeight),
Expand All @@ -129,6 +136,8 @@ window.addEventListener('DOMContentLoaded', async () => {
const imageData = ctx.createImageData(request.width, request.height)
imageData.data.set(new Uint8ClampedArray(data.buffer))
ctx.putImageData(imageData, 0, 0)

document.documentElement.classList.remove('loading')
}

let debounceTimer
Expand All @@ -142,30 +151,54 @@ window.addEventListener('DOMContentLoaded', async () => {
canvas.width = window.innerWidth
canvas.height = window.innerHeight
clearTimeout(debounceTimer)
setTimeout(fetchCanvas, 1000)
debounceTimer = setTimeout(fetchCanvas, 500)
document.documentElement.classList.add('loading')
}
})

await fetchCanvas()

function draw(event: DrawEvent) {
let velocity = 0
async function drawAsync(event: DrawEvent) {
let msgs: IPaintEvent[] = []
let size = 20
const color = event.color
if (event.force) {
size = Math.min(size + event.force * (shared.brushSize - 20), shared.brushSize)
}
if (event.lastPos) {
const dx = event.lastPos.x - event.pos.x
const dy = event.lastPos.y - event.pos.y
const dt = event.pos.timestamp - event.lastPos.timestamp
velocity = Math.sqrt(dx*dx + dy*dy) / dt
const d = Math.sqrt(dx*dx + dy*dy)
if (!event.force) {
const dt = event.pos.timestamp - event.lastPos.timestamp
size = Math.min(size + 20 * (d / dt), shared.brushSize)
}
const interpSteps = ~~(d / (size / 4))
for (let i = 0; i < interpSteps; i++) {
const p = (i + 1) / (interpSteps + 1)
const x = event.lastPos.x * p + event.pos.x * (1 - p)
const y = event.lastPos.y * p + event.pos.y * (1 - p)
msgs.push({x, y, color, size})
}
}
const msg = {
msgs.push({
x: event.pos.x,
y: event.pos.y,
color: event.color,
size: 20 + velocity * 20
size: Math.min(size, shared.brushSize),
})
let drawCalls = []
for (const msg of msgs) {
shared.paint(msg, ctx)
drawCalls.push(client.service.paint(msg))
}
client.service.paint(msg).catch((error: Error) => {
console.warn('error drawing', error.message)
await Promise.all(drawCalls)
}

function draw(event: DrawEvent) {
drawAsync(event).catch((error) => {
console.warn('error drawing', error)
})
shared.paint(msg, ctx)
}

let mouseDraw: DrawEvent|undefined
Expand Down Expand Up @@ -212,6 +245,7 @@ window.addEventListener('DOMContentLoaded', async () => {
y: touch.screenY,
timestamp: now(),
},
force: touch['force'],
color: activeColor
}
draw(fingerDraw[touch.identifier])
Expand All @@ -230,6 +264,7 @@ window.addEventListener('DOMContentLoaded', async () => {
y: touch.screenY,
timestamp: now(),
}
drawEvent.force = touch['force']
draw(drawEvent)
}
}
Expand Down
18 changes: 18 additions & 0 deletions examples/painter/client/contents/style.css
Original file line number Diff line number Diff line change
Expand Up @@ -62,4 +62,22 @@ canvas {
outline: 0.2em solid white;
border-left: 0.2em solid black;
z-index: 1;
}

div.loading {
text-align: center;
position: absolute;
left: 50%;
top: 50%;
width: 10em;
height: 2em;
line-height: 2em;
margin-left: -5em;
margin-top: -1em;
background: rgba(0, 0, 0, 0.45);
display: none;
}

.loading div.loading {
display: block;
}
2 changes: 2 additions & 0 deletions examples/painter/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -5,8 +5,10 @@
"canvas": "./client/canvas.js"
},
"dependencies": {
"@types/lru-cache": "^4.0.0",
"browserify-zlib-next": "^1.0.1",
"canvas": "github:automattic/node-canvas",
"lru-cache": "^4.0.2",
"node-dev": "^3.1.3",
"protobufjs": "github:dcodeio/protobuf.js#3a95945",
"ts-node": "^3.0.2",
Expand Down
14 changes: 7 additions & 7 deletions examples/painter/protocol/service.proto
Original file line number Diff line number Diff line change
Expand Up @@ -8,19 +8,19 @@ service Painter {
message Empty {}

message PaintEvent {
required float x = 1;
required float y = 2;
required float size = 3;
required int32 color = 4;
required int32 x = 1;
required int32 y = 2;
required uint32 size = 3;
required uint32 color = 4;
}

message StatusEvent {
required int32 users = 1;
required uint32 users = 1;
}

message CanvasRequest {
required int32 width = 1;
required int32 height = 2;
required uint32 width = 1;
required uint32 height = 2;
}

message CanvasResponse {
Expand Down
20 changes: 20 additions & 0 deletions examples/painter/server/server.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ import * as protobuf from 'protobufjs'
import * as zlib from 'zlib'
import * as Canvas from 'canvas'
import * as fs from 'fs'
import * as path from 'path'

import {PaintEvent, StatusEvent, CanvasRequest} from './../protocol/service'
import * as shared from './../shared/paint'
Expand Down Expand Up @@ -31,13 +32,32 @@ process.on('exit', () => {
})
process.on('SIGINT', () => process.exit())

let canvasDirty = false
if (process.env['SAVE_INTERVAL'] && process.env['SAVE_DIR']) {
const interval = parseInt(process.env['SAVE_INTERVAL'])
const dir = process.env['SAVE_DIR']
console.log(`saving canvas to ${ dir } every ${ interval } seconds`)
const save = async () => {
if (!canvasDirty) {
console.log('canvas not dirty, skipping save')
return
}
const filename = path.join(dir, `canvas-${ new Date().toISOString() }.png`)
console.log(`saving canvas to ${ filename }`)
canvas.pngStream().pipe(fs.createWriteStream(filename))
canvasDirty = false
}
setInterval(save, interval * 1000)
}

const server = new wsrpc.Server({
port: 4242,
service: proto.lookupService('Painter')
})

server.implement('paint', async (event: PaintEvent, sender) => {
shared.paint(event, ctx)
canvasDirty = true
const broadcast = PaintEvent.encode(event).finish()
for (const connection of server.connections) {
if (connection === sender) {
Expand Down
52 changes: 34 additions & 18 deletions examples/painter/shared/paint.ts
Original file line number Diff line number Diff line change
@@ -1,37 +1,38 @@
import {IPaintEvent} from './../protocol/service'
const Canvas = require('canvas')
import * as LRUCache from 'lru-cache'

export const canvasWidth = process.env['CANVAS_WIDTH'] ? parseInt(process.env['CANVAS_WIDTH']) : 2048
export const canvasHeight = process.env['CANVAS_HEIGHT'] ? parseInt(process.env['CANVAS_HEIGHT']) :2048

const brushSize = 124
const brushCache: {[color: number]: HTMLCanvasElement} = {}
export const brushSize = 124
const brushCache = LRUCache<HTMLCanvasElement>({max: 20})
const brushImage = new Canvas.Image()
brushImage.src = ''

let brushData: Uint8ClampedArray

function createCanvas():HTMLCanvasElement {
function createCanvas(width: number, height: number):HTMLCanvasElement {
if (process.title === 'browser') {
return document.createElement('canvas')
const rv = document.createElement('canvas')
rv.width = width
rv.height = height
return rv
} else {
return new Canvas()
return new Canvas(width, height)
}
}

function getBrush(color: number) {
if (brushCache[color]) {
return brushCache[color]
function getBrush(color: number):HTMLCanvasElement {
if (brushCache.has(color)) {
return brushCache.get(color)
}

const r = (color >> 16) & 0xff
const g = (color >> 8) & 0xff
const b = color & 0xff

const brush = createCanvas()
brush.width = brushSize
brush.height = brushSize

const brush = createCanvas(brushSize, brushSize)
const ctx = brush.getContext('2d')

if (!brushData) {
Expand All @@ -50,15 +51,30 @@ function getBrush(color: number) {

ctx.putImageData(imageData, 0, 0)

brushCache[color] = brush
brushCache.set(color, brush)
return brush
}

export function paint(p: IPaintEvent, ctx: CanvasRenderingContext2D) {
export function paint(event: IPaintEvent, ctx: CanvasRenderingContext2D) {
ctx.globalAlpha = 0.4
ctx.globalCompositeOperation = 'source-over'
let s = Math.min(p.size, 124)
if (s < 1) { s = 1 }
const o = s / 2
ctx.drawImage(getBrush(p.color), ~~(p.x - o), ~~(p.y - o), ~~s, ~~s)

if (event.size < 1 || event.size > brushSize) {
throw new Error('Invalid size')
}

if (Math.abs(event.x) > 0xffff || Math.abs(event.y) > 0xffff) {
throw new Error('Invalid position')
}

if (event.color > 0xffffff) {
throw new Error('Invalid color')
}

const offset = ~~(event.size / 2)
const x = ~~(event.x - offset)
const y = ~~(event.y - offset)
const s = ~~event.size

ctx.drawImage(getBrush(event.color), x, y, s, s)
}

0 comments on commit 7d1615f

Please sign in to comment.