Brushes = require "./brushes"
{circle, line, rect, rectOutline, endDeltoid} = require "./util"
neighbors = (point) ->
[
{x: point.x, y: point.y-1}
{x: point.x-1, y: point.y}
{x: point.x+1, y: point.y}
{x: point.x, y: point.y+1}
]
shapeTool = (hotkey, offsetX, offsetY, icon, fn) ->
start = null
end = null
hotkeys: hotkey
iconUrl: icon
iconOffset:
x: offsetX
y: offsetY
touch: ({position}) ->
start = position
move: ({editor, position}) ->
end = position
editor.restore()
fn(editor, editor.canvas, start, end)
release: ({position, editor}) ->
editor.restore()
fn(editor, editor.canvas, start, end)
sizedTool = (hotkey, offsetX, offsetY, icon, options) ->
previousPosition = null
OP = (out) ->
(p) ->
out(p, options)
paint = (out) ->
(p) ->
brush = Brushes.sizes[self.settings.size.value()]
brush(p).forEach OP out
self =
hotkeys: hotkey
iconUrl: icon
iconOffset:
x: offsetX
y: offsetY
touch: ({position, editor})->
paint(editor.draw) position
previousPosition = position
move: ({editor, position})->
line previousPosition, position, paint(editor.draw)
previousPosition = position
release: ->
previousPosition = null
settings:
size:
type: 'range'
min: 0
max: 3
value: Observable 0
Default tools.
TOOLS =
Draw a line when moving while touching.
pencil: sizedTool "p", 4, 14,
""
eraser: sizedTool "e", 4, 11,
"",
color: "transparent"
dropper:
hotkeys: "i"
iconUrl: ""
iconOffset:
x: 13
y: 13
touch: ({position, editor}) ->
editor.activeColor(editor.getColor(position))
move: ({position, editor}) ->
editor.activeColor(editor.getColor(position))
release: ->
# Return to the previous tool
editor.activeTool editor.previousTool()
move: require("./tools/selection")()
Fill a connected area.
fill:
hotkeys: "f"
iconUrl: ""
iconOffset:
x: 12
y: 13
touch: ({position, editor}) ->
color = editor.colorAsInt()
imageData = editor.getSnapshot()
{width, height} = imageData
data = new Uint32Array(imageData.data.buffer)
set = ({x, y}, color) ->
if 0 <= x < width
if 0 <= y < height
data[y * width + x] = color
get = ({x, y}) ->
if 0 <= x < width
if 0 <= y < height
data[y * width + x]
target = get(position)
return unless target?
return if color is target
queue = [position]
set(position, color)
# Allow for interrupts if it takes too long
safetyHatch = width * height
while(queue.length and safetyHatch > 0)
position = queue.pop()
neighbors(position).forEach (position) ->
pixelColor = get(position)
if pixelColor is target
# This is here because I HAVE been burned
# Later I should fix the underlying cause, but it seems handy to keep
# a hatch on any while loops.
safetyHatch -= 1
set position, color
queue.push(position)
editor.putImageData(imageData)
return
move: ->
release: ->
rect: shapeTool "r", 1, 4,
""
(editor, canvas, start, end) ->
color = editor.activeColor()
delta = end.subtract(start)
editor.withCanvasMods (canvas) ->
canvas.drawRect
x: start.x
y: start.y
width: delta.x
height: delta.y
color: color
rectOutline: shapeTool "shift+r", 1, 4,
""
(editor, canvas, start, end) ->
delta = end.subtract(start)
color = editor.activeColor()
editor.withCanvasMods (canvas) ->
canvas.drawRect
x: start.x - 0.5
y: start.y - 0.5
width: delta.x
height: delta.y
stroke:
color: color
width: 1
circle: shapeTool "c", 0, 0, # TODO: Real offset
""
(editor, canvas, start, end) ->
circle start, end, (x, y) ->
editor.draw({x, y})
line: shapeTool "l", 0, 0,
""
(editor, canvas, start, end) ->
color = editor.activeColor()
# Have to draw our own lines if we want them crisp ;_;
line start, end, editor.draw
module.exports = (I={}, self=Core(I)) ->
self.extend
addTool: (tool) ->
[].concat(tool.hotkeys or []).forEach (hotkey) ->
self.addHotkey
hotkey: hotkey
method: -> self.activeTool tool
self.tools.push tool
activeTool: Observable()
detailTool: Observable()
previousTool: Observable()
tools: Observable []
# TODO: Probably want to let the editor add its own tools so this is more
# reusable
Object.keys(TOOLS).forEach (name) ->
self.addTool TOOLS[name]
setNthTool = (n) ->
->
if tool = self.tools.get(n)
self.activeTool tool
[1..9].forEach (n) ->
self.addHotkey
hotkey: n.toString()
method: setNthTool(n-1)
self.addHotkey
hotkey: "0",
method: setNthTool(9)
prevTool = null
self.activeTool.observe (newTool) ->
self.previousTool prevTool
prevTool = newTool
self.activeTool(self.tools()[0])
return self