├── .jshintignore ├── src ├── util │ ├── util.extend.js │ └── point.extend.js └── brushes │ ├── stroke.js │ ├── drip.js │ ├── marker_brush.js │ ├── crayon_brush.js │ ├── ink_brush.js │ └── spray_brush.js ├── .gitignore ├── bower.json ├── LICENSE ├── package.json ├── README.md └── dist └── fabric-brush.min.js /.jshintignore: -------------------------------------------------------------------------------- 1 | bower_components/** 2 | node_modules/** 3 | 4 | demo/** 5 | dist/** -------------------------------------------------------------------------------- /src/util/util.extend.js: -------------------------------------------------------------------------------- 1 | (function(fabric){ 2 | 3 | fabric.util.getRandom = function(max, min){ 4 | min = min ? min : 0; 5 | return Math.random() * ((max ? max : 1) - min) + min; 6 | }; 7 | 8 | fabric.util.clamp = function (n, max, min) { 9 | if (typeof min !== 'number') min = 0; 10 | return n > max ? max : n < min ? min : n; 11 | }; 12 | 13 | })(fabric); -------------------------------------------------------------------------------- /src/util/point.extend.js: -------------------------------------------------------------------------------- 1 | (function(fabric) { 2 | 3 | fabric.Point.prototype.angleBetween = function(that){ 4 | return Math.atan2( this.x - that.x, this.y - that.y); 5 | }; 6 | 7 | fabric.Point.prototype.normalize = function(thickness) { 8 | if (null === thickness || undefined === thickness) { 9 | thickness = 1; 10 | } 11 | 12 | var length = this.distanceFrom({ x: 0, y: 0 }); 13 | 14 | if (length > 0) { 15 | this.x = this.x / length * thickness; 16 | this.y = this.y / length * thickness; 17 | } 18 | 19 | return this; 20 | }; 21 | 22 | })(fabric); -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | 2 | # Created by https://www.gitignore.io/api/osx 3 | 4 | ### OSX ### 5 | .DS_Store 6 | .AppleDouble 7 | .LSOverride 8 | 9 | # Icon must end with two \r 10 | Icon 11 | 12 | 13 | # Thumbnails 14 | ._* 15 | 16 | # Files that might appear in the root of a volume 17 | .DocumentRevisions-V100 18 | .fseventsd 19 | .Spotlight-V100 20 | .TemporaryItems 21 | .Trashes 22 | .VolumeIcon.icns 23 | 24 | # Directories potentially created on remote AFP share 25 | .AppleDB 26 | .AppleDesktop 27 | Network Trash Folder 28 | Temporary Items 29 | .apdisk 30 | 31 | ### NodeJS ### 32 | node_modules/ 33 | 34 | ### Bower ### 35 | bower_components/ 36 | -------------------------------------------------------------------------------- /bower.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "fabric-brush", 3 | "authors": [ 4 | "Tennison Chan " 5 | ], 6 | "description": "A variety of brushes based on an awesome HTML5 canvas framework fabric.js", 7 | "license": "MIT", 8 | "homepage": "https://tennisonchan.github.com/fabric-brush", 9 | "ignore": [ 10 | "**/.*", 11 | "node_modules", 12 | "bower_components", 13 | "test", 14 | "tests" 15 | ], 16 | "keywords": [ 17 | "fabric", 18 | "fabric.js", 19 | "brush", 20 | "brushes", 21 | "canvas", 22 | "HTML5", 23 | "drawing", 24 | "spray", 25 | "stroke", 26 | "crayon", 27 | "marker", 28 | "ink", 29 | "drip", 30 | "graffiti" 31 | ], 32 | "dependencies": { 33 | "fabric.js": "fabric#^1.5.0" 34 | } 35 | } 36 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | Copyright (c) 2016 Tennison Chan 2 | 3 | Permission is hereby granted, free of charge, to any person obtaining a copy 4 | of this software and associated documentation files (the "Software"), to deal 5 | in the Software without restriction, including without limitation the rights 6 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 7 | copies of the Software, and to permit persons to whom the Software is 8 | furnished to do so, subject to the following conditions: 9 | 10 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 11 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 12 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 13 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 14 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 15 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 16 | SOFTWARE. -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "fabric-brush", 3 | "version": "0.0.1", 4 | "description": "A variety of brushes based on an awesome HTML5 canvas framework fabric.js", 5 | "scripts": { 6 | "clean": "rimraf dist/*", 7 | "lint": "jshint */**.js", 8 | "prebuild": "npm run clean -s && mkdir -p dist", 9 | "prebuild:scripts": "npm run lint", 10 | "build:scripts": "browserify ./src/**/*.js --outfile dist/fabric-brush.js", 11 | "build:scripts:min": "browserify ./src/**/*.js -d -p [minifyify --no-map] --outfile dist/fabric-brush.min.js", 12 | "build": "npm run build:scripts && npm run build:scripts:min" 13 | }, 14 | "keywords": [ 15 | "fabric", 16 | "fabric.js", 17 | "brush", 18 | "brushes", 19 | "canvas", 20 | "HTML5", 21 | "drawing", 22 | "spray", 23 | "stroke", 24 | "crayon", 25 | "marker", 26 | "ink", 27 | "drip", 28 | "graffiti" 29 | ], 30 | "engines": { 31 | "node": ">= 0.4.4" 32 | }, 33 | "homepage": "https://tennisonchan.github.com/fabric-brush", 34 | "author": "Tennison Chan (https://github.com/tennisonchan)", 35 | "repository": { 36 | "type": "git", 37 | "url": "git://github.com/tennisonchan/fabric-brush.git" 38 | }, 39 | "bugs": { 40 | "url": "https://github.com/tennisonchan/fabric-brush/issues" 41 | }, 42 | "license": "MIT", 43 | "devDependencies": { 44 | "browserify": "~13.0.0", 45 | "jshint": "~2.9.1", 46 | "minifyify": "^7.3.1", 47 | "rimraf": "~2.5.2" 48 | } 49 | } 50 | -------------------------------------------------------------------------------- /src/brushes/stroke.js: -------------------------------------------------------------------------------- 1 | /** 2 | * Stroke class 3 | * @class fabric.Stroke 4 | * @extends fabric.Object 5 | */ 6 | (function(fabric) { 7 | 8 | fabric.Stroke = fabric.util.createClass(fabric.Object,{ 9 | color: null, 10 | inkAmount: null, 11 | lineWidth: null, 12 | _point: null, 13 | _lastPoint: null, 14 | _currentLineWidth: null, 15 | 16 | initialize: function(ctx, pointer, range, color, lineWidth, inkAmount){ 17 | 18 | var rx = fabric.util.getRandom(range), 19 | c = fabric.util.getRandom(Math.PI * 2), 20 | c0 = fabric.util.getRandom(Math.PI * 2), 21 | x0 = rx * Math.sin(c0), 22 | y0 = rx / 2 * Math.cos(c0), 23 | cos = Math.cos(c), 24 | sin = Math.sin(c); 25 | 26 | this.ctx = ctx; 27 | this.color = color; 28 | this._point = new fabric.Point(pointer.x + x0 * cos - y0 * sin, pointer.y + x0 * sin + y0 * cos); 29 | this.lineWidth = lineWidth; 30 | this.inkAmount = inkAmount; 31 | this._currentLineWidth = lineWidth; 32 | 33 | ctx.lineCap = "round"; 34 | }, 35 | 36 | update: function(pointer, subtractPoint, distance) { 37 | this._lastPoint = fabric.util.object.clone(this._point); 38 | this._point = this._point.addEquals({ x: subtractPoint.x, y: subtractPoint.y }); 39 | 40 | var n = this.inkAmount / (distance + 1); 41 | var per = (n > 0.3 ? 0.2 : n < 0 ? 0 : n); 42 | this._currentLineWidth = this.lineWidth * per; 43 | }, 44 | 45 | draw: function(){ 46 | var ctx = this.ctx; 47 | ctx.save(); 48 | this.line(ctx, this._lastPoint, this._point, this.color, this._currentLineWidth); 49 | ctx.restore(); 50 | }, 51 | 52 | line: function(ctx, point1, point2, color, lineWidth) { 53 | ctx.strokeStyle = color; 54 | ctx.lineWidth = lineWidth; 55 | ctx.beginPath(); 56 | ctx.moveTo(point1.x, point1.y); 57 | ctx.lineTo(point2.x, point2.y); 58 | 59 | ctx.stroke(); 60 | } 61 | }); 62 | 63 | })(fabric); 64 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Fabric Brush - Canvas Brushes on Fabric.js 2 | Fabric Brush is a collection of brushes built on an awesome canvas framework Fabric.js 3 | 4 | ## Quickstart 5 | - [Install fabric with bower](https://github.com/kangax/fabric.js): `bower install fabric` 6 | - Install fabric-brush with bower: `bower install fabric-brush` 7 | 8 | ## [Demo](https://tennisonchan.github.io/fabric-brush) 9 | 10 | [crayon-brush](https://tennisonchan.github.io/fabric-brush) 11 | [ink-brush](https://tennisonchan.github.io/fabric-brush) 12 | [marker-brush](https://tennisonchan.github.io/fabric-brush) 13 | [spray-brush](https://tennisonchan.github.io/fabric-brush) 14 | 15 | ### Examples of use 16 | Set the crayon brush as the free drawing brush 17 | 18 | ```html 19 | 20 | 21 | 22 | 23 | ``` 24 | ```javascript 25 | var canvas = new fabric.Canvas('c'); 26 | 27 | // Crayon Brush 28 | canvas.freeDrawingBrush = new fabric.CrayonBrush(canvas, { 29 | width: 70, 30 | opacity: 0.6, 31 | color: "#ff0000" 32 | }); 33 | 34 | // Ink Brush 35 | // canvas.freeDrawingBrush = new fabric.InkBrush(canvas); 36 | 37 | // Marker Brush 38 | // canvas.freeDrawingBrush = new fabric.MarkerBrush(canvas); 39 | 40 | // Spray Brush 41 | // canvas.freeDrawingBrush = new fabric.SprayBrush(canvas); 42 | ``` 43 | 44 | Change color of the brush into red 45 | ``` 46 | canvas.freeDrawingBrush.changeColor("#ff0000"); 47 | ``` 48 | Change opacity of the brush into 0.6 49 | ``` 50 | canvas.freeDrawingBrush.changeOpacity(0.6); 51 | ``` 52 | -------------------------------------------------------------------------------- /src/brushes/drip.js: -------------------------------------------------------------------------------- 1 | /** 2 | * Drip class 3 | * @class fabric.Drip 4 | * @extends fabric.Object 5 | */ 6 | (function(fabric) { 7 | 8 | fabric.Drip = fabric.util.createClass(fabric.Object, { 9 | rate: 0, 10 | color: "#000000", 11 | amount: 10, 12 | life: 10, 13 | _point: null, 14 | _lastPoint: null, 15 | _strokeId: 0, 16 | _interval: 20, 17 | 18 | initialize: function(ctx, pointer, amount, color, _strokeId) { 19 | this.ctx = ctx; 20 | this._point = pointer; 21 | this._strokeId = _strokeId; 22 | this.amount = fabric.util.getRandom(amount, amount * 0.5); 23 | this.color = color; 24 | this.life = this.amount * 1.5; 25 | ctx.lineCap = ctx.lineJoin = "round"; 26 | 27 | this._render(); 28 | }, 29 | 30 | _update: function(brush) { 31 | this._lastPoint = fabric.util.object.clone(this._point); 32 | this._point.addEquals({ 33 | x: this.life * this.rate, 34 | y: fabric.util.getRandom(this.life * this.amount / 30) 35 | }); 36 | 37 | this.life -= 0.05; 38 | 39 | if (fabric.util.getRandom() < 0.03) { 40 | this.rate += fabric.util.getRandom(0.03, - 0.03); 41 | } else if (fabric.util.getRandom() < 0.05) { 42 | this.rate *= 0.01; 43 | } 44 | }, 45 | 46 | _draw: function() { 47 | this.ctx.save(); 48 | this.line(this.ctx, this._lastPoint, this._point, this.color, this.amount * 0.8 + this.life * 0.2); 49 | this.ctx.restore(); 50 | }, 51 | 52 | _render: function() { 53 | var context = this; 54 | 55 | setTimeout(draw, this._interval); 56 | 57 | function draw() { 58 | context._update(); 59 | context._draw(); 60 | if(context.life > 0) { 61 | setTimeout(draw, context._interval); 62 | } 63 | } 64 | }, 65 | 66 | line: function(ctx, point1, point2, color, lineWidth) { 67 | ctx.strokeStyle = color; 68 | ctx.lineWidth = lineWidth; 69 | ctx.beginPath(); 70 | ctx.moveTo(point1.x, point1.y); 71 | ctx.lineTo(point2.x, point2.y); 72 | 73 | ctx.stroke(); 74 | } 75 | }); 76 | 77 | })(fabric); 78 | -------------------------------------------------------------------------------- /src/brushes/marker_brush.js: -------------------------------------------------------------------------------- 1 | /** 2 | * MarkerBrush class 3 | * @class fabric.MarkerBrush 4 | * @extends fabric.BaseBrush 5 | */ 6 | (function(fabric) { 7 | 8 | fabric.MarkerBrush = fabric.util.createClass(fabric.BaseBrush, { 9 | 10 | color: "#000000", 11 | opacity: 1, 12 | width: 30, 13 | 14 | _baseWidth: 10, 15 | _lastPoint: null, 16 | _lineWidth: 3, 17 | _point: null, 18 | _size: 0, 19 | 20 | initialize: function(canvas, opt) { 21 | opt = opt || {}; 22 | 23 | this.canvas = canvas; 24 | this.width = opt.width || canvas.freeDrawingBrush.width; 25 | this.color = opt.color || canvas.freeDrawingBrush.color; 26 | this.opacity = opt.opacity || canvas.contextTop.globalAlpha; 27 | this._point = new fabric.Point(); 28 | 29 | this.canvas.contextTop.lineJoin = 'round'; 30 | this.canvas.contextTop.lineCap = 'round'; 31 | }, 32 | 33 | changeColor: function(color) { 34 | this.color = color; 35 | }, 36 | 37 | changeOpacity: function(value) { 38 | this.opacity = value; 39 | }, 40 | 41 | _render: function(pointer) { 42 | var ctx, lineWidthDiff, i; 43 | 44 | ctx = this.canvas.contextTop; 45 | 46 | ctx.beginPath(); 47 | 48 | for(i = 0, len = (this._size / this._lineWidth) / 2; i < len; i++) { 49 | lineWidthDiff = (this._lineWidth - 1) * i; 50 | 51 | ctx.globalAlpha = 0.8 * this.opacity; 52 | ctx.moveTo(this._lastPoint.x + lineWidthDiff, this._lastPoint.y + lineWidthDiff); 53 | ctx.lineTo(pointer.x + lineWidthDiff, pointer.y + lineWidthDiff); 54 | ctx.stroke(); 55 | } 56 | 57 | this._lastPoint = new fabric.Point(pointer.x, pointer.y); 58 | }, 59 | 60 | onMouseDown: function(pointer) { 61 | this._lastPoint = pointer; 62 | this.canvas.contextTop.strokeStyle = this.color; 63 | this.canvas.contextTop.lineWidth = this._lineWidth; 64 | this._size = this.width + this._baseWidth; 65 | }, 66 | 67 | onMouseMove: function(pointer) { 68 | if (this.canvas._isCurrentlyDrawing) { 69 | this._render(pointer); 70 | } 71 | }, 72 | 73 | onMouseUp: function() { 74 | this.canvas.contextTop.globalAlpha = this.opacity; 75 | } 76 | }); 77 | 78 | })(fabric); 79 | -------------------------------------------------------------------------------- /src/brushes/crayon_brush.js: -------------------------------------------------------------------------------- 1 | /** 2 | * CrayonBrush class 3 | * @class fabric.CrayonBrush 4 | * @extends fabric.BaseBrush 5 | */ 6 | (function(fabric){ 7 | 8 | fabric.CrayonBrush = fabric.util.createClass(fabric.BaseBrush, { 9 | 10 | color: "#000000", 11 | opacity: 0.6, 12 | width: 30, 13 | 14 | _baseWidth: 20, 15 | _inkAmount: 10, 16 | _latestStrokeLength: 0, 17 | _point: null, 18 | _sep: 5, 19 | _size: 0, 20 | 21 | initialize: function(canvas, opt) { 22 | opt = opt || {}; 23 | 24 | this.canvas = canvas; 25 | this.width = opt.width || canvas.freeDrawingBrush.width; 26 | this.color = opt.color || canvas.freeDrawingBrush.color; 27 | this.opacity = opt.opacity || canvas.contextTop.globalAlpha; 28 | this._point = new fabric.Point(0, 0); 29 | }, 30 | 31 | changeColor: function(color){ 32 | this.color = color; 33 | }, 34 | 35 | changeOpacity: function(value){ 36 | this.opacity = value; 37 | }, 38 | 39 | onMouseDown: function(pointer){ 40 | this.canvas.contextTop.globalAlpha = this.opacity; 41 | this._size = this.width / 2 + this._baseWidth; 42 | this.set(pointer); 43 | }, 44 | 45 | onMouseMove: function(pointer){ 46 | this.update(pointer); 47 | this.draw(this.canvas.contextTop); 48 | }, 49 | 50 | onMouseUp: function(pointer){ 51 | }, 52 | 53 | set: function(p) { 54 | if (this._latest) { 55 | this._latest.setFromPoint(this._point); 56 | } else { 57 | this._latest = new fabric.Point(p.x, p.y); 58 | } 59 | fabric.Point.prototype.setFromPoint.call(this._point, p); 60 | }, 61 | 62 | update: function(p) { 63 | this.set(p); 64 | this._latestStrokeLength = this._point.subtract(this._latest).distanceFrom({ x: 0, y: 0 }); 65 | }, 66 | 67 | draw: function(ctx) { 68 | var i, j, p, r, c, x, y, w, h, v, s, stepNum, dotSize, dotNum, range; 69 | 70 | v = this._point.subtract(this._latest); 71 | s = Math.ceil(this._size / 2); 72 | stepNum = Math.floor(v.distanceFrom({ x: 0, y: 0 }) / s) + 1; 73 | v.normalize(s); 74 | 75 | dotSize = this._sep * fabric.util.clamp(this._inkAmount / this._latestStrokeLength * 3, 1, 0.5); 76 | dotNum = Math.ceil(this._size * this._sep); 77 | 78 | range = this._size / 2; 79 | 80 | ctx.save(); 81 | ctx.fillStyle = this.color; 82 | ctx.beginPath(); 83 | for (i = 0; i < dotNum; i++) { 84 | for (j = 0; j < stepNum; j++) { 85 | p = this._latest.add(v.multiply(j)); 86 | r = fabric.util.getRandom(range); 87 | c = fabric.util.getRandom(Math.PI * 2); 88 | w = fabric.util.getRandom(dotSize, dotSize / 2); 89 | h = fabric.util.getRandom(dotSize, dotSize / 2); 90 | x = p.x + r * Math.sin(c) - w / 2; 91 | y = p.y + r * Math.cos(c) - h / 2; 92 | ctx.rect(x, y, w, h); 93 | } 94 | } 95 | ctx.fill(); 96 | ctx.restore(); 97 | } 98 | }); 99 | 100 | })(fabric); 101 | -------------------------------------------------------------------------------- /src/brushes/ink_brush.js: -------------------------------------------------------------------------------- 1 | /** 2 | * InkBrush class 3 | * @class fabric.InkBrush 4 | * @extends fabric.BaseBrush 5 | */ 6 | (function(fabric){ 7 | 8 | fabric.InkBrush = fabric.util.createClass(fabric.BaseBrush, { 9 | 10 | color: "#000000", 11 | opacity: 1, 12 | width: 30, 13 | 14 | _baseWidth: 20, 15 | _dripCount: 0, 16 | _drips: [], 17 | _inkAmount: 7, 18 | _lastPoint: null, 19 | _point: null, 20 | _range: 10, 21 | _strokeCount: 0, 22 | _strokeId: null, 23 | _strokeNum: 40, 24 | _strokes: null, 25 | 26 | initialize: function(canvas, opt) { 27 | opt = opt || {}; 28 | 29 | this.canvas = canvas; 30 | this.width = opt.width || canvas.freeDrawingBrush.width; 31 | this.color = opt.color || canvas.freeDrawingBrush.color; 32 | this.opacity = opt.opacity || canvas.contextTop.globalAlpha; 33 | 34 | this._point = new fabric.Point(); 35 | }, 36 | 37 | changeColor: function(color){ 38 | this.color = color; 39 | }, 40 | 41 | changeOpacity: function(value){ 42 | this.opacity = value; 43 | this.canvas.contextTop.globalAlpha = value; 44 | }, 45 | 46 | _render: function(pointer){ 47 | var subtractPoint, distance, point, i, len, strokes, stroke; 48 | this._strokeCount++; 49 | if (this._strokeCount % 120 === 0 && this._dripCount < 10) { 50 | this._dripCount++; 51 | } 52 | 53 | point = this.setPointer(pointer); 54 | subtractPoint = point.subtract(this._lastPoint); 55 | distance = point.distanceFrom(this._lastPoint); 56 | strokes = this._strokes; 57 | 58 | for (i = 0, len = strokes.length; i < len; i++) { 59 | stroke = strokes[i]; 60 | stroke.update(point, subtractPoint, distance); 61 | stroke.draw(); 62 | } 63 | 64 | if (distance > 30) { 65 | this.drawSplash(point, this._inkAmount); 66 | } else if (distance < 10 && fabric.util.getRandom() < 0.085 && this._dripCount) { 67 | this._drips.push(new fabric.Drip(this.canvas.contextTop, point, fabric.util.getRandom(this.size * 0.25, this.size * 0.1), this.color, this._strokeId)); 68 | this._dripCount--; 69 | } 70 | }, 71 | 72 | onMouseDown: function(pointer){ 73 | this._resetTip(pointer); 74 | this._strokeId = +new Date(); 75 | this._dripCount = fabric.util.getRandom(6, 3) | 0; 76 | }, 77 | 78 | onMouseMove: function(pointer){ 79 | if(this.canvas._isCurrentlyDrawing){ 80 | this._render(pointer); 81 | } 82 | }, 83 | 84 | onMouseUp: function(){ 85 | this._strokeCount = 0; 86 | this._dripCount = 0; 87 | this._strokeId = null; 88 | }, 89 | 90 | drawSplash: function(pointer, maxSize) { 91 | var c, r, i, point, 92 | ctx = this.canvas.contextTop, 93 | num = fabric.util.getRandom(12), 94 | range = maxSize * 10, 95 | color = this.color; 96 | 97 | ctx.save(); 98 | for (i = 0; i < num; i++) { 99 | r = fabric.util.getRandom(range, 1); 100 | c = fabric.util.getRandom(Math.PI * 2); 101 | point = new fabric.Point(pointer.x + r * Math.sin(c), pointer.y + r * Math.cos(c)); 102 | 103 | ctx.fillStyle = color; 104 | ctx.beginPath(); 105 | ctx.arc(point.x, point.y, fabric.util.getRandom(maxSize) / 2, 0, Math.PI * 2, false); 106 | ctx.fill(); 107 | } 108 | ctx.restore(); 109 | }, 110 | 111 | setPointer: function(pointer) { 112 | var point = new fabric.Point(pointer.x, pointer.y); 113 | 114 | this._lastPoint = fabric.util.object.clone(this._point); 115 | this._point = point; 116 | 117 | return point; 118 | }, 119 | 120 | _resetTip: function(pointer){ 121 | var strokes, point, len, i; 122 | 123 | point = this.setPointer(pointer); 124 | strokes = this._strokes = []; 125 | this.size = this.width / 5 + this._baseWidth; 126 | this._strokeNum = this.size; 127 | this._range = this.size / 2; 128 | 129 | for (i = 0, len = this._strokeNum; i < len; i++) { 130 | strokes[i] = new fabric.Stroke(this.canvas.contextTop, point, this._range, this.color, this.width, this._inkAmount); 131 | } 132 | } 133 | }); 134 | 135 | })(fabric); 136 | -------------------------------------------------------------------------------- /src/brushes/spray_brush.js: -------------------------------------------------------------------------------- 1 | /** 2 | * SprayBrush class 3 | * @class fabric.SprayBrush 4 | * @extends fabric.BaseBrush 5 | */ 6 | (function(fabric){ 7 | 8 | fabric.SprayBrush = fabric.util.createClass(fabric.BaseBrush, { 9 | 10 | color: "#000000", 11 | opacity: 1, 12 | width: 30, 13 | 14 | _baseWidth: 40, 15 | _drips: [], 16 | _dripThreshold: 15, 17 | _inkAmount: 0, 18 | _interval: 20, 19 | _lastPoint: null, 20 | _point: null, 21 | _strokeId: 0, 22 | brush: null, 23 | sprayBrushDataUrl: "", 24 | 25 | initialize: function(canvas, opt) { 26 | var context = this; 27 | opt = opt || {}; 28 | 29 | this.canvas = canvas; 30 | this.width = opt.width || canvas.freeDrawingBrush.width; 31 | this.opacity = opt.opacity || canvas.contextTop.globalAlpha; 32 | this.color = opt.color || canvas.freeDrawingBrush.color; 33 | 34 | this.canvas.contextTop.lineJoin = "round"; 35 | this.canvas.contextTop.lineCap = "round"; 36 | 37 | this._reset(); 38 | 39 | fabric.Image.fromURL(this.sprayBrushDataUrl, function(brush) { 40 | context.brush = brush; 41 | context.brush.filters = []; 42 | context.changeColor(context.color || this.color); 43 | }, { crossOrigin: "anonymous" }); 44 | }, 45 | 46 | changeColor: function(color) { 47 | this.color = color; 48 | this.brush.filters[0] = new fabric.Image.filters.Tint({ color: color }); 49 | this.brush.applyFilters(this.canvas.renderAll.bind(this.canvas)); 50 | }, 51 | 52 | changeOpacity: function(value) { 53 | this.opacity = value; 54 | this.canvas.contextTop.globalAlpha = value; 55 | }, 56 | 57 | onMouseDown: function(pointer) { 58 | this._point = new fabric.Point(pointer.x, pointer.y); 59 | this._lastPoint = this._point; 60 | 61 | this.size = this.width + this._baseWidth; 62 | this._strokeId = +new Date(); 63 | this._inkAmount = 0; 64 | 65 | this.changeColor(this.color); 66 | this._render(); 67 | }, 68 | 69 | onMouseMove: function(pointer) { 70 | this._lastPoint = this._point; 71 | this._point = new fabric.Point(pointer.x, pointer.y); 72 | }, 73 | 74 | onMouseUp: function(pointer) { 75 | }, 76 | 77 | _render: function() { 78 | var context = this; 79 | 80 | setTimeout(draw, this._interval); 81 | 82 | function draw() { 83 | var point, distance, angle, amount, x, y; 84 | 85 | point = new fabric.Point(context._point.x || 0, context._point.y || 0); 86 | distance = point.distanceFrom(context._lastPoint); 87 | angle = point.angleBetween(context._lastPoint); 88 | amount = (100 / context.size) / (Math.pow(distance, 2) + 1); 89 | 90 | context._inkAmount += amount; 91 | context._inkAmount = Math.max(context._inkAmount - distance / 10, 0); 92 | if (context._inkAmount > context._dripThreshold) { 93 | context._drips.push(new fabric.Drip(context.canvas.contextTop, point, context._inkAmount / 2, context.color, context._strokeId)); 94 | context._inkAmount = 0; 95 | } 96 | 97 | x = context._lastPoint.x + Math.sin(angle) - context.size / 2; 98 | y = context._lastPoint.y + Math.cos(angle) - context.size / 2; 99 | context.canvas.contextTop.drawImage(context.brush._element, x, y, context.size, context.size); 100 | 101 | if (context.canvas._isCurrentlyDrawing) { 102 | setTimeout(draw, context._interval); 103 | } else { 104 | context._reset(); 105 | } 106 | } 107 | }, 108 | 109 | _reset: function() { 110 | this._drips.length = 0; 111 | this._point = null; 112 | this._lastPoint = null; 113 | } 114 | }); 115 | 116 | })(fabric); 117 | -------------------------------------------------------------------------------- /dist/fabric-brush.min.js: -------------------------------------------------------------------------------- 1 | (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;os;s++)for(o=0;_>o;o++)e=this._latest.add(u.multiply(o)),n=t.util.getRandom(g),a=t.util.getRandom(2*Math.PI),c=t.util.getRandom(d,d/2),r=t.util.getRandom(d,d/2),h=e.x+n*Math.sin(a)-c/2,l=e.y+n*Math.cos(a)-r/2,i.rect(h,l,c,r);i.fill(),i.restore()}})}(fabric); 3 | },{}],2:[function(require,module,exports){ 4 | !function(t){t.Drip=t.util.createClass(t.Object,{rate:0,color:"#000000",amount:10,life:10,_point:null,_lastPoint:null,_strokeId:0,_interval:20,initialize:function(i,e,n,o,s){this.ctx=i,this._point=e,this._strokeId=s,this.amount=t.util.getRandom(n,.5*n),this.color=o,this.life=1.5*this.amount,i.lineCap=i.lineJoin="round",this._render()},_update:function(i){this._lastPoint=t.util.object.clone(this._point),this._point.addEquals({x:this.life*this.rate,y:t.util.getRandom(this.life*this.amount/30)}),this.life-=.05,t.util.getRandom()<.03?this.rate+=t.util.getRandom(.03,-.03):t.util.getRandom()<.05&&(this.rate*=.01)},_draw:function(){this.ctx.save(),this.line(this.ctx,this._lastPoint,this._point,this.color,.8*this.amount+.2*this.life),this.ctx.restore()},_render:function(){function t(){i._update(),i._draw(),i.life>0&&setTimeout(t,i._interval)}var i=this;setTimeout(t,this._interval)},line:function(t,i,e,n,o){t.strokeStyle=n,t.lineWidth=o,t.beginPath(),t.moveTo(i.x,i.y),t.lineTo(e.x,e.y),t.stroke()}})}(fabric); 5 | },{}],3:[function(require,module,exports){ 6 | !function(t){t.InkBrush=t.util.createClass(t.BaseBrush,{color:"#000000",opacity:1,width:30,_baseWidth:20,_dripCount:0,_drips:[],_inkAmount:7,_lastPoint:null,_point:null,_range:10,_strokeCount:0,_strokeId:null,_strokeNum:40,_strokes:null,initialize:function(i,s){s=s||{},this.canvas=i,this.width=s.width||i.freeDrawingBrush.width,this.color=s.color||i.freeDrawingBrush.color,this.opacity=s.opacity||i.contextTop.globalAlpha,this._point=new t.Point},changeColor:function(t){this.color=t},changeOpacity:function(t){this.opacity=t,this.canvas.contextTop.globalAlpha=t},_render:function(i){var s,o,n,e,h,r,a;for(this._strokeCount++,this._strokeCount%120===0&&this._dripCount<10&&this._dripCount++,n=this.setPointer(i),s=n.subtract(this._lastPoint),o=n.distanceFrom(this._lastPoint),r=this._strokes,e=0,h=r.length;h>e;e++)a=r[e],a.update(n,s,o),a.draw();o>30?this.drawSplash(n,this._inkAmount):10>o&&t.util.getRandom()<.085&&this._dripCount&&(this._drips.push(new t.Drip(this.canvas.contextTop,n,t.util.getRandom(.25*this.size,.1*this.size),this.color,this._strokeId)),this._dripCount--)},onMouseDown:function(i){this._resetTip(i),this._strokeId=+new Date,this._dripCount=0|t.util.getRandom(6,3)},onMouseMove:function(t){this.canvas._isCurrentlyDrawing&&this._render(t)},onMouseUp:function(){this._strokeCount=0,this._dripCount=0,this._strokeId=null},drawSplash:function(i,s){var o,n,e,h,r=this.canvas.contextTop,a=t.util.getRandom(12),u=10*s,l=this.color;for(r.save(),e=0;a>e;e++)n=t.util.getRandom(u,1),o=t.util.getRandom(2*Math.PI),h=new t.Point(i.x+n*Math.sin(o),i.y+n*Math.cos(o)),r.fillStyle=l,r.beginPath(),r.arc(h.x,h.y,t.util.getRandom(s)/2,0,2*Math.PI,!1),r.fill();r.restore()},setPointer:function(i){var s=new t.Point(i.x,i.y);return this._lastPoint=t.util.object.clone(this._point),this._point=s,s},_resetTip:function(i){var s,o,n,e;for(o=this.setPointer(i),s=this._strokes=[],this.size=this.width/5+this._baseWidth,this._strokeNum=this.size,this._range=this.size/2,e=0,n=this._strokeNum;n>e;e++)s[e]=new t.Stroke(this.canvas.contextTop,o,this._range,this.color,this.width,this._inkAmount)}})}(fabric); 7 | },{}],4:[function(require,module,exports){ 8 | !function(t){t.MarkerBrush=t.util.createClass(t.BaseBrush,{color:"#000000",opacity:1,width:30,_baseWidth:10,_lastPoint:null,_lineWidth:3,_point:null,_size:0,initialize:function(i,n){n=n||{},this.canvas=i,this.width=n.width||i.freeDrawingBrush.width,this.color=n.color||i.freeDrawingBrush.color,this.opacity=n.opacity||i.contextTop.globalAlpha,this._point=new t.Point,this.canvas.contextTop.lineJoin="round",this.canvas.contextTop.lineCap="round"},changeColor:function(t){this.color=t},changeOpacity:function(t){this.opacity=t},_render:function(i){var n,o,s;for(n=this.canvas.contextTop,n.beginPath(),s=0,len=this._size/this._lineWidth/2;sB._dripThreshold&&(B._drips.push(new s.Drip(B.canvas.contextTop,t,B._inkAmount/2,B.color,B._strokeId)),B._inkAmount=0),g=B._lastPoint.x+Math.sin(W)-B.size/2,m=B._lastPoint.y+Math.cos(W)-B.size/2,B.canvas.contextTop.drawImage(B.brush._element,g,m,B.size,B.size),B.canvas._isCurrentlyDrawing?setTimeout(A,B._interval):B._reset()}var B=this;setTimeout(A,this._interval)},_reset:function(){this._drips.length=0,this._point=null,this._lastPoint=null}})}(fabric); 12 | },{}],6:[function(require,module,exports){ 13 | !function(t){t.Stroke=t.util.createClass(t.Object,{color:null,inkAmount:null,lineWidth:null,_point:null,_lastPoint:null,_currentLineWidth:null,initialize:function(i,n,o,e,l,h){var s=t.util.getRandom(o),a=t.util.getRandom(2*Math.PI),u=t.util.getRandom(2*Math.PI),r=s*Math.sin(u),c=s/2*Math.cos(u),d=Math.cos(a),_=Math.sin(a);this.ctx=i,this.color=e,this._point=new t.Point(n.x+r*d-c*_,n.y+r*_+c*d),this.lineWidth=l,this.inkAmount=h,this._currentLineWidth=l,i.lineCap="round"},update:function(i,n,o){this._lastPoint=t.util.object.clone(this._point),this._point=this._point.addEquals({x:n.x,y:n.y});var e=this.inkAmount/(o+1),l=e>.3?.2:0>e?0:e;this._currentLineWidth=this.lineWidth*l},draw:function(){var t=this.ctx;t.save(),this.line(t,this._lastPoint,this._point,this.color,this._currentLineWidth),t.restore()},line:function(t,i,n,o,e){t.strokeStyle=o,t.lineWidth=e,t.beginPath(),t.moveTo(i.x,i.y),t.lineTo(n.x,n.y),t.stroke()}})}(fabric); 14 | },{}],7:[function(require,module,exports){ 15 | !function(t){t.Point.prototype.angleBetween=function(t){return Math.atan2(this.x-t.x,this.y-t.y)},t.Point.prototype.normalize=function(t){(null===t||void 0===t)&&(t=1);var i=this.distanceFrom({x:0,y:0});return i>0&&(this.x=this.x/i*t,this.y=this.y/i*t),this}}(fabric); 16 | },{}],8:[function(require,module,exports){ 17 | !function(n){n.util.getRandom=function(n,t){return t=t?t:0,Math.random()*((n?n:1)-t)+t},n.util.clamp=function(n,t,u){return"number"!=typeof u&&(u=0),n>t?t:u>n?u:n}}(fabric); 18 | },{}]},{},[1,2,3,4,5,6,7,8]) 19 | 20 | --------------------------------------------------------------------------------