Skip to content

Instantly share code, notes, and snippets.

@dmirtyisme
Created February 11, 2016 10:19
Show Gist options
  • Save dmirtyisme/1f3db52a1ca0d6392e5a to your computer and use it in GitHub Desktop.
Save dmirtyisme/1f3db52a1ca0d6392e5a to your computer and use it in GitHub Desktop.
HTML5 Drawing Studio
<!--
Documentation and changelog available on GitHub:
https://github.com/boylett/HTML5-Drawing-Studio
-->
<div class="controls">
<a href="#save" onclick="return Artboard.Save(), !1" class="fa fa-save" title="Save .h5i File"></a>
<a href="#import" class="fa fa-folder-open-o"><input type="file" onchange="Artboard.Import(this)" title="Import Image File" /></a>
<a href="#export" onclick="return Artboard.Export(), !1" class="fa fa-mail-forward" title="Export As PNG"></a>
</div>
<div class="controls right">
<a href="#undo" onclick="return Artboard.Undo(), !1" class="fa fa-rotate-left" title="Undo"></a>
<a href="#redo" onclick="return Artboard.Redo(), !1" class="fa fa-rotate-right" title="Redo"></a>
</div>
<div class="sidebar">
<div class="controls-section top">
<div class="controls layers">
<a href="#add" onclick="return Artboard.Layers.Add().Focus(), !1" class="fa fa-plus"></a>
<h1>Layers</h1>
<ul></ul>
</div>
</div>
<div class="controls-section bottom">
<div class="controls thickness">
<h1><span class="current-thickness">10</span> Thickness</h1>
<input type="range" min="1" max="100" value="10" name="thickness" />
</div>
<div class="controls color-picker">
<span class="current-color" style="background-color: #000;"></span>
<h1>Colour</h1>
</div>
</div>
</div>
<div class="artboards"></div>
console.clear();
console.log('HTML Drawing Studio v1.0.1', ' ', ' ', 'Check the HTML tab for a changelog');
var Brush = function(options)
{
var $this = this;
$this.Foreground = '#000000';
$this.Thickness = 10;
$this.LastPaint = null;
$this.Multitouch = false;
$this.Start = function(e)
{
Artboard.Layer.Context.globalCompositeOperation = 'source-over';
Artboard.Layer.Context.strokeStyle = $this.Foreground;
Artboard.Layer.Context.lineJoin =
Artboard.Layer.Context.lineCap = 'round';
$this.LastPaint = null;
};
$this.Stop = function(e)
{
$this.LastPaint = null;
};
$this.Draw = function(e)
{
if($this.LastPaint)
{
Artboard.Layer.Context.beginPath();
Artboard.Layer.Context.lineWidth = e.brushPressure * $this.Thickness;
Artboard.Layer.Context.moveTo($this.LastPaint[0], $this.LastPaint[1]);
Artboard.Layer.Context.lineTo(e.brushX, e.brushY);
Artboard.Layer.Context.stroke();
}
$this.LastPaint = [e.brushX, e.brushY, e.brushPressure];
};
$this.Pick = function()
{
Artboard.Brush = $this;
Artboard.ColorPickerLabel.css('background-color', $this.Foreground);
Artboard.ThicknessLabel.html($this.Thickness);
};
for(var i in options)
{
$this[i] = options[i];
}
if($this.Init)
{
$this.Init.call($this);
}
return $this;
};
var Layer = function()
{
var $this = this;
$this.Active = false;
$this.Hidden = false;
$this.Canvas = $('<canvas class="layer">').prependTo('.artboards').data('Layer', $this);
$this.Context = $this.Canvas[0].getContext('2d');
$this.Drawing = false;
$this.Erasing = false;
$this.Index = -1;
Object.defineProperty($this, 'Name',
{
get: function()
{
return $this.MenuItem.find('> font').text().trim();
},
set: function()
{
$this.MenuItem.find('> font').html(arguments[0]);
}
});
$this.Resize = function(width, height)
{
$this.Canvas
.attr('width', (width !== undefined) ? width : $this.Canvas.parent().width())
.attr('height', (height !== undefined) ? height : $this.Canvas.parent().height());
return $this;
};
$this.MenuItem = $('<div class="layer"><font>Layer ' + ($this.Index + 1) + '</font></div>')
.prependTo('.sidebar .controls.layers > ul')
.data('Layer', $this)
.on('click', function()
{
return $this.Focus(), !1;
})
.on('dblclick', function()
{
var newname = prompt('Enter a new name for layer ' + ($this.Index + 1) + ':', $this.Name);
if(newname && newname != $this.Name)
{
$this.Name = newname;
}
return false;
});
$this.HideButton = $('<a href="#hide" class="hide-layer"></a>')
.appendTo($this.MenuItem)
.on('mouseenter mouseleave', function(e)
{
$('html')[e.altKey ? 'addClass' : 'removeClass']('alt');
})
.on('click', function(e)
{
if(e.altKey || e.button == 5 || e.button == '5')
{
if(confirm('Are you sure you want to delete ' + $this.Name + '?'))
{
$this.Canvas.remove();
$this.MenuItem.remove();
var layers = [];
for(var i in Artboard.Layers.List)
{
if(Artboard.Layers.List[i] != $this)
{
layers.push(Artboard.Layers.List[i]);
}
}
Artboard.Layers.List = layers;
}
$('html').removeClass('alt');
}
else
{
$this.Hidden = !$this.Hidden;
$this.MenuItem[$this.Hidden ? 'addClass' : 'removeClass']('hidden');
$this.Canvas[$this.Hidden ? 'addClass' : 'removeClass']('hidden');
}
return false;
});
$this.Focus = function()
{
$('.layer.active').removeClass('active');
for(var i in Artboard.Layers.List)
{
Artboard.Layers.List[i].Active = false;
}
Artboard.Layer = $this;
$this.Active = true;
$this.Canvas.addClass('active');
$this.MenuItem.addClass('active');
return $this;
};
$this.GetBrushCoords = function(e)
{
var offset = $this.Context.canvas.getBoundingClientRect(),
x = e.pageX - offset.left,
y = e.pageY - offset.top,
z = ((e.pressure !== undefined) ? e.pressure : ((e.mozPressure !== undefined) ? e.mozPressure : 1));
return (
{
brushX: x,
brushY: y,
brushPressure: z
});
};
$this.Draw = function(e)
{
if(!$this.Drawing) return;
var brush = $this.GetBrushCoords(e);
e.brushX = brush.brushX;
e.brushY = brush.brushY;
e.brushPressure = brush.brushPressure;
($this.Erasing ? Artboard.Brushes.Eraser : Artboard.Brush).Draw(e);
};
$this.DrawStart = function(e)
{
var brush = $this.GetBrushCoords(e);
e.brushX = brush.brushX;
e.brushY = brush.brushY;
e.brushPressure = brush.brushPressure;
$this.Erasing = (e.button == 5 || e.button == '5' || e.altKey);
($this.Erasing ? Artboard.Brushes.Eraser : Artboard.Brush).Start(e);
$this.Drawing = true;
$('html').addClass('drawing');
};
$this.DrawStop = function(e)
{
$this.Drawing = false;
$('html').removeClass('drawing');
var brush = $this.GetBrushCoords(e);
e.brushX = brush.brushX;
e.brushY = brush.brushY;
e.brushPressure = brush.brushPressure;
($this.Erasing ? Artboard.Brushes.Eraser : Artboard.Brush).Stop(e);
$this.Erasing = false;
};
return $this;
};
var Artboard = new (Artboard = function()
{
var $this = this;
Object.defineProperty($this, 'Background',
{
get: function()
{
return $('.artboards').css('background-color');
},
set: function()
{
return $('.artboards').css('background-color', arguments[0]);
}
});
$this.Layer = null;
$this.Layers =
{
List: [],
Add: function(callback)
{
var layer = new Layer();
layer.Index = $this.Layers.List.push(layer) - 1;
layer.MenuItem.find('> font').html('Layer ' + (layer.Index + 1));
layer.Resize();
for(var i in $this.Layers.List)
{
$this.Layers.List[i].Canvas
.prependTo('.artboards')
.css('z-index', i);
}
if(callback)
{
callback.call($this, layer);
}
return layer;
},
Focus: function(index)
{
if($this.Layers.List[index])
{
return $this.Layers.List[index].Focus();
}
return null;
}
};
$('.sidebar .controls-section .layers > ul').sortable(
{
axis: 'y',
container: 'parent',
revert: 250,
stop: function(e, ui)
{
var layers = [];
$($(this).closest('ul').find('> .layer').get().reverse()).each(function()
{
var layer = $(this).data('Layer');
layer.Index = layers.push(layer) - 1;
});
$this.Layers.List = layers;
for(var i in $this.Layers.List)
{
$this.Layers.List[i].Canvas
.prependTo('.artboards')
.css('z-index', i);
}
}
});
$this.Direction = 0;
$this.LastPaint = null;
$this.History = [];
$this.HistoryStep = 0;
$this.Undo = function()
{
if($this.HistoryStep > 0)
{
$this.HistoryStep --;
$this.History[$this.HistoryStep].call($this);
}
return $this;
};
$this.Redo = function()
{
if($this.History[$this.HistoryStep + 1])
{
$this.HistoryStep ++;
$this.History[$this.HistoryStep].call($this);
}
return $this;
};
$this.SaveHistoryState = function(callback)
{
$this.History = $this.History.slice(0, $this.HistoryStep);
$this.HistoryStep = $this.History.push(callback) - 1;
return $this;
};
$this.Brushes =
{
Default: new Brush(),
Eraser: new Brush(
{
Thickness: 20,
Start: function(e)
{
Artboard.Layer.Context.globalCompositeOperation = 'destination-out';
Artboard.Layer.Context.strokeStyle = '#000';
Artboard.Layer.Context.lineJoin =
Artboard.Layer.Context.lineCap = 'round';
this.LastPaint = null;
},
Stop: function(e)
{
$this.Erasing = false;
this.LastPaint = null;
},
Draw: function(e)
{
if(this.LastPaint)
{
Artboard.Layer.Context.beginPath();
Artboard.Layer.Context.lineWidth = e.brushPressure * this.Thickness;
Artboard.Layer.Context.moveTo(this.LastPaint[0], this.LastPaint[1]);
Artboard.Layer.Context.lineTo(e.brushX, e.brushY);
Artboard.Layer.Context.stroke();
}
this.LastPaint = [e.brushX, e.brushY, e.brushPressure];
}
}),
Frayed: new Brush(
{
Thickness: 25,
Start: function(e)
{
Artboard.Layer.Context.globalCompositeOperation = 'source-over';
Artboard.Layer.Context.strokeStyle = $this.Foreground;
Artboard.Layer.Context.lineJoin = 'round';
Artboard.Layer.Context.lineCap = 'butt';
$this.LastPaint = null;
}
}),
Shape: new Brush(
{
Background: 'rgba(255, 0, 0, .25)',
Foreground: '#000000',
Thickness: 5,
Points: [],
Start: function(e)
{
return false;
},
Stop: function(e)
{
return false;
},
Draw: function(e)
{
return false;
}
})
};
$this.Brush = $this.Brushes.Default;
$this.ColorPickerLabel = $('.color-picker .current-color');
$this.ColorPickerCanvas = $('<canvas>')
.appendTo('.color-picker')
.on('mousedown', function(e)
{
return $(this).data('Picking', true), false;
})
.on('mouseup mouseleave', function(e)
{
return $(this).data('Picking', false), false;
})
.on('mousemove', function(e)
{
if($(this).data('Picking'))
{
var off = $(this).offset(),
x = e.pageX - off.left,
y = e.pageY - off.top,
pixel = $this.ColorPicker.getImageData(x, y, 1, 1).data,
color = 'rgb(' + pixel[0] + ',' + pixel[1] + ',' + pixel[2] + ')';
$this.ColorPickerLabel.css('background-color', color);
$this.Brush.Foreground = color;
}
return false;
});
$this.ColorPicker = $this.ColorPickerCanvas[0].getContext('2d');
var img = new Image();
img.onload = function()
{
var width = $this.ColorPickerCanvas.parent().width();
$this.ColorPickerCanvas
.attr('width', width)
.attr('height', width);
$this.ColorPicker.drawImage(this, 0, 0, this.width, this.height, 0, 0, width, width);
};
img.src = '';
$this.ThicknessLabel = $('.controls-section .thickness .current-thickness');
$this.ThicknessSlider = $('.controls-section .thickness input[name="thickness"]')
.on('input', function()
{
$this.ThicknessLabel.html(this.value);
$this.Brush.Thickness = parseInt(this.value);
});
$this.Import = function(file)
{
if(typeof file == 'object' && file.tagName && file.tagName.toLowerCase() == 'input')
{
var input = file;
for(var i in input.files)
{
$this.Import(input.files[i]);
}
input.value = '';
return;
}
if(file && file instanceof File)
{
var filetype = file.name.replace(/^(.*)\.(.*?)$/, '$2').toLowerCase();
switch(filetype)
{
case 'png': case 'jpg': case 'jpeg': case 'gif': case 'bmp':
$this.Layers.Add(function(layer)
{
layer.Name = file.name.replace(/^(.*)\.(.*?)$/, '$1');
var reader = new FileReader();
reader.onload = function(r)
{
var img = new Image();
img.onload = function()
{
layer.Context.drawImage(this, 0, 0);
};
img.src = r.target.result;
};
reader.readAsDataURL(file);
});
break;
case 'psd':
$this.Layers.Add(function(layer)
{
layer.Name = file.name.replace(/^(.*)\.(.*?)$/, '$1');
var PSD = require('psd');
PSD.fromDroppedFile(file).then(function(psd)
{
var img = new Image();
img.onload = function()
{
layer.Context.drawImage(this, 0, 0);
};
img.src = psd.image.toBase64();
});
});
break;
case 'h5i':
var reader = new FileReader();
reader.onload = function(r)
{
try
{
var data = $.parseJSON(r.target.result);
for(var i in data.layers)
{
var layerdata = data.layers[i],
newlayer = $this.Layers.Add(),
img = new Image();
newlayer.Name = layerdata.name;
img.onload = function()
{
newlayer.Context.drawImage(this, 0, 0);
};
img.src = layerdata.image;
}
}
catch(e){}
};
reader.readAsText(file);
break;
}
}
return $this;
};
$this.Export = function()
{
var cvs = document.createElement('canvas'),
ctx = cvs.getContext('2d');
for(var i in $this.Layers.List)
{
if($this.Layers.List[i].Context.canvas.width > cvs.width)
{
cvs.width = $this.Layers.List[i].Context.canvas.width;
}
if($this.Layers.List[i].Context.canvas.height > cvs.height)
{
cvs.height = $this.Layers.List[i].Context.canvas.height;
}
}
ctx.fillStyle = $this.Background;
ctx.fillRect(0, 0, cvs.width, cvs.height);
for(var i in $this.Layers.List)
{
if(!$this.Layers.List[i].Hidden)
{
ctx.drawImage($this.Layers.List[i].Context.canvas, 0, 0);
}
}
cvs.toBlob(function(blob)
{
saveAs(blob, 'image.png');
});
return $this;
};
$this.Save = function()
{
if(!window.JSON) return;
var data = { layers: [] };
for(var i in $this.Layers.List)
{
var layer = $this.Layers.List[i];
data.layers.push(
{
name: layer.Name,
image: layer.Context.canvas.toDataURL('image/png')
});
}
data = JSON.stringify(data);
saveAs(new Blob([data], {type: "text/plain;charset=utf-8"}), 'image.h5i');
return $this;
};
$this.Resize = function(width, height)
{
for(var i in $this.Layers.List)
{
$this.Layers.List[i].Resize(width, height);
}
return $this;
};
$this.Resize();
if(window.PointerEvent)
{
$('.artboards')
.on('pointerup', 'canvas.layer.active', function(e)
{
var layer = $(this).data('Layer');
if(!layer.Active || layer.Hidden || e.originalEvent.pointerType == 'touch') return false;
return layer.DrawStop.call(this, e.originalEvent), false;
})
.on('pointerdown', 'canvas.layer.active', function(e)
{
var layer = $(this).data('Layer');
if(!layer.Active || layer.Hidden || e.originalEvent.pointerType == 'touch') return false;
return layer.DrawStart.call(this, e.originalEvent), false;
})
.on('pointermove', 'canvas.layer.active', function(e)
{
var layer = $(this).data('Layer');
if(!layer.Active || layer.Hidden || e.originalEvent.pointerType == 'touch') return false;
return layer.Draw.call(this, e.originalEvent), false;
});
}
else
{
$('.artboards')
.on('mousedown', 'canvas.layer.active', function(e)
{
var layer = $(this).data('Layer');
if(!layer.Active || layer.Hidden || e.originalEvent.pointerType == 'touch') return false;
return layer.DrawStart.call(this, e.originalEvent), false;
})
.on('mousemove', 'canvas.layer.active', function(e)
{
var layer = $(this).data('Layer');
if(!layer.Active || layer.Hidden || e.originalEvent.pointerType == 'touch') return false;
return layer.Draw.call(this, e.originalEvent), false;
});
}
/*$('.artboards')
.on('touchstart', 'canvas.layer.active', function(e)
{
return false;
})
.on('touchmove', 'canvas.layer.active', function(e)
{
return false;
})
.on('touchend touchleave', 'canvas.layer.active', function(e)
{
return false;
})
.on('touchcancel', 'canvas.layer.active', function(e)
{
return false;
});*/
$(document)
.on('mouseenter mouseleave mousemove', function(e)
{
$('html')[e.altKey ? 'addClass' : 'removeClass']('alt');
})
.on('mouseup mouseleave blur', function(e)
{
for(var i in $this.Layers.List)
{
$this.Layers.List[i].DrawStop.call($this.Layers.List[i].Canvas[0], e.originalEvent);
}
})
.on('dragenter dragover', function(e)
{
return e.preventDefault(), false;
})
.on('drop', function(e)
{
for(var i in e.originalEvent.dataTransfer.files)
{
$this.Import(e.originalEvent.dataTransfer.files[i]);
}
return false;
});
$(window)
.on('keydown keyup', function(e)
{
$('html')[e.altKey ? 'addClass' : 'removeClass']('alt');
})
.on('resize', function()
{
$('.controls-section.top').css('height', $(window).height() - $('.sidebar .controls-section.bottom').height());
})
.on('beforeunload', function()
{
return 'You will lose any unsaved work!';
})
.trigger('resize');
$(function()
{
$this.Layers.Add().Focus();
});
return $this;
});
<script src="//cdnjs.cloudflare.com/ajax/libs/jquery/2.1.3/jquery.min.js"></script>
<script src="//cdnjs.cloudflare.com/ajax/libs/jqueryui/1.11.2/jquery-ui.min.js"></script>
<script src="http://c.boylett.uk/libraries/psd.js"></script>
<script src="http://c.boylett.uk/libraries/Blob.js"></script>
<script src="http://c.boylett.uk/libraries/FileSaver.min.js"></script>
<script src="http://c.boylett.uk/libraries/canvas-toBlob.js"></script>
html, body
{
width: 100%;
height: 100%;
overflow: hidden;
font-family: "Helvetica Neue", Helvetica, Arial, sans-serif;
font-size: 10px;
zoom: reset;
}
.controls
{
> a
{
position: relative;
z-index: 1;
display: block;
float: left;
margin: 0 -1px 0 0;
padding: .75em 1em;
overflow: hidden;
background: rgba(#FFF, .5);
border: 1px solid #D9D9D9;
text-decoration: none;
font-weight: 300;
font-size: 1.5em;
color: rgba(#000, .66);
> input[type="file"]
{
display: block;
position: absolute;
top: 0;
left: 0;
width: 110%;
height: 110%;
opacity: 0;
}
&.unavailable
{
opacity: .4;
pointer-events: none;
}
&:first-child
{
border-radius: .33em 0 0 .33em;
}
&:last-child
{
margin-right: 0;
border-radius: 0 .33em .33em 0;
&:first-child
{
border-radius: .33em;
}
}
&:hover
{
z-index: 2;
background: #FFF;
border-color: #BBB;
color: #000;
}
}
}
body > img
{
display: block;
position: absolute;
right: 100%;
bottom: 100%;
pointer-events: none;
}
body > .controls
{
display: block;
position: absolute;
z-index: 9999;
top: 1em;
left: 1em;
html.drawing &
{
pointer-events: none;
}
&.right
{
right: 21em;
left: auto;
}
}
body > .sidebar
{
display: block;
position: absolute;
top: 0;
right: 0;
bottom: 0;
width: 20em;
padding: 3.8em 0 0;
background: #EEE;
border-left: 1px solid #D9D9D9;
-webkit-user-select: none;
-moz-user-select: none;
-ms-user-select: none;
user-select: none;
html.drawing &
{
pointer-events: none;
}
> .controls-section
{
position: absolute;
top: 0;
right: 0;
left: 0;
overflow: auto;
overflow-x: hidden;
.controls
{
position: relative;
> h1
{
display: block;
padding: 0 1.55em;
border-bottom: 1px solid #D9D9D9;
text-transform: uppercase;
font-weight: 800;
font-size: 1.3em;
line-height: 2.92em;
color: rgba(#000, .5);
&:last-child
{
border-bottom: none;
}
}
> .current-color
{
display: block;
position: absolute;
top: .5em;
top: calc(.5em + 1px);
right: .5em;
width: 2.7em;
height: 2.7em;
border-radius: .25em;
}
> a
{
float: right;
background: transparent;
border: none;
border-radius: 0;
}
> canvas
{
display: block;
}
&.layers > ul > .layer
{
position: relative;
display: block;
padding: 0 1em;
border-bottom: 1px solid #D9D9D9;
text-align: left;
line-height: 2.35em;
font-weight: 200;
font-size: 2em;
color: rgba(#000, .5);
cursor: pointer;
&.ui-sortable-helper
{
box-shadow: 0 -1px #D9D9D9;
}
> .hide-layer
{
display: block;
position: absolute;
top: 0;
right: 0;
bottom: 0;
width: 4.7em;
background: #EEE;
border-left: 1px solid #D9D9D9;
font-size: .5em;
text-align: center;
text-decoration: none;
line-height: 4.7em;
color: rgba(#000, .5);
&:hover
{
color: #000;
}
&:before
{
content: "\f06e";
font-family: FontAwesome;
font-size: 1.4em;
}
}
&:hover, &.active
{
background: #FFF;
color: #000;
> .hide-layer
{
background: #FFF;
}
}
&.active
{
font-weight: 400;
}
&.hidden
{
opacity: .75;
color: rgba(#000, .5);
> .hide-layer
{
&:before
{
content: "\f070";
}
}
}
html.alt &
{
> .hide-layer
{
&:before
{
content: "\f014";
}
}
}
}
&.thickness
{
> h1 > .current-thickness
{
float: right;
color: rgba(#000, .33);
}
> input
{
display: block;
margin: 1em 7.5%;
width: 85%;
}
}
}
&.bottom
{
top: auto;
bottom: 0;
.controls
{
border-top: 1px solid #D9D9D9;
border-bottom: none;
}
}
}
}
body > .artboards
{
position: absolute;
top: 0;
right: 20em;
right: calc(20em + 1px);
bottom: 0;
left: 0;
background: #FFF;
> canvas
{
display: block;
position: absolute;
top: 0;
left: 0;
touch-action: none;
pointer-events: none;
cursor: crosshair;
html.pen-mode &
{
cursor: none;
}
&.hidden
{
display: none;
}
&.active
{
pointer-events: auto;
}
}
}
<link href="https://maxcdn.bootstrapcdn.com/font-awesome/4.5.0/css/font-awesome.min.css" rel="stylesheet" />
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment