mirror of
https://github.com/openvk/openvk
synced 2025-01-25 17:19:24 +03:00
6434 lines
188 KiB
JavaScript
6434 lines
188 KiB
JavaScript
|
(function(f){if(typeof exports==="object"&&typeof module!=="undefined"){module.exports=f()}else if(typeof define==="function"&&define.amd){define([],f)}else{var g;if(typeof window!=="undefined"){g=window}else if(typeof global!=="undefined"){g=global}else if(typeof self!=="undefined"){g=self}else{g=this}g.LC = f()}})(function(){var define,module,exports;return (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;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
|
||
|
/**
|
||
|
* Copyright 2013-2015, Facebook, Inc.
|
||
|
* All rights reserved.
|
||
|
*
|
||
|
* This source code is licensed under the BSD-style license found in the
|
||
|
* LICENSE file in the root directory of this source tree. An additional grant
|
||
|
* of patent rights can be found in the PATENTS file in the same directory.
|
||
|
*
|
||
|
* @providesModule shallowEqual
|
||
|
* @typechecks
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
'use strict';
|
||
|
|
||
|
var hasOwnProperty = Object.prototype.hasOwnProperty;
|
||
|
|
||
|
/**
|
||
|
* Performs equality by iterating through keys on an object and returning false
|
||
|
* when any key has values which are not strictly equal between the arguments.
|
||
|
* Returns true when the values of all keys are strictly equal.
|
||
|
*/
|
||
|
function shallowEqual(objA, objB) {
|
||
|
if (objA === objB) {
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
if (typeof objA !== 'object' || objA === null || typeof objB !== 'object' || objB === null) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
var keysA = Object.keys(objA);
|
||
|
var keysB = Object.keys(objB);
|
||
|
|
||
|
if (keysA.length !== keysB.length) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Test for A's keys different from B.
|
||
|
var bHasOwnProperty = hasOwnProperty.bind(objB);
|
||
|
for (var i = 0; i < keysA.length; i++) {
|
||
|
if (!bHasOwnProperty(keysA[i]) || objA[keysA[i]] !== objB[keysA[i]]) {
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
module.exports = shallowEqual;
|
||
|
},{}],2:[function(require,module,exports){
|
||
|
module.exports = require('react/lib/ReactComponentWithPureRenderMixin');
|
||
|
},{"react/lib/ReactComponentWithPureRenderMixin":3}],3:[function(require,module,exports){
|
||
|
/**
|
||
|
* Copyright 2013-2015, Facebook, Inc.
|
||
|
* All rights reserved.
|
||
|
*
|
||
|
* This source code is licensed under the BSD-style license found in the
|
||
|
* LICENSE file in the root directory of this source tree. An additional grant
|
||
|
* of patent rights can be found in the PATENTS file in the same directory.
|
||
|
*
|
||
|
* @providesModule ReactComponentWithPureRenderMixin
|
||
|
*/
|
||
|
|
||
|
'use strict';
|
||
|
|
||
|
var shallowCompare = require('./shallowCompare');
|
||
|
|
||
|
/**
|
||
|
* If your React component's render function is "pure", e.g. it will render the
|
||
|
* same result given the same props and state, provide this Mixin for a
|
||
|
* considerable performance boost.
|
||
|
*
|
||
|
* Most React components have pure render functions.
|
||
|
*
|
||
|
* Example:
|
||
|
*
|
||
|
* var ReactComponentWithPureRenderMixin =
|
||
|
* require('ReactComponentWithPureRenderMixin');
|
||
|
* React.createClass({
|
||
|
* mixins: [ReactComponentWithPureRenderMixin],
|
||
|
*
|
||
|
* render: function() {
|
||
|
* return <div className={this.props.className}>foo</div>;
|
||
|
* }
|
||
|
* });
|
||
|
*
|
||
|
* Note: This only checks shallow equality for props and state. If these contain
|
||
|
* complex data structures this mixin may have false-negatives for deeper
|
||
|
* differences. Only mixin to components which have simple props and state, or
|
||
|
* use `forceUpdate()` when you know deep data structures have changed.
|
||
|
*/
|
||
|
var ReactComponentWithPureRenderMixin = {
|
||
|
shouldComponentUpdate: function (nextProps, nextState) {
|
||
|
return shallowCompare(this, nextProps, nextState);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
module.exports = ReactComponentWithPureRenderMixin;
|
||
|
},{"./shallowCompare":4}],4:[function(require,module,exports){
|
||
|
/**
|
||
|
* Copyright 2013-2015, Facebook, Inc.
|
||
|
* All rights reserved.
|
||
|
*
|
||
|
* This source code is licensed under the BSD-style license found in the
|
||
|
* LICENSE file in the root directory of this source tree. An additional grant
|
||
|
* of patent rights can be found in the PATENTS file in the same directory.
|
||
|
*
|
||
|
* @providesModule shallowCompare
|
||
|
*/
|
||
|
|
||
|
'use strict';
|
||
|
|
||
|
var shallowEqual = require('fbjs/lib/shallowEqual');
|
||
|
|
||
|
/**
|
||
|
* Does a shallow comparison for props and state.
|
||
|
* See ReactComponentWithPureRenderMixin
|
||
|
*/
|
||
|
function shallowCompare(instance, nextProps, nextState) {
|
||
|
return !shallowEqual(instance.props, nextProps) || !shallowEqual(instance.state, nextState);
|
||
|
}
|
||
|
|
||
|
module.exports = shallowCompare;
|
||
|
},{"fbjs/lib/shallowEqual":1}],5:[function(require,module,exports){
|
||
|
var INFINITE, JSONToShape, LiterallyCanvas, Pencil, actions, bindEvents, createShape, math, ref, renderShapeToContext, renderShapeToSVG, renderSnapshotToImage, renderSnapshotToSVG, shapeToJSON, util,
|
||
|
bind = function(fn, me){ return function(){ return fn.apply(me, arguments); }; },
|
||
|
slice = [].slice,
|
||
|
indexOf = [].indexOf || function(item) { for (var i = 0, l = this.length; i < l; i++) { if (i in this && this[i] === item) return i; } return -1; };
|
||
|
|
||
|
actions = require('./actions');
|
||
|
|
||
|
bindEvents = require('./bindEvents');
|
||
|
|
||
|
math = require('./math');
|
||
|
|
||
|
ref = require('./shapes'), createShape = ref.createShape, shapeToJSON = ref.shapeToJSON, JSONToShape = ref.JSONToShape;
|
||
|
|
||
|
renderShapeToContext = require('./canvasRenderer').renderShapeToContext;
|
||
|
|
||
|
renderShapeToSVG = require('./svgRenderer').renderShapeToSVG;
|
||
|
|
||
|
renderSnapshotToImage = require('./renderSnapshotToImage');
|
||
|
|
||
|
renderSnapshotToSVG = require('./renderSnapshotToSVG');
|
||
|
|
||
|
Pencil = require('../tools/Pencil');
|
||
|
|
||
|
util = require('./util');
|
||
|
|
||
|
INFINITE = 'infinite';
|
||
|
|
||
|
module.exports = LiterallyCanvas = (function() {
|
||
|
function LiterallyCanvas(arg1, arg2) {
|
||
|
this.setImageSize = bind(this.setImageSize, this);
|
||
|
var containerEl, opts;
|
||
|
opts = null;
|
||
|
containerEl = null;
|
||
|
if (arg1 instanceof HTMLElement) {
|
||
|
containerEl = arg1;
|
||
|
opts = arg2;
|
||
|
} else {
|
||
|
opts = arg1;
|
||
|
}
|
||
|
this.opts = opts || {};
|
||
|
this.config = {
|
||
|
zoomMin: opts.zoomMin || 0.2,
|
||
|
zoomMax: opts.zoomMax || 4.0,
|
||
|
zoomStep: opts.zoomStep || 0.2
|
||
|
};
|
||
|
this.colors = {
|
||
|
primary: opts.primaryColor || '#000',
|
||
|
secondary: opts.secondaryColor || '#fff',
|
||
|
background: opts.backgroundColor || 'transparent'
|
||
|
};
|
||
|
this.watermarkImage = opts.watermarkImage;
|
||
|
this.watermarkScale = opts.watermarkScale || 1;
|
||
|
this.backgroundCanvas = document.createElement('canvas');
|
||
|
this.backgroundCtx = this.backgroundCanvas.getContext('2d');
|
||
|
this.canvas = document.createElement('canvas');
|
||
|
this.canvas.style['background-color'] = 'transparent';
|
||
|
this.buffer = document.createElement('canvas');
|
||
|
this.buffer.style['background-color'] = 'transparent';
|
||
|
this.ctx = this.canvas.getContext('2d');
|
||
|
this.bufferCtx = this.buffer.getContext('2d');
|
||
|
this.backingScale = util.getBackingScale(this.ctx);
|
||
|
this.backgroundShapes = opts.backgroundShapes || [];
|
||
|
this._shapesInProgress = [];
|
||
|
this.shapes = [];
|
||
|
this.undoStack = [];
|
||
|
this.redoStack = [];
|
||
|
this.isDragging = false;
|
||
|
this.position = {
|
||
|
x: 0,
|
||
|
y: 0
|
||
|
};
|
||
|
this.scale = 1.0;
|
||
|
this.setTool(new this.opts.tools[0](this));
|
||
|
this.width = opts.imageSize.width || INFINITE;
|
||
|
this.height = opts.imageSize.height || INFINITE;
|
||
|
this.setZoom(this.scale);
|
||
|
if (opts.snapshot) {
|
||
|
this.loadSnapshot(opts.snapshot);
|
||
|
}
|
||
|
this.isBound = false;
|
||
|
if (containerEl) {
|
||
|
this.bindToElement(containerEl);
|
||
|
}
|
||
|
this.respondToSizeChange = function() {};
|
||
|
}
|
||
|
|
||
|
LiterallyCanvas.prototype.bindToElement = function(containerEl) {
|
||
|
var ref1, repaintAll;
|
||
|
if (this.containerEl) {
|
||
|
console.warn("Trying to bind Literally Canvas to a DOM element more than once is unsupported.");
|
||
|
return;
|
||
|
}
|
||
|
this.containerEl = containerEl;
|
||
|
this._unsubscribeEvents = bindEvents(this, this.containerEl, this.opts.keyboardShortcuts);
|
||
|
this.containerEl.style['background-color'] = this.colors.background;
|
||
|
this.containerEl.appendChild(this.backgroundCanvas);
|
||
|
this.containerEl.appendChild(this.canvas);
|
||
|
this.isBound = true;
|
||
|
repaintAll = (function(_this) {
|
||
|
return function() {
|
||
|
_this.keepPanInImageBounds();
|
||
|
return _this.repaintAllLayers();
|
||
|
};
|
||
|
})(this);
|
||
|
this.respondToSizeChange = util.matchElementSize(this.containerEl, [this.backgroundCanvas, this.canvas], this.backingScale, repaintAll);
|
||
|
if (this.watermarkImage) {
|
||
|
this.watermarkImage.onload = (function(_this) {
|
||
|
return function() {
|
||
|
return _this.repaintLayer('background');
|
||
|
};
|
||
|
})(this);
|
||
|
}
|
||
|
if ((ref1 = this.tool) != null) {
|
||
|
ref1.didBecomeActive(this);
|
||
|
}
|
||
|
return repaintAll();
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype._teardown = function() {
|
||
|
var ref1;
|
||
|
if ((ref1 = this.tool) != null) {
|
||
|
ref1.willBecomeInactive(this);
|
||
|
}
|
||
|
if (typeof this._unsubscribeEvents === "function") {
|
||
|
this._unsubscribeEvents();
|
||
|
}
|
||
|
this.tool = null;
|
||
|
this.containerEl = null;
|
||
|
return this.isBound = false;
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.trigger = function(name, data) {
|
||
|
this.canvas.dispatchEvent(new CustomEvent(name, {
|
||
|
detail: data
|
||
|
}));
|
||
|
return null;
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.on = function(name, fn) {
|
||
|
var wrapper;
|
||
|
wrapper = function(e) {
|
||
|
return fn(e.detail);
|
||
|
};
|
||
|
this.canvas.addEventListener(name, wrapper);
|
||
|
return (function(_this) {
|
||
|
return function() {
|
||
|
return _this.canvas.removeEventListener(name, wrapper);
|
||
|
};
|
||
|
})(this);
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.getRenderScale = function() {
|
||
|
return this.scale * this.backingScale;
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.clientCoordsToDrawingCoords = function(x, y) {
|
||
|
return {
|
||
|
x: (x * this.backingScale - this.position.x) / this.getRenderScale(),
|
||
|
y: (y * this.backingScale - this.position.y) / this.getRenderScale()
|
||
|
};
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.drawingCoordsToClientCoords = function(x, y) {
|
||
|
return {
|
||
|
x: x * this.getRenderScale() + this.position.x,
|
||
|
y: y * this.getRenderScale() + this.position.y
|
||
|
};
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.setImageSize = function(width, height) {
|
||
|
this.width = width || INFINITE;
|
||
|
this.height = height || INFINITE;
|
||
|
this.keepPanInImageBounds();
|
||
|
this.repaintAllLayers();
|
||
|
return this.trigger('imageSizeChange', {
|
||
|
width: this.width,
|
||
|
height: this.height
|
||
|
});
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.setTool = function(tool) {
|
||
|
var ref1;
|
||
|
if (this.isBound) {
|
||
|
if ((ref1 = this.tool) != null) {
|
||
|
ref1.willBecomeInactive(this);
|
||
|
}
|
||
|
}
|
||
|
this.tool = tool;
|
||
|
this.trigger('toolChange', {
|
||
|
tool: tool
|
||
|
});
|
||
|
if (this.isBound) {
|
||
|
return this.tool.didBecomeActive(this);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.setShapesInProgress = function(newVal) {
|
||
|
return this._shapesInProgress = newVal;
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.pointerDown = function(x, y) {
|
||
|
var p;
|
||
|
p = this.clientCoordsToDrawingCoords(x, y);
|
||
|
if (this.tool.usesSimpleAPI) {
|
||
|
this.tool.begin(p.x, p.y, this);
|
||
|
this.isDragging = true;
|
||
|
return this.trigger("drawStart", {
|
||
|
tool: this.tool
|
||
|
});
|
||
|
} else {
|
||
|
this.isDragging = true;
|
||
|
return this.trigger("lc-pointerdown", {
|
||
|
tool: this.tool,
|
||
|
x: p.x,
|
||
|
y: p.y,
|
||
|
rawX: x,
|
||
|
rawY: y
|
||
|
});
|
||
|
}
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.pointerMove = function(x, y) {
|
||
|
return util.requestAnimationFrame((function(_this) {
|
||
|
return function() {
|
||
|
var p, ref1;
|
||
|
p = _this.clientCoordsToDrawingCoords(x, y);
|
||
|
if ((ref1 = _this.tool) != null ? ref1.usesSimpleAPI : void 0) {
|
||
|
if (_this.isDragging) {
|
||
|
_this.tool["continue"](p.x, p.y, _this);
|
||
|
return _this.trigger("drawContinue", {
|
||
|
tool: _this.tool
|
||
|
});
|
||
|
}
|
||
|
} else {
|
||
|
if (_this.isDragging) {
|
||
|
return _this.trigger("lc-pointerdrag", {
|
||
|
tool: _this.tool,
|
||
|
x: p.x,
|
||
|
y: p.y,
|
||
|
rawX: x,
|
||
|
rawY: y
|
||
|
});
|
||
|
} else {
|
||
|
return _this.trigger("lc-pointermove", {
|
||
|
tool: _this.tool,
|
||
|
x: p.x,
|
||
|
y: p.y,
|
||
|
rawX: x,
|
||
|
rawY: y
|
||
|
});
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
})(this));
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.pointerUp = function(x, y) {
|
||
|
var p;
|
||
|
p = this.clientCoordsToDrawingCoords(x, y);
|
||
|
if (this.tool.usesSimpleAPI) {
|
||
|
if (this.isDragging) {
|
||
|
this.tool.end(p.x, p.y, this);
|
||
|
this.isDragging = false;
|
||
|
return this.trigger("drawEnd", {
|
||
|
tool: this.tool
|
||
|
});
|
||
|
}
|
||
|
} else {
|
||
|
this.isDragging = false;
|
||
|
return this.trigger("lc-pointerup", {
|
||
|
tool: this.tool,
|
||
|
x: p.x,
|
||
|
y: p.y,
|
||
|
rawX: x,
|
||
|
rawY: y
|
||
|
});
|
||
|
}
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.setColor = function(name, color) {
|
||
|
this.colors[name] = color;
|
||
|
if (!this.isBound) {
|
||
|
return;
|
||
|
}
|
||
|
switch (name) {
|
||
|
case 'background':
|
||
|
this.containerEl.style.backgroundColor = this.colors.background;
|
||
|
this.repaintLayer('background');
|
||
|
break;
|
||
|
case 'primary':
|
||
|
this.repaintLayer('main');
|
||
|
break;
|
||
|
case 'secondary':
|
||
|
this.repaintLayer('main');
|
||
|
}
|
||
|
this.trigger(name + "ColorChange", this.colors[name]);
|
||
|
if (name === 'background') {
|
||
|
return this.trigger("drawingChange");
|
||
|
}
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.getColor = function(name) {
|
||
|
return this.colors[name];
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.saveShape = function(shape, triggerShapeSaveEvent, previousShapeId) {
|
||
|
if (triggerShapeSaveEvent == null) {
|
||
|
triggerShapeSaveEvent = true;
|
||
|
}
|
||
|
if (previousShapeId == null) {
|
||
|
previousShapeId = null;
|
||
|
}
|
||
|
if (!previousShapeId) {
|
||
|
previousShapeId = this.shapes.length ? this.shapes[this.shapes.length - 1].id : null;
|
||
|
}
|
||
|
this.execute(new actions.AddShapeAction(this, shape, previousShapeId));
|
||
|
if (triggerShapeSaveEvent) {
|
||
|
this.trigger('shapeSave', {
|
||
|
shape: shape,
|
||
|
previousShapeId: previousShapeId
|
||
|
});
|
||
|
}
|
||
|
return this.trigger('drawingChange');
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.pan = function(x, y) {
|
||
|
return this.setPan(this.position.x - x, this.position.y - y);
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.keepPanInImageBounds = function() {
|
||
|
var ref1, renderScale, x, y;
|
||
|
renderScale = this.getRenderScale();
|
||
|
ref1 = this.position, x = ref1.x, y = ref1.y;
|
||
|
if (this.width !== INFINITE) {
|
||
|
if (this.canvas.width > this.width * renderScale) {
|
||
|
x = (this.canvas.width - this.width * renderScale) / 2;
|
||
|
} else {
|
||
|
x = Math.max(Math.min(0, x), this.canvas.width - this.width * renderScale);
|
||
|
}
|
||
|
}
|
||
|
if (this.height !== INFINITE) {
|
||
|
if (this.canvas.height > this.height * renderScale) {
|
||
|
y = (this.canvas.height - this.height * renderScale) / 2;
|
||
|
} else {
|
||
|
y = Math.max(Math.min(0, y), this.canvas.height - this.height * renderScale);
|
||
|
}
|
||
|
}
|
||
|
return this.position = {
|
||
|
x: x,
|
||
|
y: y
|
||
|
};
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.setPan = function(x, y) {
|
||
|
this.position = {
|
||
|
x: x,
|
||
|
y: y
|
||
|
};
|
||
|
this.keepPanInImageBounds();
|
||
|
this.repaintAllLayers();
|
||
|
return this.trigger('pan', {
|
||
|
x: this.position.x,
|
||
|
y: this.position.y
|
||
|
});
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.zoom = function(factor) {
|
||
|
var newScale;
|
||
|
newScale = this.scale + factor;
|
||
|
newScale = Math.max(newScale, this.config.zoomMin);
|
||
|
newScale = Math.min(newScale, this.config.zoomMax);
|
||
|
newScale = Math.round(newScale * 100) / 100;
|
||
|
return this.setZoom(newScale);
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.setZoom = function(scale) {
|
||
|
var oldScale;
|
||
|
oldScale = this.scale;
|
||
|
this.scale = scale;
|
||
|
this.position.x = math.scalePositionScalar(this.position.x, this.canvas.width, oldScale, this.scale);
|
||
|
this.position.y = math.scalePositionScalar(this.position.y, this.canvas.height, oldScale, this.scale);
|
||
|
this.keepPanInImageBounds();
|
||
|
this.repaintAllLayers();
|
||
|
return this.trigger('zoom', {
|
||
|
oldScale: oldScale,
|
||
|
newScale: this.scale
|
||
|
});
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.setWatermarkImage = function(newImage) {
|
||
|
this.watermarkImage = newImage;
|
||
|
util.addImageOnload(newImage, (function(_this) {
|
||
|
return function() {
|
||
|
return _this.repaintLayer('background');
|
||
|
};
|
||
|
})(this));
|
||
|
if (newImage.width) {
|
||
|
return this.repaintLayer('background');
|
||
|
}
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.repaintAllLayers = function() {
|
||
|
var i, key, len, ref1;
|
||
|
ref1 = ['background', 'main'];
|
||
|
for (i = 0, len = ref1.length; i < len; i++) {
|
||
|
key = ref1[i];
|
||
|
this.repaintLayer(key);
|
||
|
}
|
||
|
return null;
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.repaintLayer = function(repaintLayerKey, dirty) {
|
||
|
var retryCallback;
|
||
|
if (dirty == null) {
|
||
|
dirty = repaintLayerKey === 'main';
|
||
|
}
|
||
|
if (!this.isBound) {
|
||
|
return;
|
||
|
}
|
||
|
switch (repaintLayerKey) {
|
||
|
case 'background':
|
||
|
this.backgroundCtx.clearRect(0, 0, this.backgroundCanvas.width, this.backgroundCanvas.height);
|
||
|
retryCallback = (function(_this) {
|
||
|
return function() {
|
||
|
return _this.repaintLayer('background');
|
||
|
};
|
||
|
})(this);
|
||
|
if (this.watermarkImage) {
|
||
|
this._renderWatermark(this.backgroundCtx, true, retryCallback);
|
||
|
}
|
||
|
this.draw(this.backgroundShapes, this.backgroundCtx, retryCallback);
|
||
|
break;
|
||
|
case 'main':
|
||
|
retryCallback = (function(_this) {
|
||
|
return function() {
|
||
|
return _this.repaintLayer('main', true);
|
||
|
};
|
||
|
})(this);
|
||
|
if (dirty) {
|
||
|
this.buffer.width = this.canvas.width;
|
||
|
this.buffer.height = this.canvas.height;
|
||
|
this.bufferCtx.clearRect(0, 0, this.buffer.width, this.buffer.height);
|
||
|
this.draw(this.shapes, this.bufferCtx, retryCallback);
|
||
|
}
|
||
|
this.ctx.clearRect(0, 0, this.canvas.width, this.canvas.height);
|
||
|
if (this.canvas.width > 0 && this.canvas.height > 0) {
|
||
|
this.ctx.fillStyle = '#ccc';
|
||
|
this.ctx.fillRect(0, 0, this.canvas.width, this.canvas.height);
|
||
|
this.clipped(((function(_this) {
|
||
|
return function() {
|
||
|
_this.ctx.clearRect(0, 0, _this.canvas.width, _this.canvas.height);
|
||
|
return _this.ctx.drawImage(_this.buffer, 0, 0);
|
||
|
};
|
||
|
})(this)), this.ctx);
|
||
|
this.clipped(((function(_this) {
|
||
|
return function() {
|
||
|
return _this.transformed((function() {
|
||
|
var i, len, ref1, results, shape;
|
||
|
ref1 = _this._shapesInProgress;
|
||
|
results = [];
|
||
|
for (i = 0, len = ref1.length; i < len; i++) {
|
||
|
shape = ref1[i];
|
||
|
results.push(renderShapeToContext(_this.ctx, shape, {
|
||
|
bufferCtx: _this.bufferCtx,
|
||
|
shouldOnlyDrawLatest: true
|
||
|
}));
|
||
|
}
|
||
|
return results;
|
||
|
}), _this.ctx, _this.bufferCtx);
|
||
|
};
|
||
|
})(this)), this.ctx, this.bufferCtx);
|
||
|
}
|
||
|
}
|
||
|
return this.trigger('repaint', {
|
||
|
layerKey: repaintLayerKey
|
||
|
});
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype._renderWatermark = function(ctx, worryAboutRetina, retryCallback) {
|
||
|
if (worryAboutRetina == null) {
|
||
|
worryAboutRetina = true;
|
||
|
}
|
||
|
if (!this.watermarkImage.width) {
|
||
|
this.watermarkImage.onload = retryCallback;
|
||
|
return;
|
||
|
}
|
||
|
ctx.save();
|
||
|
ctx.translate(ctx.canvas.width / 2, ctx.canvas.height / 2);
|
||
|
ctx.scale(this.watermarkScale, this.watermarkScale);
|
||
|
if (worryAboutRetina) {
|
||
|
ctx.scale(this.backingScale, this.backingScale);
|
||
|
}
|
||
|
ctx.drawImage(this.watermarkImage, -this.watermarkImage.width / 2, -this.watermarkImage.height / 2);
|
||
|
return ctx.restore();
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.drawShapeInProgress = function(shape) {
|
||
|
this.repaintLayer('main', false);
|
||
|
return this.clipped(((function(_this) {
|
||
|
return function() {
|
||
|
return _this.transformed((function() {
|
||
|
return renderShapeToContext(_this.ctx, shape, {
|
||
|
bufferCtx: _this.bufferCtx,
|
||
|
shouldOnlyDrawLatest: true
|
||
|
});
|
||
|
}), _this.ctx, _this.bufferCtx);
|
||
|
};
|
||
|
})(this)), this.ctx, this.bufferCtx);
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.draw = function(shapes, ctx, retryCallback) {
|
||
|
var drawShapes;
|
||
|
if (!shapes.length) {
|
||
|
return;
|
||
|
}
|
||
|
drawShapes = (function(_this) {
|
||
|
return function() {
|
||
|
var i, len, results, shape;
|
||
|
results = [];
|
||
|
for (i = 0, len = shapes.length; i < len; i++) {
|
||
|
shape = shapes[i];
|
||
|
results.push(renderShapeToContext(ctx, shape, {
|
||
|
retryCallback: retryCallback
|
||
|
}));
|
||
|
}
|
||
|
return results;
|
||
|
};
|
||
|
})(this);
|
||
|
return this.clipped(((function(_this) {
|
||
|
return function() {
|
||
|
return _this.transformed(drawShapes, ctx);
|
||
|
};
|
||
|
})(this)), ctx);
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.clipped = function() {
|
||
|
var contexts, ctx, fn, height, i, j, len, len1, results, width, x, y;
|
||
|
fn = arguments[0], contexts = 2 <= arguments.length ? slice.call(arguments, 1) : [];
|
||
|
x = this.width === INFINITE ? 0 : this.position.x;
|
||
|
y = this.height === INFINITE ? 0 : this.position.y;
|
||
|
width = (function() {
|
||
|
switch (this.width) {
|
||
|
case INFINITE:
|
||
|
return this.canvas.width;
|
||
|
default:
|
||
|
return this.width * this.getRenderScale();
|
||
|
}
|
||
|
}).call(this);
|
||
|
height = (function() {
|
||
|
switch (this.height) {
|
||
|
case INFINITE:
|
||
|
return this.canvas.height;
|
||
|
default:
|
||
|
return this.height * this.getRenderScale();
|
||
|
}
|
||
|
}).call(this);
|
||
|
for (i = 0, len = contexts.length; i < len; i++) {
|
||
|
ctx = contexts[i];
|
||
|
ctx.save();
|
||
|
ctx.beginPath();
|
||
|
ctx.rect(x, y, width, height);
|
||
|
ctx.clip();
|
||
|
}
|
||
|
fn();
|
||
|
results = [];
|
||
|
for (j = 0, len1 = contexts.length; j < len1; j++) {
|
||
|
ctx = contexts[j];
|
||
|
results.push(ctx.restore());
|
||
|
}
|
||
|
return results;
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.transformed = function() {
|
||
|
var contexts, ctx, fn, i, j, len, len1, results, scale;
|
||
|
fn = arguments[0], contexts = 2 <= arguments.length ? slice.call(arguments, 1) : [];
|
||
|
for (i = 0, len = contexts.length; i < len; i++) {
|
||
|
ctx = contexts[i];
|
||
|
ctx.save();
|
||
|
ctx.translate(Math.floor(this.position.x), Math.floor(this.position.y));
|
||
|
scale = this.getRenderScale();
|
||
|
ctx.scale(scale, scale);
|
||
|
}
|
||
|
fn();
|
||
|
results = [];
|
||
|
for (j = 0, len1 = contexts.length; j < len1; j++) {
|
||
|
ctx = contexts[j];
|
||
|
results.push(ctx.restore());
|
||
|
}
|
||
|
return results;
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.clear = function(triggerClearEvent) {
|
||
|
var newShapes, oldShapes;
|
||
|
if (triggerClearEvent == null) {
|
||
|
triggerClearEvent = true;
|
||
|
}
|
||
|
oldShapes = this.shapes;
|
||
|
newShapes = [];
|
||
|
this.setShapesInProgress([]);
|
||
|
this.execute(new actions.ClearAction(this, oldShapes, newShapes));
|
||
|
this.repaintLayer('main');
|
||
|
if (triggerClearEvent) {
|
||
|
this.trigger('clear', null);
|
||
|
}
|
||
|
return this.trigger('drawingChange', {});
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.execute = function(action) {
|
||
|
this.undoStack.push(action);
|
||
|
action["do"]();
|
||
|
return this.redoStack = [];
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.undo = function() {
|
||
|
var action;
|
||
|
if (!this.undoStack.length) {
|
||
|
return;
|
||
|
}
|
||
|
action = this.undoStack.pop();
|
||
|
action.undo();
|
||
|
this.redoStack.push(action);
|
||
|
this.trigger('undo', {
|
||
|
action: action
|
||
|
});
|
||
|
return this.trigger('drawingChange', {});
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.redo = function() {
|
||
|
var action;
|
||
|
if (!this.redoStack.length) {
|
||
|
return;
|
||
|
}
|
||
|
action = this.redoStack.pop();
|
||
|
this.undoStack.push(action);
|
||
|
action["do"]();
|
||
|
this.trigger('redo', {
|
||
|
action: action
|
||
|
});
|
||
|
return this.trigger('drawingChange', {});
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.canUndo = function() {
|
||
|
return !!this.undoStack.length;
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.canRedo = function() {
|
||
|
return !!this.redoStack.length;
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.getPixel = function(x, y) {
|
||
|
var p, pixel;
|
||
|
p = this.drawingCoordsToClientCoords(x, y);
|
||
|
pixel = this.ctx.getImageData(p.x, p.y, 1, 1).data;
|
||
|
if (pixel[3]) {
|
||
|
return "rgb(" + pixel[0] + ", " + pixel[1] + ", " + pixel[2] + ")";
|
||
|
} else {
|
||
|
return null;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.getContentBounds = function() {
|
||
|
return util.getBoundingRect((this.shapes.concat(this.backgroundShapes)).map(function(s) {
|
||
|
return s.getBoundingRect();
|
||
|
}), this.width === INFINITE ? 0 : this.width, this.height === INFINITE ? 0 : this.height);
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.getDefaultImageRect = function(explicitSize, margin) {
|
||
|
var s;
|
||
|
if (explicitSize == null) {
|
||
|
explicitSize = {
|
||
|
width: 0,
|
||
|
height: 0
|
||
|
};
|
||
|
}
|
||
|
if (margin == null) {
|
||
|
margin = {
|
||
|
top: 0,
|
||
|
right: 0,
|
||
|
bottom: 0,
|
||
|
left: 0
|
||
|
};
|
||
|
}
|
||
|
return util.getDefaultImageRect((function() {
|
||
|
var i, len, ref1, results;
|
||
|
ref1 = this.shapes.concat(this.backgroundShapes);
|
||
|
results = [];
|
||
|
for (i = 0, len = ref1.length; i < len; i++) {
|
||
|
s = ref1[i];
|
||
|
results.push(s.getBoundingRect(this.ctx));
|
||
|
}
|
||
|
return results;
|
||
|
}).call(this), explicitSize, margin);
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.getImage = function(opts) {
|
||
|
if (opts == null) {
|
||
|
opts = {};
|
||
|
}
|
||
|
if (opts.includeWatermark == null) {
|
||
|
opts.includeWatermark = true;
|
||
|
}
|
||
|
if (opts.scaleDownRetina == null) {
|
||
|
opts.scaleDownRetina = true;
|
||
|
}
|
||
|
if (opts.scale == null) {
|
||
|
opts.scale = 1;
|
||
|
}
|
||
|
if (!opts.scaleDownRetina) {
|
||
|
opts.scale *= this.backingScale;
|
||
|
}
|
||
|
if (opts.includeWatermark) {
|
||
|
opts.watermarkImage = this.watermarkImage;
|
||
|
opts.watermarkScale = this.watermarkScale;
|
||
|
if (!opts.scaleDownRetina) {
|
||
|
opts.watermarkScale *= this.backingScale;
|
||
|
}
|
||
|
}
|
||
|
return renderSnapshotToImage(this.getSnapshot(), opts);
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.canvasForExport = function() {
|
||
|
this.repaintAllLayers();
|
||
|
return util.combineCanvases(this.backgroundCanvas, this.canvas);
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.canvasWithBackground = function(backgroundImageOrCanvas) {
|
||
|
return util.combineCanvases(backgroundImageOrCanvas, this.canvasForExport());
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.getSnapshot = function(keys) {
|
||
|
var i, k, len, ref1, shape, snapshot;
|
||
|
if (keys == null) {
|
||
|
keys = null;
|
||
|
}
|
||
|
if (keys == null) {
|
||
|
keys = ['shapes', 'imageSize', 'colors', 'position', 'scale', 'backgroundShapes'];
|
||
|
}
|
||
|
snapshot = {};
|
||
|
ref1 = ['colors', 'position', 'scale'];
|
||
|
for (i = 0, len = ref1.length; i < len; i++) {
|
||
|
k = ref1[i];
|
||
|
if (indexOf.call(keys, k) >= 0) {
|
||
|
snapshot[k] = this[k];
|
||
|
}
|
||
|
}
|
||
|
if (indexOf.call(keys, 'shapes') >= 0) {
|
||
|
snapshot.shapes = (function() {
|
||
|
var j, len1, ref2, results;
|
||
|
ref2 = this.shapes;
|
||
|
results = [];
|
||
|
for (j = 0, len1 = ref2.length; j < len1; j++) {
|
||
|
shape = ref2[j];
|
||
|
results.push(shapeToJSON(shape));
|
||
|
}
|
||
|
return results;
|
||
|
}).call(this);
|
||
|
}
|
||
|
if (indexOf.call(keys, 'backgroundShapes') >= 0) {
|
||
|
snapshot.backgroundShapes = (function() {
|
||
|
var j, len1, ref2, results;
|
||
|
ref2 = this.backgroundShapes;
|
||
|
results = [];
|
||
|
for (j = 0, len1 = ref2.length; j < len1; j++) {
|
||
|
shape = ref2[j];
|
||
|
results.push(shapeToJSON(shape));
|
||
|
}
|
||
|
return results;
|
||
|
}).call(this);
|
||
|
}
|
||
|
if (indexOf.call(keys, 'imageSize') >= 0) {
|
||
|
snapshot.imageSize = {
|
||
|
width: this.width,
|
||
|
height: this.height
|
||
|
};
|
||
|
}
|
||
|
return snapshot;
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.getSnapshotJSON = function() {
|
||
|
console.warn("lc.getSnapshotJSON() is deprecated. use JSON.stringify(lc.getSnapshot()) instead.");
|
||
|
return JSON.stringify(this.getSnapshot());
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.getSVGString = function(opts) {
|
||
|
if (opts == null) {
|
||
|
opts = {};
|
||
|
}
|
||
|
return renderSnapshotToSVG(this.getSnapshot(), opts);
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.loadSnapshot = function(snapshot) {
|
||
|
var i, j, k, len, len1, ref1, ref2, s, shape, shapeRepr;
|
||
|
if (!snapshot) {
|
||
|
return;
|
||
|
}
|
||
|
if (snapshot.colors) {
|
||
|
ref1 = ['primary', 'secondary', 'background'];
|
||
|
for (i = 0, len = ref1.length; i < len; i++) {
|
||
|
k = ref1[i];
|
||
|
this.setColor(k, snapshot.colors[k]);
|
||
|
}
|
||
|
}
|
||
|
if (snapshot.shapes) {
|
||
|
this.shapes = [];
|
||
|
ref2 = snapshot.shapes;
|
||
|
for (j = 0, len1 = ref2.length; j < len1; j++) {
|
||
|
shapeRepr = ref2[j];
|
||
|
shape = JSONToShape(shapeRepr);
|
||
|
if (shape) {
|
||
|
this.execute(new actions.AddShapeAction(this, shape));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if (snapshot.backgroundShapes) {
|
||
|
this.backgroundShapes = (function() {
|
||
|
var l, len2, ref3, results;
|
||
|
ref3 = snapshot.backgroundShapes;
|
||
|
results = [];
|
||
|
for (l = 0, len2 = ref3.length; l < len2; l++) {
|
||
|
s = ref3[l];
|
||
|
results.push(JSONToShape(s));
|
||
|
}
|
||
|
return results;
|
||
|
})();
|
||
|
}
|
||
|
if (snapshot.imageSize) {
|
||
|
this.width = snapshot.imageSize.width;
|
||
|
this.height = snapshot.imageSize.height;
|
||
|
}
|
||
|
if (snapshot.position) {
|
||
|
this.position = snapshot.position;
|
||
|
}
|
||
|
if (snapshot.scale) {
|
||
|
this.scale = snapshot.scale;
|
||
|
}
|
||
|
this.repaintAllLayers();
|
||
|
this.trigger('snapshotLoad');
|
||
|
return this.trigger('drawingChange', {});
|
||
|
};
|
||
|
|
||
|
LiterallyCanvas.prototype.loadSnapshotJSON = function(str) {
|
||
|
console.warn("lc.loadSnapshotJSON() is deprecated. use lc.loadSnapshot(JSON.parse(snapshot)) instead.");
|
||
|
return this.loadSnapshot(JSON.parse(str));
|
||
|
};
|
||
|
|
||
|
return LiterallyCanvas;
|
||
|
|
||
|
})();
|
||
|
|
||
|
|
||
|
},{"../tools/Pencil":48,"./actions":7,"./bindEvents":8,"./canvasRenderer":9,"./math":14,"./renderSnapshotToImage":15,"./renderSnapshotToSVG":16,"./shapes":17,"./svgRenderer":18,"./util":19}],6:[function(require,module,exports){
|
||
|
var TextRenderer, getLinesToRender, getNextLine, parseFontString;
|
||
|
|
||
|
require('./fontmetrics.js');
|
||
|
|
||
|
parseFontString = function(font) {
|
||
|
var fontFamily, fontItems, fontSize, item, j, len, maybeSize, remainingFontString;
|
||
|
fontItems = font.split(' ');
|
||
|
fontSize = 0;
|
||
|
for (j = 0, len = fontItems.length; j < len; j++) {
|
||
|
item = fontItems[j];
|
||
|
maybeSize = parseInt(item.replace("px", ""), 10);
|
||
|
if (!isNaN(maybeSize)) {
|
||
|
fontSize = maybeSize;
|
||
|
}
|
||
|
}
|
||
|
if (!fontSize) {
|
||
|
throw "Font size not found";
|
||
|
}
|
||
|
remainingFontString = font.substring(fontItems[0].length + 1).replace('bold ', '').replace('italic ', '').replace('underline ', '');
|
||
|
fontFamily = remainingFontString;
|
||
|
return {
|
||
|
fontSize: fontSize,
|
||
|
fontFamily: fontFamily
|
||
|
};
|
||
|
};
|
||
|
|
||
|
getNextLine = function(ctx, text, forcedWidth) {
|
||
|
var doesSubstringFit, endIndex, isEndOfString, isNonWord, isWhitespace, lastGoodIndex, lastOkayIndex, nextWordStartIndex, textToHere, wasInWord;
|
||
|
if (!text.length) {
|
||
|
return ['', ''];
|
||
|
}
|
||
|
endIndex = 0;
|
||
|
lastGoodIndex = 0;
|
||
|
lastOkayIndex = 0;
|
||
|
wasInWord = false;
|
||
|
while (true) {
|
||
|
endIndex += 1;
|
||
|
isEndOfString = endIndex >= text.length;
|
||
|
isWhitespace = (!isEndOfString) && text[endIndex].match(/\s/);
|
||
|
isNonWord = isWhitespace || isEndOfString;
|
||
|
textToHere = text.substring(0, endIndex);
|
||
|
doesSubstringFit = forcedWidth ? ctx.measureTextWidth(textToHere).width <= forcedWidth : true;
|
||
|
if (doesSubstringFit) {
|
||
|
lastOkayIndex = endIndex;
|
||
|
}
|
||
|
if (isNonWord && wasInWord) {
|
||
|
wasInWord = false;
|
||
|
if (doesSubstringFit) {
|
||
|
lastGoodIndex = endIndex;
|
||
|
}
|
||
|
}
|
||
|
wasInWord = !isWhitespace;
|
||
|
if (isEndOfString || !doesSubstringFit) {
|
||
|
if (doesSubstringFit) {
|
||
|
return [text, ''];
|
||
|
} else if (lastGoodIndex > 0) {
|
||
|
nextWordStartIndex = lastGoodIndex + 1;
|
||
|
while (nextWordStartIndex < text.length && text[nextWordStartIndex].match('/\s/')) {
|
||
|
nextWordStartIndex += 1;
|
||
|
}
|
||
|
return [text.substring(0, lastGoodIndex), text.substring(nextWordStartIndex)];
|
||
|
} else {
|
||
|
return [text.substring(0, lastOkayIndex), text.substring(lastOkayIndex)];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
getLinesToRender = function(ctx, text, forcedWidth) {
|
||
|
var j, len, lines, nextLine, ref, ref1, remainingText, textLine, textSplitOnLines;
|
||
|
textSplitOnLines = text.split(/\r\n|\r|\n/g);
|
||
|
lines = [];
|
||
|
for (j = 0, len = textSplitOnLines.length; j < len; j++) {
|
||
|
textLine = textSplitOnLines[j];
|
||
|
ref = getNextLine(ctx, textLine, forcedWidth), nextLine = ref[0], remainingText = ref[1];
|
||
|
if (nextLine) {
|
||
|
while (nextLine) {
|
||
|
lines.push(nextLine);
|
||
|
ref1 = getNextLine(ctx, remainingText, forcedWidth), nextLine = ref1[0], remainingText = ref1[1];
|
||
|
}
|
||
|
} else {
|
||
|
lines.push(textLine);
|
||
|
}
|
||
|
}
|
||
|
return lines;
|
||
|
};
|
||
|
|
||
|
TextRenderer = (function() {
|
||
|
function TextRenderer(ctx, text1, font1, forcedWidth1, forcedHeight) {
|
||
|
var fontFamily, fontSize, ref;
|
||
|
this.text = text1;
|
||
|
this.font = font1;
|
||
|
this.forcedWidth = forcedWidth1;
|
||
|
this.forcedHeight = forcedHeight;
|
||
|
ref = parseFontString(this.font), fontFamily = ref.fontFamily, fontSize = ref.fontSize;
|
||
|
ctx.font = this.font;
|
||
|
ctx.textBaseline = 'baseline';
|
||
|
this.emDashWidth = ctx.measureTextWidth('—', fontSize, fontFamily).width;
|
||
|
this.caratWidth = ctx.measureTextWidth('|', fontSize, fontFamily).width;
|
||
|
this.lines = getLinesToRender(ctx, this.text, this.forcedWidth);
|
||
|
this.metricses = this.lines.map((function(_this) {
|
||
|
return function(line) {
|
||
|
return ctx.measureText2(line || 'X', fontSize, _this.font);
|
||
|
};
|
||
|
})(this));
|
||
|
this.metrics = {
|
||
|
ascent: Math.max.apply(Math, this.metricses.map(function(arg) {
|
||
|
var ascent;
|
||
|
ascent = arg.ascent;
|
||
|
return ascent;
|
||
|
})),
|
||
|
descent: Math.max.apply(Math, this.metricses.map(function(arg) {
|
||
|
var descent;
|
||
|
descent = arg.descent;
|
||
|
return descent;
|
||
|
})),
|
||
|
fontsize: Math.max.apply(Math, this.metricses.map(function(arg) {
|
||
|
var fontsize;
|
||
|
fontsize = arg.fontsize;
|
||
|
return fontsize;
|
||
|
})),
|
||
|
leading: Math.max.apply(Math, this.metricses.map(function(arg) {
|
||
|
var leading;
|
||
|
leading = arg.leading;
|
||
|
return leading;
|
||
|
})),
|
||
|
width: Math.max.apply(Math, this.metricses.map(function(arg) {
|
||
|
var width;
|
||
|
width = arg.width;
|
||
|
return width;
|
||
|
})),
|
||
|
height: Math.max.apply(Math, this.metricses.map(function(arg) {
|
||
|
var height;
|
||
|
height = arg.height;
|
||
|
return height;
|
||
|
})),
|
||
|
bounds: {
|
||
|
minx: Math.min.apply(Math, this.metricses.map(function(arg) {
|
||
|
var bounds;
|
||
|
bounds = arg.bounds;
|
||
|
return bounds.minx;
|
||
|
})),
|
||
|
miny: Math.min.apply(Math, this.metricses.map(function(arg) {
|
||
|
var bounds;
|
||
|
bounds = arg.bounds;
|
||
|
return bounds.miny;
|
||
|
})),
|
||
|
maxx: Math.max.apply(Math, this.metricses.map(function(arg) {
|
||
|
var bounds;
|
||
|
bounds = arg.bounds;
|
||
|
return bounds.maxx;
|
||
|
})),
|
||
|
maxy: Math.max.apply(Math, this.metricses.map(function(arg) {
|
||
|
var bounds;
|
||
|
bounds = arg.bounds;
|
||
|
return bounds.maxy;
|
||
|
}))
|
||
|
}
|
||
|
};
|
||
|
this.boundingBoxWidth = Math.ceil(this.metrics.width);
|
||
|
}
|
||
|
|
||
|
TextRenderer.prototype.draw = function(ctx, x, y) {
|
||
|
var i, j, len, line, ref, results;
|
||
|
ctx.textBaseline = 'top';
|
||
|
ctx.font = this.font;
|
||
|
i = 0;
|
||
|
ref = this.lines;
|
||
|
results = [];
|
||
|
for (j = 0, len = ref.length; j < len; j++) {
|
||
|
line = ref[j];
|
||
|
ctx.fillText(line, x, y + i * this.metrics.leading);
|
||
|
results.push(i += 1);
|
||
|
}
|
||
|
return results;
|
||
|
};
|
||
|
|
||
|
TextRenderer.prototype.getWidth = function(isEditing) {
|
||
|
if (isEditing == null) {
|
||
|
isEditing = false;
|
||
|
}
|
||
|
if (this.forcedWidth) {
|
||
|
return this.forcedWidth;
|
||
|
} else {
|
||
|
if (isEditing) {
|
||
|
return this.metrics.bounds.maxx + this.caratWidth;
|
||
|
} else {
|
||
|
return this.metrics.bounds.maxx;
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
TextRenderer.prototype.getHeight = function() {
|
||
|
return this.forcedHeight || (this.metrics.leading * this.lines.length);
|
||
|
};
|
||
|
|
||
|
return TextRenderer;
|
||
|
|
||
|
})();
|
||
|
|
||
|
module.exports = TextRenderer;
|
||
|
|
||
|
|
||
|
},{"./fontmetrics.js":11}],7:[function(require,module,exports){
|
||
|
var AddShapeAction, ClearAction;
|
||
|
|
||
|
ClearAction = (function() {
|
||
|
function ClearAction(lc1, oldShapes, newShapes1) {
|
||
|
this.lc = lc1;
|
||
|
this.oldShapes = oldShapes;
|
||
|
this.newShapes = newShapes1;
|
||
|
}
|
||
|
|
||
|
ClearAction.prototype["do"] = function() {
|
||
|
this.lc.shapes = this.newShapes;
|
||
|
return this.lc.repaintLayer('main');
|
||
|
};
|
||
|
|
||
|
ClearAction.prototype.undo = function() {
|
||
|
this.lc.shapes = this.oldShapes;
|
||
|
return this.lc.repaintLayer('main');
|
||
|
};
|
||
|
|
||
|
return ClearAction;
|
||
|
|
||
|
})();
|
||
|
|
||
|
AddShapeAction = (function() {
|
||
|
function AddShapeAction(lc1, shape1, previousShapeId) {
|
||
|
this.lc = lc1;
|
||
|
this.shape = shape1;
|
||
|
this.previousShapeId = previousShapeId != null ? previousShapeId : null;
|
||
|
}
|
||
|
|
||
|
AddShapeAction.prototype["do"] = function() {
|
||
|
var found, i, len, newShapes, ref, shape;
|
||
|
if (!this.lc.shapes.length || this.lc.shapes[this.lc.shapes.length - 1].id === this.previousShapeId || this.previousShapeId === null) {
|
||
|
this.lc.shapes.push(this.shape);
|
||
|
} else {
|
||
|
newShapes = [];
|
||
|
found = false;
|
||
|
ref = this.lc.shapes;
|
||
|
for (i = 0, len = ref.length; i < len; i++) {
|
||
|
shape = ref[i];
|
||
|
newShapes.push(shape);
|
||
|
if (shape.id === this.previousShapeId) {
|
||
|
newShapes.push(this.shape);
|
||
|
found = true;
|
||
|
}
|
||
|
}
|
||
|
if (!found) {
|
||
|
newShapes.push(this.shape);
|
||
|
}
|
||
|
this.lc.shapes = newShapes;
|
||
|
}
|
||
|
return this.lc.repaintLayer('main');
|
||
|
};
|
||
|
|
||
|
AddShapeAction.prototype.undo = function() {
|
||
|
var i, len, newShapes, ref, shape;
|
||
|
if (this.lc.shapes[this.lc.shapes.length - 1].id === this.shape.id) {
|
||
|
this.lc.shapes.pop();
|
||
|
} else {
|
||
|
newShapes = [];
|
||
|
ref = this.lc.shapes;
|
||
|
for (i = 0, len = ref.length; i < len; i++) {
|
||
|
shape = ref[i];
|
||
|
if (shape.id !== this.shape.id) {
|
||
|
newShapes.push(shape);
|
||
|
}
|
||
|
}
|
||
|
lc.shapes = newShapes;
|
||
|
}
|
||
|
return this.lc.repaintLayer('main');
|
||
|
};
|
||
|
|
||
|
return AddShapeAction;
|
||
|
|
||
|
})();
|
||
|
|
||
|
module.exports = {
|
||
|
ClearAction: ClearAction,
|
||
|
AddShapeAction: AddShapeAction
|
||
|
};
|
||
|
|
||
|
|
||
|
},{}],8:[function(require,module,exports){
|
||
|
var bindEvents, buttonIsDown, coordsForTouchEvent, position;
|
||
|
|
||
|
coordsForTouchEvent = function(el, e) {
|
||
|
var p, tx, ty;
|
||
|
tx = e.changedTouches[0].clientX;
|
||
|
ty = e.changedTouches[0].clientY;
|
||
|
p = el.getBoundingClientRect();
|
||
|
return [tx - p.left, ty - p.top];
|
||
|
};
|
||
|
|
||
|
position = function(el, e) {
|
||
|
var p;
|
||
|
p = el.getBoundingClientRect();
|
||
|
return {
|
||
|
left: e.clientX - p.left,
|
||
|
top: e.clientY - p.top
|
||
|
};
|
||
|
};
|
||
|
|
||
|
buttonIsDown = function(e) {
|
||
|
if (e.buttons != null) {
|
||
|
return e.buttons === 1;
|
||
|
} else {
|
||
|
return e.which > 0;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
module.exports = bindEvents = function(lc, canvas, panWithKeyboard) {
|
||
|
var listener, mouseMoveListener, mouseUpListener, touchEndListener, touchMoveListener, unsubs;
|
||
|
if (panWithKeyboard == null) {
|
||
|
panWithKeyboard = false;
|
||
|
}
|
||
|
unsubs = [];
|
||
|
mouseMoveListener = (function(_this) {
|
||
|
return function(e) {
|
||
|
var p;
|
||
|
e.preventDefault();
|
||
|
p = position(canvas, e);
|
||
|
return lc.pointerMove(p.left, p.top);
|
||
|
};
|
||
|
})(this);
|
||
|
mouseUpListener = (function(_this) {
|
||
|
return function(e) {
|
||
|
var p;
|
||
|
e.preventDefault();
|
||
|
canvas.onselectstart = function() {
|
||
|
return true;
|
||
|
};
|
||
|
p = position(canvas, e);
|
||
|
lc.pointerUp(p.left, p.top);
|
||
|
document.removeEventListener('mousemove', mouseMoveListener);
|
||
|
document.removeEventListener('mouseup', mouseUpListener);
|
||
|
return canvas.addEventListener('mousemove', mouseMoveListener);
|
||
|
};
|
||
|
})(this);
|
||
|
canvas.addEventListener('mousedown', (function(_this) {
|
||
|
return function(e) {
|
||
|
var down, p;
|
||
|
if (e.target.tagName.toLowerCase() !== 'canvas') {
|
||
|
return;
|
||
|
}
|
||
|
down = true;
|
||
|
e.preventDefault();
|
||
|
canvas.onselectstart = function() {
|
||
|
return false;
|
||
|
};
|
||
|
p = position(canvas, e);
|
||
|
lc.pointerDown(p.left, p.top);
|
||
|
canvas.removeEventListener('mousemove', mouseMoveListener);
|
||
|
document.addEventListener('mousemove', mouseMoveListener);
|
||
|
return document.addEventListener('mouseup', mouseUpListener);
|
||
|
};
|
||
|
})(this));
|
||
|
touchMoveListener = function(e) {
|
||
|
e.preventDefault();
|
||
|
return lc.pointerMove.apply(lc, coordsForTouchEvent(canvas, e));
|
||
|
};
|
||
|
touchEndListener = function(e) {
|
||
|
e.preventDefault();
|
||
|
lc.pointerUp.apply(lc, coordsForTouchEvent(canvas, e));
|
||
|
document.removeEventListener('touchmove', touchMoveListener);
|
||
|
document.removeEventListener('touchend', touchEndListener);
|
||
|
return document.removeEventListener('touchcancel', touchEndListener);
|
||
|
};
|
||
|
canvas.addEventListener('touchstart', function(e) {
|
||
|
if (e.target.tagName.toLowerCase() !== 'canvas') {
|
||
|
return;
|
||
|
}
|
||
|
e.preventDefault();
|
||
|
if (e.touches.length === 1) {
|
||
|
lc.pointerDown.apply(lc, coordsForTouchEvent(canvas, e));
|
||
|
document.addEventListener('touchmove', touchMoveListener);
|
||
|
document.addEventListener('touchend', touchEndListener);
|
||
|
return document.addEventListener('touchcancel', touchEndListener);
|
||
|
} else {
|
||
|
return lc.pointerMove.apply(lc, coordsForTouchEvent(canvas, e));
|
||
|
}
|
||
|
});
|
||
|
if (panWithKeyboard) {
|
||
|
console.warn("Keyboard panning is deprecated.");
|
||
|
listener = function(e) {
|
||
|
switch (e.keyCode) {
|
||
|
case 37:
|
||
|
lc.pan(-10, 0);
|
||
|
break;
|
||
|
case 38:
|
||
|
lc.pan(0, -10);
|
||
|
break;
|
||
|
case 39:
|
||
|
lc.pan(10, 0);
|
||
|
break;
|
||
|
case 40:
|
||
|
lc.pan(0, 10);
|
||
|
}
|
||
|
return lc.repaintAllLayers();
|
||
|
};
|
||
|
document.addEventListener('keydown', listener);
|
||
|
unsubs.push(function() {
|
||
|
return document.removeEventListener(listener);
|
||
|
});
|
||
|
}
|
||
|
return function() {
|
||
|
var f, i, len, results;
|
||
|
results = [];
|
||
|
for (i = 0, len = unsubs.length; i < len; i++) {
|
||
|
f = unsubs[i];
|
||
|
results.push(f());
|
||
|
}
|
||
|
return results;
|
||
|
};
|
||
|
};
|
||
|
|
||
|
|
||
|
},{}],9:[function(require,module,exports){
|
||
|
var _drawRawLinePath, defineCanvasRenderer, drawErasedLinePath, drawErasedLinePathLatest, drawLinePath, drawLinePathLatest, lineEndCapShapes, noop, renderShapeToCanvas, renderShapeToContext, renderers;
|
||
|
|
||
|
lineEndCapShapes = require('./lineEndCapShapes');
|
||
|
|
||
|
renderers = {};
|
||
|
|
||
|
defineCanvasRenderer = function(shapeName, drawFunc, drawLatestFunc) {
|
||
|
return renderers[shapeName] = {
|
||
|
drawFunc: drawFunc,
|
||
|
drawLatestFunc: drawLatestFunc
|
||
|
};
|
||
|
};
|
||
|
|
||
|
noop = function() {};
|
||
|
|
||
|
renderShapeToContext = function(ctx, shape, opts) {
|
||
|
var bufferCtx;
|
||
|
if (opts == null) {
|
||
|
opts = {};
|
||
|
}
|
||
|
if (opts.shouldIgnoreUnsupportedShapes == null) {
|
||
|
opts.shouldIgnoreUnsupportedShapes = false;
|
||
|
}
|
||
|
if (opts.retryCallback == null) {
|
||
|
opts.retryCallback = noop;
|
||
|
}
|
||
|
if (opts.shouldOnlyDrawLatest == null) {
|
||
|
opts.shouldOnlyDrawLatest = false;
|
||
|
}
|
||
|
if (opts.bufferCtx == null) {
|
||
|
opts.bufferCtx = null;
|
||
|
}
|
||
|
bufferCtx = opts.bufferCtx;
|
||
|
if (renderers[shape.className]) {
|
||
|
if (opts.shouldOnlyDrawLatest && renderers[shape.className].drawLatestFunc) {
|
||
|
return renderers[shape.className].drawLatestFunc(ctx, bufferCtx, shape, opts.retryCallback);
|
||
|
} else {
|
||
|
return renderers[shape.className].drawFunc(ctx, shape, opts.retryCallback);
|
||
|
}
|
||
|
} else if (opts.shouldIgnoreUnsupportedShapes) {
|
||
|
return console.warn("Can't render shape of type " + shape.className + " to canvas");
|
||
|
} else {
|
||
|
throw "Can't render shape of type " + shape.className + " to canvas";
|
||
|
}
|
||
|
};
|
||
|
|
||
|
renderShapeToCanvas = function(canvas, shape, opts) {
|
||
|
return renderShapeToContext(canvas.getContext('2d'), shape, opts);
|
||
|
};
|
||
|
|
||
|
defineCanvasRenderer('Rectangle', function(ctx, shape) {
|
||
|
var x, y;
|
||
|
x = shape.x;
|
||
|
y = shape.y;
|
||
|
if (shape.strokeWidth % 2 !== 0) {
|
||
|
x += 0.5;
|
||
|
y += 0.5;
|
||
|
}
|
||
|
ctx.fillStyle = shape.fillColor;
|
||
|
ctx.fillRect(x, y, shape.width, shape.height);
|
||
|
ctx.lineWidth = shape.strokeWidth;
|
||
|
ctx.strokeStyle = shape.strokeColor;
|
||
|
return ctx.strokeRect(x, y, shape.width, shape.height);
|
||
|
});
|
||
|
|
||
|
defineCanvasRenderer('Ellipse', function(ctx, shape) {
|
||
|
var centerX, centerY, halfHeight, halfWidth;
|
||
|
ctx.save();
|
||
|
halfWidth = Math.floor(shape.width / 2);
|
||
|
halfHeight = Math.floor(shape.height / 2);
|
||
|
centerX = shape.x + halfWidth;
|
||
|
centerY = shape.y + halfHeight;
|
||
|
ctx.translate(centerX, centerY);
|
||
|
ctx.scale(1, Math.abs(shape.height / shape.width));
|
||
|
ctx.beginPath();
|
||
|
ctx.arc(0, 0, Math.abs(halfWidth), 0, Math.PI * 2);
|
||
|
ctx.closePath();
|
||
|
ctx.restore();
|
||
|
ctx.fillStyle = shape.fillColor;
|
||
|
ctx.fill();
|
||
|
ctx.lineWidth = shape.strokeWidth;
|
||
|
ctx.strokeStyle = shape.strokeColor;
|
||
|
return ctx.stroke();
|
||
|
});
|
||
|
|
||
|
defineCanvasRenderer('SelectionBox', (function() {
|
||
|
var _drawHandle;
|
||
|
_drawHandle = function(ctx, arg, handleSize) {
|
||
|
var x, y;
|
||
|
x = arg.x, y = arg.y;
|
||
|
if (handleSize === 0) {
|
||
|
return;
|
||
|
}
|
||
|
ctx.fillStyle = '#fff';
|
||
|
ctx.fillRect(x, y, handleSize, handleSize);
|
||
|
ctx.strokeStyle = '#000';
|
||
|
return ctx.strokeRect(x, y, handleSize, handleSize);
|
||
|
};
|
||
|
return function(ctx, shape) {
|
||
|
_drawHandle(ctx, shape.getTopLeftHandleRect(), shape.handleSize);
|
||
|
_drawHandle(ctx, shape.getTopRightHandleRect(), shape.handleSize);
|
||
|
_drawHandle(ctx, shape.getBottomLeftHandleRect(), shape.handleSize);
|
||
|
_drawHandle(ctx, shape.getBottomRightHandleRect(), shape.handleSize);
|
||
|
if (shape.backgroundColor) {
|
||
|
ctx.fillStyle = shape.backgroundColor;
|
||
|
ctx.fillRect(shape._br.x - shape.margin, shape._br.y - shape.margin, shape._br.width + shape.margin * 2, shape._br.height + shape.margin * 2);
|
||
|
}
|
||
|
ctx.lineWidth = 1;
|
||
|
ctx.strokeStyle = '#000';
|
||
|
ctx.setLineDash([2, 4]);
|
||
|
ctx.strokeRect(shape._br.x - shape.margin, shape._br.y - shape.margin, shape._br.width + shape.margin * 2, shape._br.height + shape.margin * 2);
|
||
|
return ctx.setLineDash([]);
|
||
|
};
|
||
|
})());
|
||
|
|
||
|
defineCanvasRenderer('Image', function(ctx, shape, retryCallback) {
|
||
|
if (shape.image.width) {
|
||
|
if (shape.scale === 1) {
|
||
|
return ctx.drawImage(shape.image, shape.x, shape.y);
|
||
|
} else {
|
||
|
return ctx.drawImage(shape.image, shape.x, shape.y, shape.image.width * shape.scale, shape.image.height * shape.scale);
|
||
|
}
|
||
|
} else if (retryCallback) {
|
||
|
return shape.image.onload = retryCallback;
|
||
|
}
|
||
|
});
|
||
|
|
||
|
defineCanvasRenderer('Line', function(ctx, shape) {
|
||
|
var arrowWidth, x1, x2, y1, y2;
|
||
|
if (shape.x1 === shape.x2 && shape.y1 === shape.y2) {
|
||
|
return;
|
||
|
}
|
||
|
x1 = shape.x1;
|
||
|
x2 = shape.x2;
|
||
|
y1 = shape.y1;
|
||
|
y2 = shape.y2;
|
||
|
if (shape.strokeWidth % 2 !== 0) {
|
||
|
x1 += 0.5;
|
||
|
x2 += 0.5;
|
||
|
y1 += 0.5;
|
||
|
y2 += 0.5;
|
||
|
}
|
||
|
ctx.lineWidth = shape.strokeWidth;
|
||
|
ctx.strokeStyle = shape.color;
|
||
|
ctx.lineCap = shape.capStyle;
|
||
|
if (shape.dash) {
|
||
|
ctx.setLineDash(shape.dash);
|
||
|
}
|
||
|
ctx.beginPath();
|
||
|
ctx.moveTo(x1, y1);
|
||
|
ctx.lineTo(x2, y2);
|
||
|
ctx.stroke();
|
||
|
if (shape.dash) {
|
||
|
ctx.setLineDash([]);
|
||
|
}
|
||
|
arrowWidth = Math.max(shape.strokeWidth * 2.2, 5);
|
||
|
if (shape.endCapShapes[0]) {
|
||
|
lineEndCapShapes[shape.endCapShapes[0]].drawToCanvas(ctx, x1, y1, Math.atan2(y1 - y2, x1 - x2), arrowWidth, shape.color);
|
||
|
}
|
||
|
if (shape.endCapShapes[1]) {
|
||
|
return lineEndCapShapes[shape.endCapShapes[1]].drawToCanvas(ctx, x2, y2, Math.atan2(y2 - y1, x2 - x1), arrowWidth, shape.color);
|
||
|
}
|
||
|
});
|
||
|
|
||
|
_drawRawLinePath = function(ctx, points, close, lineCap) {
|
||
|
var i, len, point, ref;
|
||
|
if (close == null) {
|
||
|
close = false;
|
||
|
}
|
||
|
if (lineCap == null) {
|
||
|
lineCap = 'round';
|
||
|
}
|
||
|
if (!points.length) {
|
||
|
return;
|
||
|
}
|
||
|
ctx.lineCap = lineCap;
|
||
|
ctx.strokeStyle = points[0].color;
|
||
|
ctx.lineWidth = points[0].size;
|
||
|
ctx.beginPath();
|
||
|
if (points[0].size % 2 === 0) {
|
||
|
ctx.moveTo(points[0].x, points[0].y);
|
||
|
} else {
|
||
|
ctx.moveTo(points[0].x + 0.5, points[0].y + 0.5);
|
||
|
}
|
||
|
ref = points.slice(1);
|
||
|
for (i = 0, len = ref.length; i < len; i++) {
|
||
|
point = ref[i];
|
||
|
if (points[0].size % 2 === 0) {
|
||
|
ctx.lineTo(point.x, point.y);
|
||
|
} else {
|
||
|
ctx.lineTo(point.x + 0.5, point.y + 0.5);
|
||
|
}
|
||
|
}
|
||
|
if (close) {
|
||
|
return ctx.closePath();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
drawLinePath = function(ctx, shape) {
|
||
|
_drawRawLinePath(ctx, shape.smoothedPoints);
|
||
|
return ctx.stroke();
|
||
|
};
|
||
|
|
||
|
drawLinePathLatest = function(ctx, bufferCtx, shape) {
|
||
|
var drawEnd, drawStart, segmentStart;
|
||
|
if (shape.tail) {
|
||
|
segmentStart = shape.smoothedPoints.length - shape.segmentSize * shape.tailSize;
|
||
|
drawStart = segmentStart < shape.segmentSize * 2 ? 0 : segmentStart;
|
||
|
drawEnd = segmentStart + shape.segmentSize + 1;
|
||
|
_drawRawLinePath(bufferCtx, shape.smoothedPoints.slice(drawStart, drawEnd));
|
||
|
return bufferCtx.stroke();
|
||
|
} else {
|
||
|
_drawRawLinePath(bufferCtx, shape.smoothedPoints);
|
||
|
return bufferCtx.stroke();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
defineCanvasRenderer('LinePath', drawLinePath, drawLinePathLatest);
|
||
|
|
||
|
drawErasedLinePath = function(ctx, shape) {
|
||
|
ctx.save();
|
||
|
ctx.globalCompositeOperation = "destination-out";
|
||
|
drawLinePath(ctx, shape);
|
||
|
return ctx.restore();
|
||
|
};
|
||
|
|
||
|
drawErasedLinePathLatest = function(ctx, bufferCtx, shape) {
|
||
|
ctx.save();
|
||
|
ctx.globalCompositeOperation = "destination-out";
|
||
|
bufferCtx.save();
|
||
|
bufferCtx.globalCompositeOperation = "destination-out";
|
||
|
drawLinePathLatest(ctx, bufferCtx, shape);
|
||
|
ctx.restore();
|
||
|
return bufferCtx.restore();
|
||
|
};
|
||
|
|
||
|
defineCanvasRenderer('ErasedLinePath', drawErasedLinePath, drawErasedLinePathLatest);
|
||
|
|
||
|
defineCanvasRenderer('Text', function(ctx, shape) {
|
||
|
if (!shape.renderer) {
|
||
|
shape._makeRenderer(ctx);
|
||
|
}
|
||
|
ctx.fillStyle = shape.color;
|
||
|
return shape.renderer.draw(ctx, shape.x, shape.y);
|
||
|
});
|
||
|
|
||
|
defineCanvasRenderer('Polygon', function(ctx, shape) {
|
||
|
ctx.fillStyle = shape.fillColor;
|
||
|
_drawRawLinePath(ctx, shape.points, shape.isClosed, 'butt');
|
||
|
ctx.fill();
|
||
|
return ctx.stroke();
|
||
|
});
|
||
|
|
||
|
module.exports = {
|
||
|
defineCanvasRenderer: defineCanvasRenderer,
|
||
|
renderShapeToCanvas: renderShapeToCanvas,
|
||
|
renderShapeToContext: renderShapeToContext
|
||
|
};
|
||
|
|
||
|
|
||
|
},{"./lineEndCapShapes":12}],10:[function(require,module,exports){
|
||
|
'use strict';
|
||
|
|
||
|
module.exports = {
|
||
|
imageURLPrefix: 'lib/img',
|
||
|
primaryColor: 'hsla(0, 0%, 0%, 1)',
|
||
|
secondaryColor: 'hsla(0, 0%, 100%, 1)',
|
||
|
backgroundColor: 'transparent',
|
||
|
strokeWidths: [1, 2, 5, 10, 20, 30],
|
||
|
defaultStrokeWidth: 5,
|
||
|
toolbarPosition: 'top',
|
||
|
keyboardShortcuts: false,
|
||
|
imageSize: { width: 'infinite', height: 'infinite' },
|
||
|
backgroundShapes: [],
|
||
|
watermarkImage: null,
|
||
|
watermarkScale: 1,
|
||
|
zoomMin: 0.2,
|
||
|
zoomMax: 4.0,
|
||
|
zoomStep: 0.2,
|
||
|
snapshot: null,
|
||
|
onInit: function onInit() {},
|
||
|
tools: [require('../tools/Pencil'), require('../tools/Eraser'), require('../tools/Line'), require('../tools/Rectangle'), require('../tools/Ellipse'), require('../tools/Text'), require('../tools/Polygon'), require('../tools/Pan'), require('../tools/Eyedropper')]
|
||
|
};
|
||
|
|
||
|
},{"../tools/Ellipse":43,"../tools/Eraser":44,"../tools/Eyedropper":45,"../tools/Line":46,"../tools/Pan":47,"../tools/Pencil":48,"../tools/Polygon":49,"../tools/Rectangle":50,"../tools/Text":52}],11:[function(require,module,exports){
|
||
|
"use strict";
|
||
|
|
||
|
/**
|
||
|
This library rewrites the Canvas2D "measureText" function
|
||
|
so that it returns a more complete metrics object.
|
||
|
This library is licensed under the MIT (Expat) license,
|
||
|
the text for which is included below.
|
||
|
|
||
|
** -----------------------------------------------------------------------------
|
||
|
|
||
|
CHANGELOG:
|
||
|
|
||
|
2012-01-21 - Whitespace handling added by Joe Turner
|
||
|
(https://github.com/oampo)
|
||
|
|
||
|
2015-06-08 - Various hacks added by Steve Johnson
|
||
|
|
||
|
** -----------------------------------------------------------------------------
|
||
|
|
||
|
Copyright (C) 2011 by Mike "Pomax" Kamermans
|
||
|
|
||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
|
of this software and associated documentation files (the "Software"), to deal
|
||
|
in the Software without restriction, including without limitation the rights
|
||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
|
copies of the Software, and to permit persons to whom the Software is
|
||
|
furnished to do so, subject to the following conditions:
|
||
|
|
||
|
The above copyright notice and this permission notice shall be included in
|
||
|
all copies or substantial portions of the Software.
|
||
|
|
||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||
|
THE SOFTWARE.
|
||
|
**/
|
||
|
(function () {
|
||
|
var NAME = "FontMetrics Library";
|
||
|
var VERSION = "1-2012.0121.1300";
|
||
|
|
||
|
// if there is no getComputedStyle, this library won't work.
|
||
|
if (!document.defaultView.getComputedStyle) {
|
||
|
throw "ERROR: 'document.defaultView.getComputedStyle' not found. This library only works in browsers that can report computed CSS values.";
|
||
|
}
|
||
|
|
||
|
// store the old text metrics function on the Canvas2D prototype
|
||
|
CanvasRenderingContext2D.prototype.measureTextWidth = CanvasRenderingContext2D.prototype.measureText;
|
||
|
|
||
|
/**
|
||
|
* shortcut function for getting computed CSS values
|
||
|
*/
|
||
|
var getCSSValue = function getCSSValue(element, property) {
|
||
|
return document.defaultView.getComputedStyle(element, null).getPropertyValue(property);
|
||
|
};
|
||
|
|
||
|
// debug function
|
||
|
var show = function show(canvas, ctx, xstart, w, h, metrics) {
|
||
|
document.body.appendChild(canvas);
|
||
|
ctx.strokeStyle = 'rgba(0, 0, 0, 0.5)';
|
||
|
|
||
|
ctx.beginPath();
|
||
|
ctx.moveTo(xstart, 0);
|
||
|
ctx.lineTo(xstart, h);
|
||
|
ctx.closePath();
|
||
|
ctx.stroke();
|
||
|
|
||
|
ctx.beginPath();
|
||
|
ctx.moveTo(xstart + metrics.bounds.maxx, 0);
|
||
|
ctx.lineTo(xstart + metrics.bounds.maxx, h);
|
||
|
ctx.closePath();
|
||
|
ctx.stroke();
|
||
|
|
||
|
ctx.beginPath();
|
||
|
ctx.moveTo(0, h / 2 - metrics.ascent);
|
||
|
ctx.lineTo(w, h / 2 - metrics.ascent);
|
||
|
ctx.closePath();
|
||
|
ctx.stroke();
|
||
|
|
||
|
ctx.beginPath();
|
||
|
ctx.moveTo(0, h / 2 + metrics.descent);
|
||
|
ctx.lineTo(w, h / 2 + metrics.descent);
|
||
|
ctx.closePath();
|
||
|
ctx.stroke();
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* The new text metrics function
|
||
|
*/
|
||
|
CanvasRenderingContext2D.prototype.measureText2 = function (textstring, fontSize, fontString) {
|
||
|
var metrics = this.measureTextWidth(textstring),
|
||
|
isSpace = !/\S/.test(textstring);
|
||
|
metrics.fontsize = fontSize;
|
||
|
|
||
|
// for text lead values, we meaure a multiline text container.
|
||
|
var leadDiv = document.createElement("div");
|
||
|
leadDiv.style.position = "absolute";
|
||
|
leadDiv.style.opacity = 0;
|
||
|
leadDiv.style.font = fontString;
|
||
|
leadDiv.innerHTML = textstring + "<br/>" + textstring;
|
||
|
document.body.appendChild(leadDiv);
|
||
|
|
||
|
// make some initial guess at the text leading (using the standard TeX ratio)
|
||
|
metrics.leading = 1.2 * fontSize;
|
||
|
|
||
|
// then we try to get the real value from the browser
|
||
|
var leadDivHeight = getCSSValue(leadDiv, "height");
|
||
|
leadDivHeight = leadDivHeight.replace("px", "");
|
||
|
if (leadDivHeight >= fontSize * 2) {
|
||
|
metrics.leading = leadDivHeight / 2 | 0;
|
||
|
}
|
||
|
document.body.removeChild(leadDiv);
|
||
|
|
||
|
// if we're not dealing with white space, we can compute metrics
|
||
|
if (!isSpace) {
|
||
|
// Have characters, so measure the text
|
||
|
var canvas = document.createElement("canvas");
|
||
|
var padding = 100;
|
||
|
canvas.width = metrics.width + padding;
|
||
|
canvas.height = 3 * fontSize;
|
||
|
canvas.style.opacity = 1;
|
||
|
canvas.style.font = fontString;
|
||
|
var ctx = canvas.getContext("2d");
|
||
|
ctx.font = fontString;
|
||
|
|
||
|
var w = canvas.width,
|
||
|
h = canvas.height,
|
||
|
baseline = h / 2;
|
||
|
|
||
|
// Set all canvas pixeldata values to 255, with all the content
|
||
|
// data being 0. This lets us scan for data[i] != 255.
|
||
|
ctx.fillStyle = "white";
|
||
|
ctx.fillRect(-1, -1, w + 2, h + 2);
|
||
|
ctx.fillStyle = "black";
|
||
|
ctx.fillText(textstring, padding / 2, baseline);
|
||
|
var pixelData = ctx.getImageData(0, 0, w, h).data;
|
||
|
|
||
|
// canvas pixel data is w*4 by h*4, because R, G, B and A are separate,
|
||
|
// consecutive values in the array, rather than stored as 32 bit ints.
|
||
|
var i = 0,
|
||
|
w4 = w * 4,
|
||
|
len = pixelData.length;
|
||
|
|
||
|
// Finding the ascent uses a normal, forward scanline
|
||
|
while (++i < len && pixelData[i] === 255) {}
|
||
|
var ascent = i / w4 | 0;
|
||
|
|
||
|
// Finding the descent uses a reverse scanline
|
||
|
i = len - 1;
|
||
|
while (--i > 0 && pixelData[i] === 255) {}
|
||
|
var descent = i / w4 | 0;
|
||
|
|
||
|
// find the min-x coordinate
|
||
|
for (i = 0; i < len && pixelData[i] === 255;) {
|
||
|
i += w4;
|
||
|
if (i >= len) {
|
||
|
i = i - len + 4;
|
||
|
}
|
||
|
}
|
||
|
var minx = i % w4 / 4 | 0;
|
||
|
|
||
|
// find the max-x coordinate
|
||
|
var step = 1;
|
||
|
for (i = len - 3; i >= 0 && pixelData[i] === 255;) {
|
||
|
i -= w4;
|
||
|
if (i < 0) {
|
||
|
i = len - 3 - step++ * 4;
|
||
|
}
|
||
|
}
|
||
|
var maxx = i % w4 / 4 + 1 | 0;
|
||
|
|
||
|
// set font metrics
|
||
|
metrics.ascent = baseline - ascent;
|
||
|
metrics.descent = descent - baseline;
|
||
|
metrics.bounds = { minx: minx - padding / 2,
|
||
|
maxx: maxx - padding / 2,
|
||
|
miny: 0,
|
||
|
maxy: descent - ascent };
|
||
|
metrics.height = 1 + (descent - ascent);
|
||
|
}
|
||
|
|
||
|
// if we ARE dealing with whitespace, most values will just be zero.
|
||
|
else {
|
||
|
// Only whitespace, so we can't measure the text
|
||
|
metrics.ascent = 0;
|
||
|
metrics.descent = 0;
|
||
|
metrics.bounds = { minx: 0,
|
||
|
maxx: metrics.width, // Best guess
|
||
|
miny: 0,
|
||
|
maxy: 0 };
|
||
|
metrics.height = 0;
|
||
|
}
|
||
|
return metrics;
|
||
|
};
|
||
|
})();
|
||
|
|
||
|
},{}],12:[function(require,module,exports){
|
||
|
module.exports = {
|
||
|
arrow: (function() {
|
||
|
var getPoints;
|
||
|
getPoints = function(x, y, angle, width, length) {
|
||
|
return [
|
||
|
{
|
||
|
x: x + Math.cos(angle + Math.PI / 2) * width / 2,
|
||
|
y: y + Math.sin(angle + Math.PI / 2) * width / 2
|
||
|
}, {
|
||
|
x: x + Math.cos(angle) * length,
|
||
|
y: y + Math.sin(angle) * length
|
||
|
}, {
|
||
|
x: x + Math.cos(angle - Math.PI / 2) * width / 2,
|
||
|
y: y + Math.sin(angle - Math.PI / 2) * width / 2
|
||
|
}
|
||
|
];
|
||
|
};
|
||
|
return {
|
||
|
drawToCanvas: function(ctx, x, y, angle, width, color, length) {
|
||
|
var points;
|
||
|
if (length == null) {
|
||
|
length = 0;
|
||
|
}
|
||
|
length = length || width;
|
||
|
ctx.fillStyle = color;
|
||
|
ctx.lineWidth = 0;
|
||
|
ctx.strokeStyle = 'transparent';
|
||
|
ctx.beginPath();
|
||
|
points = getPoints(x, y, angle, width, length);
|
||
|
ctx.moveTo(points[0].x, points[0].y);
|
||
|
ctx.lineTo(points[1].x, points[1].y);
|
||
|
ctx.lineTo(points[2].x, points[2].y);
|
||
|
return ctx.fill();
|
||
|
},
|
||
|
svg: function(x, y, angle, width, color, length) {
|
||
|
var points;
|
||
|
if (length == null) {
|
||
|
length = 0;
|
||
|
}
|
||
|
length = length || width;
|
||
|
points = getPoints(x, y, angle, width, length);
|
||
|
return "<polygon fill='" + color + "' stroke='none' points='" + (points.map(function(p) {
|
||
|
return p.x + "," + p.y;
|
||
|
})) + "' />";
|
||
|
}
|
||
|
};
|
||
|
})()
|
||
|
};
|
||
|
|
||
|
|
||
|
},{}],13:[function(require,module,exports){
|
||
|
var _, localize, strings;
|
||
|
|
||
|
strings = {};
|
||
|
|
||
|
localize = function(localStrings) {
|
||
|
return strings = localStrings;
|
||
|
};
|
||
|
|
||
|
_ = function(string) {
|
||
|
var translation;
|
||
|
translation = strings[string];
|
||
|
return translation || string;
|
||
|
};
|
||
|
|
||
|
module.exports = {
|
||
|
localize: localize,
|
||
|
_: _
|
||
|
};
|
||
|
|
||
|
|
||
|
},{}],14:[function(require,module,exports){
|
||
|
var Point, _slope, math, normals, unit, util;
|
||
|
|
||
|
Point = require('./shapes').Point;
|
||
|
|
||
|
util = require('./util');
|
||
|
|
||
|
math = {};
|
||
|
|
||
|
math.toPoly = function(line) {
|
||
|
var i, index, len, n, point, polyLeft, polyRight;
|
||
|
polyLeft = [];
|
||
|
polyRight = [];
|
||
|
index = 0;
|
||
|
for (i = 0, len = line.length; i < len; i++) {
|
||
|
point = line[i];
|
||
|
n = normals(point, _slope(line, index));
|
||
|
polyLeft = polyLeft.concat([n[0]]);
|
||
|
polyRight = [n[1]].concat(polyRight);
|
||
|
index += 1;
|
||
|
}
|
||
|
return polyLeft.concat(polyRight);
|
||
|
};
|
||
|
|
||
|
_slope = function(line, index) {
|
||
|
var point;
|
||
|
if (line.length < 3) {
|
||
|
point = {
|
||
|
x: 0,
|
||
|
y: 0
|
||
|
};
|
||
|
}
|
||
|
if (index === 0) {
|
||
|
point = _slope(line, index + 1);
|
||
|
} else if (index === line.length - 1) {
|
||
|
point = _slope(line, index - 1);
|
||
|
} else {
|
||
|
point = math.diff(line[index - 1], line[index + 1]);
|
||
|
}
|
||
|
return point;
|
||
|
};
|
||
|
|
||
|
math.diff = function(a, b) {
|
||
|
return {
|
||
|
x: b.x - a.x,
|
||
|
y: b.y - a.y
|
||
|
};
|
||
|
};
|
||
|
|
||
|
unit = function(vector) {
|
||
|
var length;
|
||
|
length = math.len(vector);
|
||
|
return {
|
||
|
x: vector.x / length,
|
||
|
y: vector.y / length
|
||
|
};
|
||
|
};
|
||
|
|
||
|
normals = function(p, slope) {
|
||
|
slope = unit(slope);
|
||
|
slope.x = slope.x * p.size / 2;
|
||
|
slope.y = slope.y * p.size / 2;
|
||
|
return [
|
||
|
{
|
||
|
x: p.x - slope.y,
|
||
|
y: p.y + slope.x,
|
||
|
color: p.color
|
||
|
}, {
|
||
|
x: p.x + slope.y,
|
||
|
y: p.y - slope.x,
|
||
|
color: p.color
|
||
|
}
|
||
|
];
|
||
|
};
|
||
|
|
||
|
math.len = function(vector) {
|
||
|
return Math.sqrt(Math.pow(vector.x, 2) + Math.pow(vector.y, 2));
|
||
|
};
|
||
|
|
||
|
math.scalePositionScalar = function(val, viewportSize, oldScale, newScale) {
|
||
|
var newSize, oldSize;
|
||
|
oldSize = viewportSize * oldScale;
|
||
|
newSize = viewportSize * newScale;
|
||
|
return val + (oldSize - newSize) / 2;
|
||
|
};
|
||
|
|
||
|
module.exports = math;
|
||
|
|
||
|
|
||
|
},{"./shapes":17,"./util":19}],15:[function(require,module,exports){
|
||
|
var INFINITE, JSONToShape, renderWatermark, util;
|
||
|
|
||
|
util = require('./util');
|
||
|
|
||
|
JSONToShape = require('./shapes').JSONToShape;
|
||
|
|
||
|
INFINITE = 'infinite';
|
||
|
|
||
|
renderWatermark = function(ctx, image, scale) {
|
||
|
if (!image.width) {
|
||
|
return;
|
||
|
}
|
||
|
ctx.save();
|
||
|
ctx.translate(ctx.canvas.width / 2, ctx.canvas.height / 2);
|
||
|
ctx.scale(scale, scale);
|
||
|
ctx.drawImage(image, -image.width / 2, -image.height / 2);
|
||
|
return ctx.restore();
|
||
|
};
|
||
|
|
||
|
module.exports = function(snapshot, opts) {
|
||
|
var allShapes, backgroundShapes, colors, imageSize, s, shapes, watermarkCanvas, watermarkCtx;
|
||
|
if (opts == null) {
|
||
|
opts = {};
|
||
|
}
|
||
|
if (opts.scale == null) {
|
||
|
opts.scale = 1;
|
||
|
}
|
||
|
shapes = (function() {
|
||
|
var i, len, ref, results;
|
||
|
ref = snapshot.shapes;
|
||
|
results = [];
|
||
|
for (i = 0, len = ref.length; i < len; i++) {
|
||
|
s = ref[i];
|
||
|
results.push(JSONToShape(s));
|
||
|
}
|
||
|
return results;
|
||
|
})();
|
||
|
backgroundShapes = [];
|
||
|
if (snapshot.backgroundShapes) {
|
||
|
backgroundShapes = (function() {
|
||
|
var i, len, ref, results;
|
||
|
ref = snapshot.backgroundShapes;
|
||
|
results = [];
|
||
|
for (i = 0, len = ref.length; i < len; i++) {
|
||
|
s = ref[i];
|
||
|
results.push(JSONToShape(s));
|
||
|
}
|
||
|
return results;
|
||
|
})();
|
||
|
}
|
||
|
if (opts.margin == null) {
|
||
|
opts.margin = {
|
||
|
top: 0,
|
||
|
right: 0,
|
||
|
bottom: 0,
|
||
|
left: 0
|
||
|
};
|
||
|
}
|
||
|
imageSize = snapshot.imageSize || {
|
||
|
width: INFINITE,
|
||
|
height: INFINITE
|
||
|
};
|
||
|
colors = snapshot.colors || {
|
||
|
background: 'transparent'
|
||
|
};
|
||
|
allShapes = shapes.concat(backgroundShapes);
|
||
|
watermarkCanvas = document.createElement('canvas');
|
||
|
watermarkCtx = watermarkCanvas.getContext('2d');
|
||
|
if (opts.rect) {
|
||
|
opts.rect.x -= opts.margin.left;
|
||
|
opts.rect.y -= opts.margin.top;
|
||
|
opts.rect.width += opts.margin.left + opts.margin.right;
|
||
|
opts.rect.height += opts.margin.top + opts.margin.bottom;
|
||
|
} else {
|
||
|
opts.rect = util.getDefaultImageRect((function() {
|
||
|
var i, len, results;
|
||
|
results = [];
|
||
|
for (i = 0, len = allShapes.length; i < len; i++) {
|
||
|
s = allShapes[i];
|
||
|
results.push(s.getBoundingRect(watermarkCtx));
|
||
|
}
|
||
|
return results;
|
||
|
})(), imageSize, opts.margin);
|
||
|
}
|
||
|
watermarkCanvas.width = opts.rect.width * opts.scale;
|
||
|
watermarkCanvas.height = opts.rect.height * opts.scale;
|
||
|
watermarkCtx.fillStyle = colors.background;
|
||
|
watermarkCtx.fillRect(0, 0, watermarkCanvas.width, watermarkCanvas.height);
|
||
|
if (!(opts.rect.width && opts.rect.height)) {
|
||
|
return null;
|
||
|
}
|
||
|
if (opts.watermarkImage) {
|
||
|
renderWatermark(watermarkCtx, opts.watermarkImage, opts.watermarkScale);
|
||
|
}
|
||
|
return util.combineCanvases(watermarkCanvas, util.renderShapes(backgroundShapes, opts.rect, opts.scale), util.renderShapes(shapes, opts.rect, opts.scale));
|
||
|
};
|
||
|
|
||
|
|
||
|
},{"./shapes":17,"./util":19}],16:[function(require,module,exports){
|
||
|
var INFINITE, JSONToShape, util;
|
||
|
|
||
|
util = require('./util');
|
||
|
|
||
|
JSONToShape = require('./shapes').JSONToShape;
|
||
|
|
||
|
INFINITE = 'infinite';
|
||
|
|
||
|
module.exports = function(snapshot, opts) {
|
||
|
var allShapes, backgroundShapes, colors, ctx, dummyCanvas, imageSize, s, shapes;
|
||
|
if (opts == null) {
|
||
|
opts = {};
|
||
|
}
|
||
|
shapes = (function() {
|
||
|
var i, len, ref, results;
|
||
|
ref = snapshot.shapes;
|
||
|
results = [];
|
||
|
for (i = 0, len = ref.length; i < len; i++) {
|
||
|
s = ref[i];
|
||
|
results.push(JSONToShape(s));
|
||
|
}
|
||
|
return results;
|
||
|
})();
|
||
|
backgroundShapes = [];
|
||
|
if (snapshot.backgroundShapes) {
|
||
|
backgroundShapes = (function() {
|
||
|
var i, len, ref, results;
|
||
|
ref = snapshot.backgroundShapes;
|
||
|
results = [];
|
||
|
for (i = 0, len = ref.length; i < len; i++) {
|
||
|
s = ref[i];
|
||
|
results.push(JSONToShape(s));
|
||
|
}
|
||
|
return results;
|
||
|
})();
|
||
|
}
|
||
|
if (opts.margin == null) {
|
||
|
opts.margin = {
|
||
|
top: 0,
|
||
|
right: 0,
|
||
|
bottom: 0,
|
||
|
left: 0
|
||
|
};
|
||
|
}
|
||
|
imageSize = snapshot.imageSize || {
|
||
|
width: INFINITE,
|
||
|
height: INFINITE
|
||
|
};
|
||
|
colors = snapshot.colors || {
|
||
|
background: 'transparent'
|
||
|
};
|
||
|
allShapes = shapes.concat(backgroundShapes);
|
||
|
dummyCanvas = document.createElement('canvas');
|
||
|
ctx = dummyCanvas.getContext('2d');
|
||
|
if (opts.rect) {
|
||
|
opts.rect.x -= opts.margin.left;
|
||
|
opts.rect.y -= opts.margin.top;
|
||
|
opts.rect.width += opts.margin.left + opts.margin.right;
|
||
|
opts.rect.height += opts.margin.top + opts.margin.bottom;
|
||
|
} else {
|
||
|
opts.rect = util.getDefaultImageRect((function() {
|
||
|
var i, len, results;
|
||
|
results = [];
|
||
|
for (i = 0, len = allShapes.length; i < len; i++) {
|
||
|
s = allShapes[i];
|
||
|
results.push(s.getBoundingRect(ctx));
|
||
|
}
|
||
|
return results;
|
||
|
})(), imageSize, opts.margin);
|
||
|
}
|
||
|
return LC.renderShapesToSVG(backgroundShapes.concat(shapes), opts.rect, colors.background);
|
||
|
};
|
||
|
|
||
|
|
||
|
},{"./shapes":17,"./util":19}],17:[function(require,module,exports){
|
||
|
var JSONToShape, LinePath, TextRenderer, _createLinePathFromData, _doAllPointsShareStyle, _dual, _mid, _refine, bspline, createShape, defineCanvasRenderer, defineSVGRenderer, defineShape, lineEndCapShapes, linePathFuncs, ref, ref1, renderShapeToContext, renderShapeToSVG, shapeToJSON, shapes, util;
|
||
|
|
||
|
util = require('./util');
|
||
|
|
||
|
TextRenderer = require('./TextRenderer');
|
||
|
|
||
|
lineEndCapShapes = require('./lineEndCapShapes');
|
||
|
|
||
|
ref = require('./canvasRenderer'), defineCanvasRenderer = ref.defineCanvasRenderer, renderShapeToContext = ref.renderShapeToContext;
|
||
|
|
||
|
ref1 = require('./svgRenderer'), defineSVGRenderer = ref1.defineSVGRenderer, renderShapeToSVG = ref1.renderShapeToSVG;
|
||
|
|
||
|
shapes = {};
|
||
|
|
||
|
defineShape = function(name, props) {
|
||
|
var Shape, drawFunc, drawLatestFunc, k, legacyDrawFunc, legacyDrawLatestFunc, legacySVGFunc, svgFunc;
|
||
|
Shape = function(a, b, c, d, e, f, g, h, i, j, k, l, m, n, o, p) {
|
||
|
props.constructor.call(this, a, b, c, d, e, f, g, h, i, j, k, l, m, n, o, p);
|
||
|
return this;
|
||
|
};
|
||
|
Shape.prototype.className = name;
|
||
|
Shape.fromJSON = props.fromJSON;
|
||
|
if (props.draw) {
|
||
|
legacyDrawFunc = props.draw;
|
||
|
legacyDrawLatestFunc = props.draw || function(ctx, bufferCtx, retryCallback) {
|
||
|
return this.draw(ctx, bufferCtx, retryCallback);
|
||
|
};
|
||
|
drawFunc = function(ctx, shape, retryCallback) {
|
||
|
return legacyDrawFunc.call(shape, ctx, retryCallback);
|
||
|
};
|
||
|
drawLatestFunc = function(ctx, bufferCtx, shape, retryCallback) {
|
||
|
return legacyDrawLatestFunc.call(shape, ctx, bufferCtx, retryCallback);
|
||
|
};
|
||
|
delete props.draw;
|
||
|
if (props.drawLatest) {
|
||
|
delete props.drawLatest;
|
||
|
}
|
||
|
defineCanvasRenderer(name, drawFunc, drawLatestFunc);
|
||
|
}
|
||
|
if (props.toSVG) {
|
||
|
legacySVGFunc = props.toSVG;
|
||
|
svgFunc = function(shape) {
|
||
|
return legacySVGFunc.call(shape);
|
||
|
};
|
||
|
delete props.toSVG;
|
||
|
defineSVGRenderer(name, svgFunc);
|
||
|
}
|
||
|
Shape.prototype.draw = function(ctx, retryCallback) {
|
||
|
return renderShapeToContext(ctx, this, {
|
||
|
retryCallback: retryCallback
|
||
|
});
|
||
|
};
|
||
|
Shape.prototype.drawLatest = function(ctx, bufferCtx, retryCallback) {
|
||
|
return renderShapeToContext(ctx, this, {
|
||
|
retryCallback: retryCallback,
|
||
|
bufferCtx: bufferCtx,
|
||
|
shouldOnlyDrawLatest: true
|
||
|
});
|
||
|
};
|
||
|
Shape.prototype.toSVG = function() {
|
||
|
return renderShapeToSVG(this);
|
||
|
};
|
||
|
for (k in props) {
|
||
|
if (k !== 'fromJSON') {
|
||
|
Shape.prototype[k] = props[k];
|
||
|
}
|
||
|
}
|
||
|
shapes[name] = Shape;
|
||
|
return Shape;
|
||
|
};
|
||
|
|
||
|
createShape = function(name, a, b, c, d, e, f, g, h, i, j, k, l, m, n, o, p) {
|
||
|
var s;
|
||
|
s = new shapes[name](a, b, c, d, e, f, g, h, i, j, k, l, m, n, o, p);
|
||
|
s.id = util.getGUID();
|
||
|
return s;
|
||
|
};
|
||
|
|
||
|
JSONToShape = function(arg) {
|
||
|
var className, data, id, shape;
|
||
|
className = arg.className, data = arg.data, id = arg.id;
|
||
|
if (className in shapes) {
|
||
|
shape = shapes[className].fromJSON(data);
|
||
|
if (shape) {
|
||
|
if (id) {
|
||
|
shape.id = id;
|
||
|
}
|
||
|
return shape;
|
||
|
} else {
|
||
|
console.log('Unreadable shape:', className, data);
|
||
|
return null;
|
||
|
}
|
||
|
} else {
|
||
|
console.log("Unknown shape:", className, data);
|
||
|
return null;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
shapeToJSON = function(shape) {
|
||
|
return {
|
||
|
className: shape.className,
|
||
|
data: shape.toJSON(),
|
||
|
id: shape.id
|
||
|
};
|
||
|
};
|
||
|
|
||
|
bspline = function(points, order) {
|
||
|
if (!order) {
|
||
|
return points;
|
||
|
}
|
||
|
return bspline(_dual(_dual(_refine(points))), order - 1);
|
||
|
};
|
||
|
|
||
|
_refine = function(points) {
|
||
|
var index, len, point, q, refined;
|
||
|
points = [points[0]].concat(points).concat(util.last(points));
|
||
|
refined = [];
|
||
|
index = 0;
|
||
|
for (q = 0, len = points.length; q < len; q++) {
|
||
|
point = points[q];
|
||
|
refined[index * 2] = point;
|
||
|
if (points[index + 1]) {
|
||
|
refined[index * 2 + 1] = _mid(point, points[index + 1]);
|
||
|
}
|
||
|
index += 1;
|
||
|
}
|
||
|
return refined;
|
||
|
};
|
||
|
|
||
|
_dual = function(points) {
|
||
|
var dualed, index, len, point, q;
|
||
|
dualed = [];
|
||
|
index = 0;
|
||
|
for (q = 0, len = points.length; q < len; q++) {
|
||
|
point = points[q];
|
||
|
if (points[index + 1]) {
|
||
|
dualed[index] = _mid(point, points[index + 1]);
|
||
|
}
|
||
|
index += 1;
|
||
|
}
|
||
|
return dualed;
|
||
|
};
|
||
|
|
||
|
_mid = function(a, b) {
|
||
|
return createShape('Point', {
|
||
|
x: a.x + ((b.x - a.x) / 2),
|
||
|
y: a.y + ((b.y - a.y) / 2),
|
||
|
size: a.size + ((b.size - a.size) / 2),
|
||
|
color: a.color
|
||
|
});
|
||
|
};
|
||
|
|
||
|
defineShape('Image', {
|
||
|
constructor: function(args) {
|
||
|
if (args == null) {
|
||
|
args = {};
|
||
|
}
|
||
|
this.x = args.x || 0;
|
||
|
this.y = args.y || 0;
|
||
|
this.scale = args.scale || 1;
|
||
|
return this.image = args.image || null;
|
||
|
},
|
||
|
getBoundingRect: function() {
|
||
|
return {
|
||
|
x: this.x,
|
||
|
y: this.y,
|
||
|
width: this.image.width * this.scale,
|
||
|
height: this.image.height * this.scale
|
||
|
};
|
||
|
},
|
||
|
toJSON: function() {
|
||
|
return {
|
||
|
x: this.x,
|
||
|
y: this.y,
|
||
|
imageSrc: this.image.src,
|
||
|
imageObject: this.image,
|
||
|
scale: this.scale
|
||
|
};
|
||
|
},
|
||
|
fromJSON: function(data) {
|
||
|
var img, ref2;
|
||
|
img = null;
|
||
|
if ((ref2 = data.imageObject) != null ? ref2.width : void 0) {
|
||
|
img = data.imageObject;
|
||
|
} else {
|
||
|
img = new Image();
|
||
|
img.src = data.imageSrc;
|
||
|
}
|
||
|
return createShape('Image', {
|
||
|
x: data.x,
|
||
|
y: data.y,
|
||
|
image: img,
|
||
|
scale: data.scale
|
||
|
});
|
||
|
},
|
||
|
move: function(moveInfo) {
|
||
|
if (moveInfo == null) {
|
||
|
moveInfo = {};
|
||
|
}
|
||
|
this.x = this.x - moveInfo.xDiff;
|
||
|
return this.y = this.y - moveInfo.yDiff;
|
||
|
},
|
||
|
setUpperLeft: function(upperLeft) {
|
||
|
if (upperLeft == null) {
|
||
|
upperLeft = {};
|
||
|
}
|
||
|
this.x = upperLeft.x;
|
||
|
return this.y = upperLeft.y;
|
||
|
}
|
||
|
});
|
||
|
|
||
|
defineShape('Rectangle', {
|
||
|
constructor: function(args) {
|
||
|
if (args == null) {
|
||
|
args = {};
|
||
|
}
|
||
|
this.x = args.x || 0;
|
||
|
this.y = args.y || 0;
|
||
|
this.width = args.width || 0;
|
||
|
this.height = args.height || 0;
|
||
|
this.strokeWidth = args.strokeWidth || 1;
|
||
|
this.strokeColor = args.strokeColor || 'black';
|
||
|
return this.fillColor = args.fillColor || 'transparent';
|
||
|
},
|
||
|
getBoundingRect: function() {
|
||
|
return {
|
||
|
x: this.x - this.strokeWidth / 2,
|
||
|
y: this.y - this.strokeWidth / 2,
|
||
|
width: this.width + this.strokeWidth,
|
||
|
height: this.height + this.strokeWidth
|
||
|
};
|
||
|
},
|
||
|
toJSON: function() {
|
||
|
return {
|
||
|
x: this.x,
|
||
|
y: this.y,
|
||
|
width: this.width,
|
||
|
height: this.height,
|
||
|
strokeWidth: this.strokeWidth,
|
||
|
strokeColor: this.strokeColor,
|
||
|
fillColor: this.fillColor
|
||
|
};
|
||
|
},
|
||
|
fromJSON: function(data) {
|
||
|
return createShape('Rectangle', data);
|
||
|
},
|
||
|
move: function(moveInfo) {
|
||
|
if (moveInfo == null) {
|
||
|
moveInfo = {};
|
||
|
}
|
||
|
this.x = this.x - moveInfo.xDiff;
|
||
|
return this.y = this.y - moveInfo.yDiff;
|
||
|
},
|
||
|
setUpperLeft: function(upperLeft) {
|
||
|
if (upperLeft == null) {
|
||
|
upperLeft = {};
|
||
|
}
|
||
|
this.x = upperLeft.x;
|
||
|
return this.y = upperLeft.y;
|
||
|
}
|
||
|
});
|
||
|
|
||
|
defineShape('Ellipse', {
|
||
|
constructor: function(args) {
|
||
|
if (args == null) {
|
||
|
args = {};
|
||
|
}
|
||
|
this.x = args.x || 0;
|
||
|
this.y = args.y || 0;
|
||
|
this.width = args.width || 0;
|
||
|
this.height = args.height || 0;
|
||
|
this.strokeWidth = args.strokeWidth || 1;
|
||
|
this.strokeColor = args.strokeColor || 'black';
|
||
|
return this.fillColor = args.fillColor || 'transparent';
|
||
|
},
|
||
|
getBoundingRect: function() {
|
||
|
return {
|
||
|
x: this.x - this.strokeWidth / 2,
|
||
|
y: this.y - this.strokeWidth / 2,
|
||
|
width: this.width + this.strokeWidth,
|
||
|
height: this.height + this.strokeWidth
|
||
|
};
|
||
|
},
|
||
|
toJSON: function() {
|
||
|
return {
|
||
|
x: this.x,
|
||
|
y: this.y,
|
||
|
width: this.width,
|
||
|
height: this.height,
|
||
|
strokeWidth: this.strokeWidth,
|
||
|
strokeColor: this.strokeColor,
|
||
|
fillColor: this.fillColor
|
||
|
};
|
||
|
},
|
||
|
fromJSON: function(data) {
|
||
|
return createShape('Ellipse', data);
|
||
|
},
|
||
|
move: function(moveInfo) {
|
||
|
if (moveInfo == null) {
|
||
|
moveInfo = {};
|
||
|
}
|
||
|
this.x = this.x - moveInfo.xDiff;
|
||
|
return this.y = this.y - moveInfo.yDiff;
|
||
|
},
|
||
|
setUpperLeft: function(upperLeft) {
|
||
|
if (upperLeft == null) {
|
||
|
upperLeft = {};
|
||
|
}
|
||
|
this.x = upperLeft.x;
|
||
|
return this.y = upperLeft.y;
|
||
|
}
|
||
|
});
|
||
|
|
||
|
defineShape('Line', {
|
||
|
constructor: function(args) {
|
||
|
if (args == null) {
|
||
|
args = {};
|
||
|
}
|
||
|
this.x1 = args.x1 || 0;
|
||
|
this.y1 = args.y1 || 0;
|
||
|
this.x2 = args.x2 || 0;
|
||
|
this.y2 = args.y2 || 0;
|
||
|
this.strokeWidth = args.strokeWidth || 1;
|
||
|
this.color = args.color || 'black';
|
||
|
this.capStyle = args.capStyle || 'round';
|
||
|
this.endCapShapes = args.endCapShapes || [null, null];
|
||
|
return this.dash = args.dash || null;
|
||
|
},
|
||
|
getBoundingRect: function() {
|
||
|
return {
|
||
|
x: Math.min(this.x1, this.x2) - this.strokeWidth / 2,
|
||
|
y: Math.min(this.y1, this.y2) - this.strokeWidth / 2,
|
||
|
width: Math.abs(this.x2 - this.x1) + this.strokeWidth / 2,
|
||
|
height: Math.abs(this.y2 - this.y1) + this.strokeWidth / 2
|
||
|
};
|
||
|
},
|
||
|
toJSON: function() {
|
||
|
return {
|
||
|
x1: this.x1,
|
||
|
y1: this.y1,
|
||
|
x2: this.x2,
|
||
|
y2: this.y2,
|
||
|
strokeWidth: this.strokeWidth,
|
||
|
color: this.color,
|
||
|
capStyle: this.capStyle,
|
||
|
dash: this.dash,
|
||
|
endCapShapes: this.endCapShapes
|
||
|
};
|
||
|
},
|
||
|
fromJSON: function(data) {
|
||
|
return createShape('Line', data);
|
||
|
},
|
||
|
move: function(moveInfo) {
|
||
|
if (moveInfo == null) {
|
||
|
moveInfo = {};
|
||
|
}
|
||
|
this.x1 = this.x1 - moveInfo.xDiff;
|
||
|
this.y1 = this.y1 - moveInfo.yDiff;
|
||
|
this.x2 = this.x2 - moveInfo.xDiff;
|
||
|
return this.y2 = this.y2 - moveInfo.yDiff;
|
||
|
},
|
||
|
setUpperLeft: function(upperLeft) {
|
||
|
var br, xDiff, yDiff;
|
||
|
if (upperLeft == null) {
|
||
|
upperLeft = {};
|
||
|
}
|
||
|
br = this.getBoundingRect();
|
||
|
xDiff = br.x - upperLeft.x;
|
||
|
yDiff = br.y - upperLeft.y;
|
||
|
return this.move({
|
||
|
xDiff: xDiff,
|
||
|
yDiff: yDiff
|
||
|
});
|
||
|
}
|
||
|
});
|
||
|
|
||
|
_doAllPointsShareStyle = function(points) {
|
||
|
var color, len, point, q, size;
|
||
|
if (!points.length) {
|
||
|
return false;
|
||
|
}
|
||
|
size = points[0].size;
|
||
|
color = points[0].color;
|
||
|
for (q = 0, len = points.length; q < len; q++) {
|
||
|
point = points[q];
|
||
|
if (!(point.size === size && point.color === color)) {
|
||
|
console.log(size, color, point.size, point.color);
|
||
|
}
|
||
|
if (!(point.size === size && point.color === color)) {
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
return true;
|
||
|
};
|
||
|
|
||
|
_createLinePathFromData = function(shapeName, data) {
|
||
|
var pointData, points, smoothedPoints, x, y;
|
||
|
points = null;
|
||
|
if (data.points) {
|
||
|
points = (function() {
|
||
|
var len, q, ref2, results;
|
||
|
ref2 = data.points;
|
||
|
results = [];
|
||
|
for (q = 0, len = ref2.length; q < len; q++) {
|
||
|
pointData = ref2[q];
|
||
|
results.push(JSONToShape(pointData));
|
||
|
}
|
||
|
return results;
|
||
|
})();
|
||
|
} else if (data.pointCoordinatePairs) {
|
||
|
points = (function() {
|
||
|
var len, q, ref2, ref3, results;
|
||
|
ref2 = data.pointCoordinatePairs;
|
||
|
results = [];
|
||
|
for (q = 0, len = ref2.length; q < len; q++) {
|
||
|
ref3 = ref2[q], x = ref3[0], y = ref3[1];
|
||
|
results.push(JSONToShape({
|
||
|
className: 'Point',
|
||
|
data: {
|
||
|
x: x,
|
||
|
y: y,
|
||
|
size: data.pointSize,
|
||
|
color: data.pointColor,
|
||
|
smooth: data.smooth
|
||
|
}
|
||
|
}));
|
||
|
}
|
||
|
return results;
|
||
|
})();
|
||
|
}
|
||
|
smoothedPoints = null;
|
||
|
if (data.smoothedPointCoordinatePairs) {
|
||
|
smoothedPoints = (function() {
|
||
|
var len, q, ref2, ref3, results;
|
||
|
ref2 = data.smoothedPointCoordinatePairs;
|
||
|
results = [];
|
||
|
for (q = 0, len = ref2.length; q < len; q++) {
|
||
|
ref3 = ref2[q], x = ref3[0], y = ref3[1];
|
||
|
results.push(JSONToShape({
|
||
|
className: 'Point',
|
||
|
data: {
|
||
|
x: x,
|
||
|
y: y,
|
||
|
size: data.pointSize,
|
||
|
color: data.pointColor,
|
||
|
smooth: data.smooth
|
||
|
}
|
||
|
}));
|
||
|
}
|
||
|
return results;
|
||
|
})();
|
||
|
}
|
||
|
if (!points[0]) {
|
||
|
return null;
|
||
|
}
|
||
|
return createShape(shapeName, {
|
||
|
points: points,
|
||
|
smoothedPoints: smoothedPoints,
|
||
|
order: data.order,
|
||
|
tailSize: data.tailSize,
|
||
|
smooth: data.smooth
|
||
|
});
|
||
|
};
|
||
|
|
||
|
linePathFuncs = {
|
||
|
constructor: function(args) {
|
||
|
var len, point, points, q, results;
|
||
|
if (args == null) {
|
||
|
args = {};
|
||
|
}
|
||
|
points = args.points || [];
|
||
|
this.order = args.order || 3;
|
||
|
this.tailSize = args.tailSize || 3;
|
||
|
this.smooth = 'smooth' in args ? args.smooth : true;
|
||
|
this.segmentSize = Math.pow(2, this.order);
|
||
|
this.sampleSize = this.tailSize + 1;
|
||
|
if (args.smoothedPoints) {
|
||
|
this.points = args.points;
|
||
|
return this.smoothedPoints = args.smoothedPoints;
|
||
|
} else {
|
||
|
this.points = [];
|
||
|
results = [];
|
||
|
for (q = 0, len = points.length; q < len; q++) {
|
||
|
point = points[q];
|
||
|
results.push(this.addPoint(point));
|
||
|
}
|
||
|
return results;
|
||
|
}
|
||
|
},
|
||
|
getBoundingRect: function() {
|
||
|
return util.getBoundingRect(this.points.map(function(p) {
|
||
|
return {
|
||
|
x: p.x - p.size / 2,
|
||
|
y: p.y - p.size / 2,
|
||
|
width: p.size,
|
||
|
height: p.size
|
||
|
};
|
||
|
}));
|
||
|
},
|
||
|
toJSON: function() {
|
||
|
var p, point;
|
||
|
if (_doAllPointsShareStyle(this.points)) {
|
||
|
return {
|
||
|
order: this.order,
|
||
|
tailSize: this.tailSize,
|
||
|
smooth: this.smooth,
|
||
|
pointCoordinatePairs: (function() {
|
||
|
var len, q, ref2, results;
|
||
|
ref2 = this.points;
|
||
|
results = [];
|
||
|
for (q = 0, len = ref2.length; q < len; q++) {
|
||
|
point = ref2[q];
|
||
|
results.push([point.x, point.y]);
|
||
|
}
|
||
|
return results;
|
||
|
}).call(this),
|
||
|
smoothedPointCoordinatePairs: (function() {
|
||
|
var len, q, ref2, results;
|
||
|
ref2 = this.smoothedPoints;
|
||
|
results = [];
|
||
|
for (q = 0, len = ref2.length; q < len; q++) {
|
||
|
point = ref2[q];
|
||
|
results.push([point.x, point.y]);
|
||
|
}
|
||
|
return results;
|
||
|
}).call(this),
|
||
|
pointSize: this.points[0].size,
|
||
|
pointColor: this.points[0].color
|
||
|
};
|
||
|
} else {
|
||
|
return {
|
||
|
order: this.order,
|
||
|
tailSize: this.tailSize,
|
||
|
smooth: this.smooth,
|
||
|
points: (function() {
|
||
|
var len, q, ref2, results;
|
||
|
ref2 = this.points;
|
||
|
results = [];
|
||
|
for (q = 0, len = ref2.length; q < len; q++) {
|
||
|
p = ref2[q];
|
||
|
results.push(shapeToJSON(p));
|
||
|
}
|
||
|
return results;
|
||
|
}).call(this)
|
||
|
};
|
||
|
}
|
||
|
},
|
||
|
fromJSON: function(data) {
|
||
|
return _createLinePathFromData('LinePath', data);
|
||
|
},
|
||
|
addPoint: function(point) {
|
||
|
this.points.push(point);
|
||
|
if (!this.smooth) {
|
||
|
this.smoothedPoints = this.points;
|
||
|
return;
|
||
|
}
|
||
|
if (!this.smoothedPoints || this.points.length < this.sampleSize) {
|
||
|
return this.smoothedPoints = bspline(this.points, this.order);
|
||
|
} else {
|
||
|
this.tail = util.last(bspline(util.last(this.points, this.sampleSize), this.order), this.segmentSize * this.tailSize);
|
||
|
return this.smoothedPoints = this.smoothedPoints.slice(0, this.smoothedPoints.length - this.segmentSize * (this.tailSize - 1)).concat(this.tail);
|
||
|
}
|
||
|
},
|
||
|
move: function(moveInfo) {
|
||
|
var len, pt, pts, q;
|
||
|
if (moveInfo == null) {
|
||
|
moveInfo = {};
|
||
|
}
|
||
|
if (!this.smooth) {
|
||
|
pts = this.points;
|
||
|
} else {
|
||
|
pts = this.smoothedPoints;
|
||
|
}
|
||
|
for (q = 0, len = pts.length; q < len; q++) {
|
||
|
pt = pts[q];
|
||
|
pt.move(moveInfo);
|
||
|
}
|
||
|
return this.points = this.smoothedPoints;
|
||
|
},
|
||
|
setUpperLeft: function(upperLeft) {
|
||
|
var br, xDiff, yDiff;
|
||
|
if (upperLeft == null) {
|
||
|
upperLeft = {};
|
||
|
}
|
||
|
br = this.getBoundingRect();
|
||
|
xDiff = br.x - upperLeft.x;
|
||
|
yDiff = br.y - upperLeft.y;
|
||
|
return this.move({
|
||
|
xDiff: xDiff,
|
||
|
yDiff: yDiff
|
||
|
});
|
||
|
}
|
||
|
};
|
||
|
|
||
|
LinePath = defineShape('LinePath', linePathFuncs);
|
||
|
|
||
|
defineShape('ErasedLinePath', {
|
||
|
constructor: linePathFuncs.constructor,
|
||
|
toJSON: linePathFuncs.toJSON,
|
||
|
addPoint: linePathFuncs.addPoint,
|
||
|
getBoundingRect: linePathFuncs.getBoundingRect,
|
||
|
fromJSON: function(data) {
|
||
|
return _createLinePathFromData('ErasedLinePath', data);
|
||
|
}
|
||
|
});
|
||
|
|
||
|
defineShape('Point', {
|
||
|
constructor: function(args) {
|
||
|
if (args == null) {
|
||
|
args = {};
|
||
|
}
|
||
|
this.x = args.x || 0;
|
||
|
this.y = args.y || 0;
|
||
|
this.size = args.size || 0;
|
||
|
return this.color = args.color || '';
|
||
|
},
|
||
|
getBoundingRect: function() {
|
||
|
return {
|
||
|
x: this.x - this.size / 2,
|
||
|
y: this.y - this.size / 2,
|
||
|
width: this.size,
|
||
|
height: this.size
|
||
|
};
|
||
|
},
|
||
|
toJSON: function() {
|
||
|
return {
|
||
|
x: this.x,
|
||
|
y: this.y,
|
||
|
size: this.size,
|
||
|
color: this.color
|
||
|
};
|
||
|
},
|
||
|
fromJSON: function(data) {
|
||
|
return createShape('Point', data);
|
||
|
},
|
||
|
move: function(moveInfo) {
|
||
|
if (moveInfo == null) {
|
||
|
moveInfo = {};
|
||
|
}
|
||
|
this.x = this.x - moveInfo.xDiff;
|
||
|
return this.y = this.y - moveInfo.yDiff;
|
||
|
},
|
||
|
setUpperLeft: function(upperLeft) {
|
||
|
if (upperLeft == null) {
|
||
|
upperLeft = {};
|
||
|
}
|
||
|
this.x = upperLeft.x;
|
||
|
return this.y = upperLeft.y;
|
||
|
}
|
||
|
});
|
||
|
|
||
|
defineShape('Polygon', {
|
||
|
constructor: function(args) {
|
||
|
var len, point, q, ref2, results;
|
||
|
if (args == null) {
|
||
|
args = {};
|
||
|
}
|
||
|
this.points = args.points;
|
||
|
this.fillColor = args.fillColor || 'white';
|
||
|
this.strokeColor = args.strokeColor || 'black';
|
||
|
this.strokeWidth = args.strokeWidth;
|
||
|
this.dash = args.dash || null;
|
||
|
if (args.isClosed == null) {
|
||
|
args.isClosed = true;
|
||
|
}
|
||
|
this.isClosed = args.isClosed;
|
||
|
ref2 = this.points;
|
||
|
results = [];
|
||
|
for (q = 0, len = ref2.length; q < len; q++) {
|
||
|
point = ref2[q];
|
||
|
point.color = this.strokeColor;
|
||
|
results.push(point.size = this.strokeWidth);
|
||
|
}
|
||
|
return results;
|
||
|
},
|
||
|
addPoint: function(x, y) {
|
||
|
return this.points.push(LC.createShape('Point', {
|
||
|
x: x,
|
||
|
y: y
|
||
|
}));
|
||
|
},
|
||
|
getBoundingRect: function() {
|
||
|
return util.getBoundingRect(this.points.map(function(p) {
|
||
|
return p.getBoundingRect();
|
||
|
}));
|
||
|
},
|
||
|
toJSON: function() {
|
||
|
return {
|
||
|
strokeWidth: this.strokeWidth,
|
||
|
fillColor: this.fillColor,
|
||
|
strokeColor: this.strokeColor,
|
||
|
dash: this.dash,
|
||
|
isClosed: this.isClosed,
|
||
|
pointCoordinatePairs: this.points.map(function(p) {
|
||
|
return [p.x, p.y];
|
||
|
})
|
||
|
};
|
||
|
},
|
||
|
fromJSON: function(data) {
|
||
|
data.points = data.pointCoordinatePairs.map(function(arg) {
|
||
|
var x, y;
|
||
|
x = arg[0], y = arg[1];
|
||
|
return createShape('Point', {
|
||
|
x: x,
|
||
|
y: y,
|
||
|
size: data.strokeWidth,
|
||
|
color: data.strokeColor
|
||
|
});
|
||
|
});
|
||
|
return createShape('Polygon', data);
|
||
|
},
|
||
|
move: function(moveInfo) {
|
||
|
var len, pt, q, ref2, results;
|
||
|
if (moveInfo == null) {
|
||
|
moveInfo = {};
|
||
|
}
|
||
|
ref2 = this.points;
|
||
|
results = [];
|
||
|
for (q = 0, len = ref2.length; q < len; q++) {
|
||
|
pt = ref2[q];
|
||
|
results.push(pt.move(moveInfo));
|
||
|
}
|
||
|
return results;
|
||
|
},
|
||
|
setUpperLeft: function(upperLeft) {
|
||
|
var br, xDiff, yDiff;
|
||
|
if (upperLeft == null) {
|
||
|
upperLeft = {};
|
||
|
}
|
||
|
br = this.getBoundingRect();
|
||
|
xDiff = br.x - upperLeft.x;
|
||
|
yDiff = br.y - upperLeft.y;
|
||
|
return this.move({
|
||
|
xDiff: xDiff,
|
||
|
yDiff: yDiff
|
||
|
});
|
||
|
}
|
||
|
});
|
||
|
|
||
|
defineShape('Text', {
|
||
|
constructor: function(args) {
|
||
|
if (args == null) {
|
||
|
args = {};
|
||
|
}
|
||
|
this.x = args.x || 0;
|
||
|
this.y = args.y || 0;
|
||
|
this.v = args.v || 0;
|
||
|
this.text = args.text || '';
|
||
|
this.color = args.color || 'black';
|
||
|
this.font = args.font || '18px sans-serif';
|
||
|
this.forcedWidth = args.forcedWidth || null;
|
||
|
return this.forcedHeight = args.forcedHeight || null;
|
||
|
},
|
||
|
_makeRenderer: function(ctx) {
|
||
|
ctx.lineHeight = 1.2;
|
||
|
this.renderer = new TextRenderer(ctx, this.text, this.font, this.forcedWidth, this.forcedHeight);
|
||
|
if (this.v < 1) {
|
||
|
console.log('repairing baseline');
|
||
|
this.v = 1;
|
||
|
this.x -= this.renderer.metrics.bounds.minx;
|
||
|
return this.y -= this.renderer.metrics.leading - this.renderer.metrics.descent;
|
||
|
}
|
||
|
},
|
||
|
setText: function(text) {
|
||
|
this.text = text;
|
||
|
return this.renderer = null;
|
||
|
},
|
||
|
setFont: function(font) {
|
||
|
this.font = font;
|
||
|
return this.renderer = null;
|
||
|
},
|
||
|
setPosition: function(x, y) {
|
||
|
this.x = x;
|
||
|
return this.y = y;
|
||
|
},
|
||
|
setSize: function(forcedWidth, forcedHeight) {
|
||
|
this.forcedWidth = Math.max(forcedWidth, 0);
|
||
|
this.forcedHeight = Math.max(forcedHeight, 0);
|
||
|
return this.renderer = null;
|
||
|
},
|
||
|
enforceMaxBoundingRect: function(lc) {
|
||
|
var br, dx, lcBoundingRect;
|
||
|
br = this.getBoundingRect(lc.ctx);
|
||
|
lcBoundingRect = {
|
||
|
x: -lc.position.x / lc.scale,
|
||
|
y: -lc.position.y / lc.scale,
|
||
|
width: lc.canvas.width / lc.scale,
|
||
|
height: lc.canvas.height / lc.scale
|
||
|
};
|
||
|
if (br.x + br.width > lcBoundingRect.x + lcBoundingRect.width) {
|
||
|
dx = br.x - lcBoundingRect.x;
|
||
|
this.forcedWidth = lcBoundingRect.width - dx - 10;
|
||
|
return this.renderer = null;
|
||
|
}
|
||
|
},
|
||
|
getBoundingRect: function(ctx, isEditing) {
|
||
|
if (isEditing == null) {
|
||
|
isEditing = false;
|
||
|
}
|
||
|
if (!this.renderer) {
|
||
|
if (ctx) {
|
||
|
this._makeRenderer(ctx);
|
||
|
} else {
|
||
|
throw "Must pass ctx if text hasn't been rendered yet";
|
||
|
}
|
||
|
}
|
||
|
return {
|
||
|
x: Math.floor(this.x),
|
||
|
y: Math.floor(this.y),
|
||
|
width: Math.ceil(this.renderer.getWidth(true)),
|
||
|
height: Math.ceil(this.renderer.getHeight())
|
||
|
};
|
||
|
},
|
||
|
toJSON: function() {
|
||
|
return {
|
||
|
x: this.x,
|
||
|
y: this.y,
|
||
|
text: this.text,
|
||
|
color: this.color,
|
||
|
font: this.font,
|
||
|
forcedWidth: this.forcedWidth,
|
||
|
forcedHeight: this.forcedHeight,
|
||
|
v: this.v
|
||
|
};
|
||
|
},
|
||
|
fromJSON: function(data) {
|
||
|
return createShape('Text', data);
|
||
|
},
|
||
|
move: function(moveInfo) {
|
||
|
if (moveInfo == null) {
|
||
|
moveInfo = {};
|
||
|
}
|
||
|
this.x = this.x - moveInfo.xDiff;
|
||
|
return this.y = this.y - moveInfo.yDiff;
|
||
|
},
|
||
|
setUpperLeft: function(upperLeft) {
|
||
|
if (upperLeft == null) {
|
||
|
upperLeft = {};
|
||
|
}
|
||
|
this.x = upperLeft.x;
|
||
|
return this.y = upperLeft.y;
|
||
|
}
|
||
|
});
|
||
|
|
||
|
defineShape('SelectionBox', {
|
||
|
constructor: function(args) {
|
||
|
if (args == null) {
|
||
|
args = {};
|
||
|
}
|
||
|
this.shape = args.shape;
|
||
|
if (args.handleSize != null) {
|
||
|
this.handleSize = args.handleSize;
|
||
|
} else {
|
||
|
this.handleSize = 10;
|
||
|
}
|
||
|
this.margin = 4;
|
||
|
this.backgroundColor = args.backgroundColor || null;
|
||
|
return this._br = this.shape.getBoundingRect(args.ctx);
|
||
|
},
|
||
|
toJSON: function() {
|
||
|
return {
|
||
|
shape: shapeToJSON(this.shape),
|
||
|
backgroundColor: this.backgroundColor
|
||
|
};
|
||
|
},
|
||
|
fromJSON: function(arg) {
|
||
|
var backgroundColor, handleSize, margin, shape;
|
||
|
shape = arg.shape, handleSize = arg.handleSize, margin = arg.margin, backgroundColor = arg.backgroundColor;
|
||
|
return createShape('SelectionBox', {
|
||
|
shape: JSONToShape(shape),
|
||
|
backgroundColor: backgroundColor
|
||
|
});
|
||
|
},
|
||
|
getTopLeftHandleRect: function() {
|
||
|
return {
|
||
|
x: this._br.x - this.handleSize - this.margin,
|
||
|
y: this._br.y - this.handleSize - this.margin,
|
||
|
width: this.handleSize,
|
||
|
height: this.handleSize
|
||
|
};
|
||
|
},
|
||
|
getBottomLeftHandleRect: function() {
|
||
|
return {
|
||
|
x: this._br.x - this.handleSize - this.margin,
|
||
|
y: this._br.y + this._br.height + this.margin,
|
||
|
width: this.handleSize,
|
||
|
height: this.handleSize
|
||
|
};
|
||
|
},
|
||
|
getTopRightHandleRect: function() {
|
||
|
return {
|
||
|
x: this._br.x + this._br.width + this.margin,
|
||
|
y: this._br.y - this.handleSize - this.margin,
|
||
|
width: this.handleSize,
|
||
|
height: this.handleSize
|
||
|
};
|
||
|
},
|
||
|
getBottomRightHandleRect: function() {
|
||
|
return {
|
||
|
x: this._br.x + this._br.width + this.margin,
|
||
|
y: this._br.y + this._br.height + this.margin,
|
||
|
width: this.handleSize,
|
||
|
height: this.handleSize
|
||
|
};
|
||
|
},
|
||
|
getBoundingRect: function() {
|
||
|
return {
|
||
|
x: this._br.x - this.margin,
|
||
|
y: this._br.y - this.margin,
|
||
|
width: this._br.width + this.margin * 2,
|
||
|
height: this._br.height + this.margin * 2
|
||
|
};
|
||
|
}
|
||
|
});
|
||
|
|
||
|
module.exports = {
|
||
|
defineShape: defineShape,
|
||
|
createShape: createShape,
|
||
|
JSONToShape: JSONToShape,
|
||
|
shapeToJSON: shapeToJSON
|
||
|
};
|
||
|
|
||
|
|
||
|
},{"./TextRenderer":6,"./canvasRenderer":9,"./lineEndCapShapes":12,"./svgRenderer":18,"./util":19}],18:[function(require,module,exports){
|
||
|
var defineSVGRenderer, lineEndCapShapes, renderShapeToSVG, renderers;
|
||
|
|
||
|
lineEndCapShapes = require('./lineEndCapShapes');
|
||
|
|
||
|
renderers = {};
|
||
|
|
||
|
defineSVGRenderer = function(shapeName, shapeToSVGFunc) {
|
||
|
return renderers[shapeName] = shapeToSVGFunc;
|
||
|
};
|
||
|
|
||
|
renderShapeToSVG = function(shape, opts) {
|
||
|
if (opts == null) {
|
||
|
opts = {};
|
||
|
}
|
||
|
if (opts.shouldIgnoreUnsupportedShapes == null) {
|
||
|
opts.shouldIgnoreUnsupportedShapes = false;
|
||
|
}
|
||
|
if (renderers[shape.className]) {
|
||
|
return renderers[shape.className](shape);
|
||
|
} else if (opts.shouldIgnoreUnsupportedShapes) {
|
||
|
console.warn("Can't render shape of type " + shape.className + " to SVG");
|
||
|
return "";
|
||
|
} else {
|
||
|
throw "Can't render shape of type " + shape.className + " to SVG";
|
||
|
}
|
||
|
};
|
||
|
|
||
|
defineSVGRenderer('Rectangle', function(shape) {
|
||
|
var height, width, x, x1, x2, y, y1, y2;
|
||
|
x1 = shape.x;
|
||
|
y1 = shape.y;
|
||
|
x2 = shape.x + shape.width;
|
||
|
y2 = shape.y + shape.height;
|
||
|
x = Math.min(x1, x2);
|
||
|
y = Math.min(y1, y2);
|
||
|
width = Math.max(x1, x2) - x;
|
||
|
height = Math.max(y1, y2) - y;
|
||
|
if (shape.strokeWidth % 2 !== 0) {
|
||
|
x += 0.5;
|
||
|
y += 0.5;
|
||
|
}
|
||
|
return "<rect x='" + x + "' y='" + y + "' width='" + width + "' height='" + height + "' stroke='" + shape.strokeColor + "' fill='" + shape.fillColor + "' stroke-width='" + shape.strokeWidth + "' />";
|
||
|
});
|
||
|
|
||
|
defineSVGRenderer('SelectionBox', function(shape) {
|
||
|
return "";
|
||
|
});
|
||
|
|
||
|
defineSVGRenderer('Ellipse', function(shape) {
|
||
|
var centerX, centerY, halfHeight, halfWidth;
|
||
|
halfWidth = Math.floor(shape.width / 2);
|
||
|
halfHeight = Math.floor(shape.height / 2);
|
||
|
centerX = shape.x + halfWidth;
|
||
|
centerY = shape.y + halfHeight;
|
||
|
return "<ellipse cx='" + centerX + "' cy='" + centerY + "' rx='" + (Math.abs(halfWidth)) + "' ry='" + (Math.abs(halfHeight)) + "' stroke='" + shape.strokeColor + "' fill='" + shape.fillColor + "' stroke-width='" + shape.strokeWidth + "' />";
|
||
|
});
|
||
|
|
||
|
defineSVGRenderer('Image', function(shape) {
|
||
|
return "<image x='" + shape.x + "' y='" + shape.y + "' width='" + (shape.image.naturalWidth * shape.scale) + "' height='" + (shape.image.naturalHeight * shape.scale) + "' xlink:href='" + shape.image.src + "' />";
|
||
|
});
|
||
|
|
||
|
defineSVGRenderer('Line', function(shape) {
|
||
|
var arrowWidth, capString, dashString, x1, x2, y1, y2;
|
||
|
dashString = shape.dash ? "stroke-dasharray='" + (shape.dash.join(', ')) + "'" : '';
|
||
|
capString = '';
|
||
|
arrowWidth = Math.max(shape.strokeWidth * 2.2, 5);
|
||
|
x1 = shape.x1;
|
||
|
x2 = shape.x2;
|
||
|
y1 = shape.y1;
|
||
|
y2 = shape.y2;
|
||
|
if (shape.strokeWidth % 2 !== 0) {
|
||
|
x1 += 0.5;
|
||
|
x2 += 0.5;
|
||
|
y1 += 0.5;
|
||
|
y2 += 0.5;
|
||
|
}
|
||
|
if (shape.endCapShapes[0]) {
|
||
|
capString += lineEndCapShapes[shape.endCapShapes[0]].svg(x1, y1, Math.atan2(y1 - y2, x1 - x2), arrowWidth, shape.color);
|
||
|
}
|
||
|
if (shape.endCapShapes[1]) {
|
||
|
capString += lineEndCapShapes[shape.endCapShapes[1]].svg(x2, y2, Math.atan2(y2 - y1, x2 - x1), arrowWidth, shape.color);
|
||
|
}
|
||
|
return "<g> <line x1='" + x1 + "' y1='" + y1 + "' x2='" + x2 + "' y2='" + y2 + "' " + dashString + " stroke-linecap='" + shape.capStyle + "' stroke='" + shape.color + " 'stroke-width='" + shape.strokeWidth + "' /> " + capString + " </g>";
|
||
|
});
|
||
|
|
||
|
defineSVGRenderer('LinePath', function(shape) {
|
||
|
return "<polyline fill='none' points='" + (shape.smoothedPoints.map(function(p) {
|
||
|
var offset;
|
||
|
offset = p.strokeWidth % 2 === 0 ? 0.0 : 0.5;
|
||
|
return (p.x + offset) + "," + (p.y + offset);
|
||
|
}).join(' ')) + "' stroke='" + shape.points[0].color + "' stroke-linecap='round' stroke-width='" + shape.points[0].size + "' />";
|
||
|
});
|
||
|
|
||
|
defineSVGRenderer('ErasedLinePath', function(shape) {
|
||
|
return "";
|
||
|
});
|
||
|
|
||
|
defineSVGRenderer('Polygon', function(shape) {
|
||
|
if (shape.isClosed) {
|
||
|
return "<polygon fill='" + shape.fillColor + "' points='" + (shape.points.map(function(p) {
|
||
|
var offset;
|
||
|
offset = p.strokeWidth % 2 === 0 ? 0.0 : 0.5;
|
||
|
return (p.x + offset) + "," + (p.y + offset);
|
||
|
}).join(' ')) + "' stroke='" + shape.strokeColor + "' stroke-width='" + shape.strokeWidth + "' />";
|
||
|
} else {
|
||
|
return "<polyline fill='" + shape.fillColor + "' points='" + (shape.points.map(function(p) {
|
||
|
var offset;
|
||
|
offset = p.strokeWidth % 2 === 0 ? 0.0 : 0.5;
|
||
|
return (p.x + offset) + "," + (p.y + offset);
|
||
|
}).join(' ')) + "' stroke='none' /> <polyline fill='none' points='" + (shape.points.map(function(p) {
|
||
|
var offset;
|
||
|
offset = p.strokeWidth % 2 === 0 ? 0.0 : 0.5;
|
||
|
return (p.x + offset) + "," + (p.y + offset);
|
||
|
}).join(' ')) + "' stroke='" + shape.strokeColor + "' stroke-width='" + shape.strokeWidth + "' />";
|
||
|
}
|
||
|
});
|
||
|
|
||
|
defineSVGRenderer('Text', function(shape) {
|
||
|
var heightString, textSplitOnLines, widthString;
|
||
|
widthString = shape.forcedWidth ? "width='" + shape.forcedWidth + "px'" : "";
|
||
|
heightString = shape.forcedHeight ? "height='" + shape.forcedHeight + "px'" : "";
|
||
|
textSplitOnLines = shape.text.split(/\r\n|\r|\n/g);
|
||
|
if (shape.renderer) {
|
||
|
textSplitOnLines = shape.renderer.lines;
|
||
|
}
|
||
|
return "<text x='" + shape.x + "' y='" + shape.y + "' " + widthString + " " + heightString + " fill='" + shape.color + "' style='font: " + shape.font + ";'> " + (textSplitOnLines.map((function(_this) {
|
||
|
return function(line, i) {
|
||
|
var dy;
|
||
|
dy = i === 0 ? 0 : '1.2em';
|
||
|
return "<tspan x='" + shape.x + "' dy='" + dy + "' alignment-baseline='text-before-edge'> " + line + " </tspan>";
|
||
|
};
|
||
|
})(this)).join('')) + " </text>";
|
||
|
});
|
||
|
|
||
|
module.exports = {
|
||
|
defineSVGRenderer: defineSVGRenderer,
|
||
|
renderShapeToSVG: renderShapeToSVG
|
||
|
};
|
||
|
|
||
|
|
||
|
},{"./lineEndCapShapes":12}],19:[function(require,module,exports){
|
||
|
var renderShapeToContext, renderShapeToSVG, slice, util,
|
||
|
slice1 = [].slice;
|
||
|
|
||
|
slice = Array.prototype.slice;
|
||
|
|
||
|
renderShapeToContext = require('./canvasRenderer').renderShapeToContext;
|
||
|
|
||
|
renderShapeToSVG = require('./svgRenderer').renderShapeToSVG;
|
||
|
|
||
|
util = {
|
||
|
addImageOnload: function(img, fn) {
|
||
|
var oldOnload;
|
||
|
oldOnload = img.onload;
|
||
|
img.onload = function() {
|
||
|
if (typeof oldOnload === "function") {
|
||
|
oldOnload();
|
||
|
}
|
||
|
return fn();
|
||
|
};
|
||
|
return img;
|
||
|
},
|
||
|
last: function(array, n) {
|
||
|
if (n == null) {
|
||
|
n = null;
|
||
|
}
|
||
|
if (n) {
|
||
|
return slice.call(array, Math.max(array.length - n, 0));
|
||
|
} else {
|
||
|
return array[array.length - 1];
|
||
|
}
|
||
|
},
|
||
|
classSet: function(classNameToIsPresent) {
|
||
|
var classNames, key;
|
||
|
classNames = [];
|
||
|
for (key in classNameToIsPresent) {
|
||
|
if (classNameToIsPresent[key]) {
|
||
|
classNames.push(key);
|
||
|
}
|
||
|
}
|
||
|
return classNames.join(' ');
|
||
|
},
|
||
|
matchElementSize: function(elementToMatch, elementsToResize, scale, callback) {
|
||
|
var resize;
|
||
|
if (callback == null) {
|
||
|
callback = function() {};
|
||
|
}
|
||
|
resize = (function(_this) {
|
||
|
return function() {
|
||
|
var el, i, len;
|
||
|
for (i = 0, len = elementsToResize.length; i < len; i++) {
|
||
|
el = elementsToResize[i];
|
||
|
el.style.width = elementToMatch.offsetWidth + "px";
|
||
|
el.style.height = elementToMatch.offsetHeight + "px";
|
||
|
if (el.width != null) {
|
||
|
el.setAttribute('width', el.offsetWidth * scale);
|
||
|
el.setAttribute('height', el.offsetHeight * scale);
|
||
|
}
|
||
|
}
|
||
|
return callback();
|
||
|
};
|
||
|
})(this);
|
||
|
elementToMatch.addEventListener('resize', resize);
|
||
|
window.addEventListener('resize', resize);
|
||
|
window.addEventListener('orientationchange', resize);
|
||
|
resize();
|
||
|
return resize;
|
||
|
},
|
||
|
combineCanvases: function() {
|
||
|
var c, canvas, canvases, ctx, i, j, len, len1;
|
||
|
canvases = 1 <= arguments.length ? slice1.call(arguments, 0) : [];
|
||
|
c = document.createElement('canvas');
|
||
|
c.width = canvases[0].width;
|
||
|
c.height = canvases[0].height;
|
||
|
for (i = 0, len = canvases.length; i < len; i++) {
|
||
|
canvas = canvases[i];
|
||
|
c.width = Math.max(canvas.width, c.width);
|
||
|
c.height = Math.max(canvas.height, c.height);
|
||
|
}
|
||
|
ctx = c.getContext('2d');
|
||
|
for (j = 0, len1 = canvases.length; j < len1; j++) {
|
||
|
canvas = canvases[j];
|
||
|
ctx.drawImage(canvas, 0, 0);
|
||
|
}
|
||
|
return c;
|
||
|
},
|
||
|
renderShapes: function(shapes, bounds, scale, canvas) {
|
||
|
var ctx, i, len, shape;
|
||
|
if (scale == null) {
|
||
|
scale = 1;
|
||
|
}
|
||
|
if (canvas == null) {
|
||
|
canvas = null;
|
||
|
}
|
||
|
canvas = canvas || document.createElement('canvas');
|
||
|
canvas.width = bounds.width * scale;
|
||
|
canvas.height = bounds.height * scale;
|
||
|
ctx = canvas.getContext('2d');
|
||
|
ctx.translate(-bounds.x * scale, -bounds.y * scale);
|
||
|
ctx.scale(scale, scale);
|
||
|
for (i = 0, len = shapes.length; i < len; i++) {
|
||
|
shape = shapes[i];
|
||
|
renderShapeToContext(ctx, shape);
|
||
|
}
|
||
|
return canvas;
|
||
|
},
|
||
|
renderShapesToSVG: function(shapes, arg, backgroundColor) {
|
||
|
var height, width, x, y;
|
||
|
x = arg.x, y = arg.y, width = arg.width, height = arg.height;
|
||
|
return ("<svg xmlns='http://www.w3.org/2000/svg' width='" + width + "' height='" + height + "' viewBox='0 0 " + width + " " + height + "'> <rect width='" + width + "' height='" + height + "' x='0' y='0' fill='" + backgroundColor + "' /> <g transform='translate(" + (-x) + ", " + (-y) + ")'> " + (shapes.map(renderShapeToSVG).join('')) + " </g> </svg>").replace(/(\r\n|\n|\r)/gm, "");
|
||
|
},
|
||
|
getBoundingRect: function(rects, width, height) {
|
||
|
var i, len, maxX, maxY, minX, minY, rect;
|
||
|
if (!rects.length) {
|
||
|
return {
|
||
|
x: 0,
|
||
|
y: 0,
|
||
|
width: 0 || width,
|
||
|
height: 0 || height
|
||
|
};
|
||
|
}
|
||
|
minX = rects[0].x;
|
||
|
minY = rects[0].y;
|
||
|
maxX = rects[0].x + rects[0].width;
|
||
|
maxY = rects[0].y + rects[0].height;
|
||
|
for (i = 0, len = rects.length; i < len; i++) {
|
||
|
rect = rects[i];
|
||
|
minX = Math.floor(Math.min(rect.x, minX));
|
||
|
minY = Math.floor(Math.min(rect.y, minY));
|
||
|
maxX = Math.ceil(Math.max(maxX, rect.x + rect.width));
|
||
|
maxY = Math.ceil(Math.max(maxY, rect.y + rect.height));
|
||
|
}
|
||
|
minX = width ? 0 : minX;
|
||
|
minY = height ? 0 : minY;
|
||
|
maxX = width || maxX;
|
||
|
maxY = height || maxY;
|
||
|
return {
|
||
|
x: minX,
|
||
|
y: minY,
|
||
|
width: maxX - minX,
|
||
|
height: maxY - minY
|
||
|
};
|
||
|
},
|
||
|
getDefaultImageRect: function(shapeBoundingRects, explicitSize, margin) {
|
||
|
var height, rect, width;
|
||
|
if (explicitSize == null) {
|
||
|
explicitSize = {
|
||
|
width: 0,
|
||
|
height: 0
|
||
|
};
|
||
|
}
|
||
|
if (margin == null) {
|
||
|
margin = {
|
||
|
top: 0,
|
||
|
right: 0,
|
||
|
bottom: 0,
|
||
|
left: 0
|
||
|
};
|
||
|
}
|
||
|
width = explicitSize.width, height = explicitSize.height;
|
||
|
rect = util.getBoundingRect(shapeBoundingRects, width === 'infinite' ? 0 : width, height === 'infinite' ? 0 : height);
|
||
|
rect.x -= margin.left;
|
||
|
rect.y -= margin.top;
|
||
|
rect.width += margin.left + margin.right;
|
||
|
rect.height += margin.top + margin.bottom;
|
||
|
return rect;
|
||
|
},
|
||
|
getBackingScale: function(context) {
|
||
|
if (window.devicePixelRatio == null) {
|
||
|
return 1;
|
||
|
}
|
||
|
if (!(window.devicePixelRatio > 1)) {
|
||
|
return 1;
|
||
|
}
|
||
|
return window.devicePixelRatio;
|
||
|
},
|
||
|
requestAnimationFrame: (window.requestAnimationFrame || window.setTimeout).bind(window),
|
||
|
getGUID: (function() {
|
||
|
var s4;
|
||
|
s4 = function() {
|
||
|
return Math.floor((1 + Math.random()) * 0x10000).toString(16).substring(1);
|
||
|
};
|
||
|
return function() {
|
||
|
return s4() + s4() + '-' + s4() + '-' + s4() + '-' + s4() + '-' + s4() + s4() + s4();
|
||
|
};
|
||
|
})(),
|
||
|
requestAnimationFrame: function(f) {
|
||
|
if (window.requestAnimationFrame) {
|
||
|
return window.requestAnimationFrame(f);
|
||
|
}
|
||
|
if (window.webkitRequestAnimationFrame) {
|
||
|
return window.webkitRequestAnimationFrame(f);
|
||
|
}
|
||
|
if (window.mozRequestAnimationFrame) {
|
||
|
return window.mozRequestAnimationFrame(f);
|
||
|
}
|
||
|
return setTimeout(f, 0);
|
||
|
},
|
||
|
cancelAnimationFrame: function(f) {
|
||
|
if (window.cancelAnimationFrame) {
|
||
|
return window.cancelAnimationFrame(f);
|
||
|
}
|
||
|
if (window.webkitCancelRequestAnimationFrame) {
|
||
|
return window.webkitCancelRequestAnimationFrame(f);
|
||
|
}
|
||
|
if (window.webkitCancelAnimationFrame) {
|
||
|
return window.webkitCancelAnimationFrame(f);
|
||
|
}
|
||
|
if (window.mozCancelAnimationFrame) {
|
||
|
return window.mozCancelAnimationFrame(f);
|
||
|
}
|
||
|
return clearTimeout(f);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
module.exports = util;
|
||
|
|
||
|
|
||
|
},{"./canvasRenderer":9,"./svgRenderer":18}],20:[function(require,module,exports){
|
||
|
'use strict';
|
||
|
|
||
|
(function () {
|
||
|
function CustomEvent(event, params) {
|
||
|
params = params || { bubbles: false, cancelable: false, detail: undefined };
|
||
|
var evt = document.createEvent('CustomEvent');
|
||
|
evt.initCustomEvent(event, params.bubbles, params.cancelable, params.detail);
|
||
|
return evt;
|
||
|
};
|
||
|
|
||
|
CustomEvent.prototype = window.CustomEvent.prototype;
|
||
|
|
||
|
window.CustomEvent = CustomEvent;
|
||
|
})();
|
||
|
|
||
|
},{}],21:[function(require,module,exports){
|
||
|
"use strict";
|
||
|
|
||
|
var hasWarned = false;
|
||
|
if (!CanvasRenderingContext2D.prototype.setLineDash) {
|
||
|
CanvasRenderingContext2D.prototype.setLineDash = function () {
|
||
|
// no-op
|
||
|
if (!hasWarned) {
|
||
|
console.warn("context2D.setLineDash is a no-op in this browser.");
|
||
|
hasWarned = true;
|
||
|
}
|
||
|
};
|
||
|
}
|
||
|
module.exports = null;
|
||
|
|
||
|
},{}],22:[function(require,module,exports){
|
||
|
var LiterallyCanvasModel, LiterallyCanvasReactComponent, baseTools, canvasRenderer, conversion, defaultImageURLPrefix, defaultOptions, defaultTools, defineOptionsStyle, init, initReactDOM, initWithoutGUI, localize, registerJQueryPlugin, renderSnapshotToImage, renderSnapshotToSVG, setDefaultImageURLPrefix, shapes, svgRenderer, tools, util;
|
||
|
|
||
|
require('./ie_customevent');
|
||
|
|
||
|
require('./ie_setLineDash');
|
||
|
|
||
|
LiterallyCanvasModel = require('./core/LiterallyCanvas');
|
||
|
|
||
|
defaultOptions = require('./core/defaultOptions');
|
||
|
|
||
|
canvasRenderer = require('./core/canvasRenderer');
|
||
|
|
||
|
svgRenderer = require('./core/svgRenderer');
|
||
|
|
||
|
shapes = require('./core/shapes');
|
||
|
|
||
|
util = require('./core/util');
|
||
|
|
||
|
renderSnapshotToImage = require('./core/renderSnapshotToImage');
|
||
|
|
||
|
renderSnapshotToSVG = require('./core/renderSnapshotToSVG');
|
||
|
|
||
|
localize = require('./core/localization').localize;
|
||
|
|
||
|
LiterallyCanvasReactComponent = require('./reactGUI/LiterallyCanvas');
|
||
|
|
||
|
initReactDOM = require('./reactGUI/initDOM');
|
||
|
|
||
|
require('./optionsStyles/font');
|
||
|
|
||
|
require('./optionsStyles/stroke-width');
|
||
|
|
||
|
require('./optionsStyles/line-options-and-stroke-width');
|
||
|
|
||
|
require('./optionsStyles/polygon-and-stroke-width');
|
||
|
|
||
|
require('./optionsStyles/stroke-or-fill');
|
||
|
|
||
|
require('./optionsStyles/null');
|
||
|
|
||
|
defineOptionsStyle = require('./optionsStyles/optionsStyles').defineOptionsStyle;
|
||
|
|
||
|
conversion = {
|
||
|
snapshotToShapes: function(snapshot) {
|
||
|
var i, len, ref, results, shape;
|
||
|
ref = snapshot.shapes;
|
||
|
results = [];
|
||
|
for (i = 0, len = ref.length; i < len; i++) {
|
||
|
shape = ref[i];
|
||
|
results.push(shapes.JSONToShape(shape));
|
||
|
}
|
||
|
return results;
|
||
|
},
|
||
|
snapshotJSONToShapes: function(json) {
|
||
|
return conversion.snapshotToShapes(JSON.parse(json));
|
||
|
}
|
||
|
};
|
||
|
|
||
|
baseTools = require('./tools/base');
|
||
|
|
||
|
tools = {
|
||
|
Pencil: require('./tools/Pencil'),
|
||
|
Eraser: require('./tools/Eraser'),
|
||
|
Line: require('./tools/Line'),
|
||
|
Rectangle: require('./tools/Rectangle'),
|
||
|
Ellipse: require('./tools/Ellipse'),
|
||
|
Text: require('./tools/Text'),
|
||
|
Polygon: require('./tools/Polygon'),
|
||
|
Pan: require('./tools/Pan'),
|
||
|
Eyedropper: require('./tools/Eyedropper'),
|
||
|
SelectShape: require('./tools/SelectShape'),
|
||
|
Tool: baseTools.Tool,
|
||
|
ToolWithStroke: baseTools.ToolWithStroke
|
||
|
};
|
||
|
|
||
|
defaultTools = defaultOptions.tools;
|
||
|
|
||
|
defaultImageURLPrefix = defaultOptions.imageURLPrefix;
|
||
|
|
||
|
setDefaultImageURLPrefix = function(newDefault) {
|
||
|
defaultImageURLPrefix = newDefault;
|
||
|
return defaultOptions.imageURLPrefix = newDefault;
|
||
|
};
|
||
|
|
||
|
init = function(el, opts) {
|
||
|
var child, i, len, opt, ref;
|
||
|
if (opts == null) {
|
||
|
opts = {};
|
||
|
}
|
||
|
for (opt in defaultOptions) {
|
||
|
if (!(opt in opts)) {
|
||
|
opts[opt] = defaultOptions[opt];
|
||
|
}
|
||
|
}
|
||
|
ref = el.children;
|
||
|
for (i = 0, len = ref.length; i < len; i++) {
|
||
|
child = ref[i];
|
||
|
el.removeChild(child);
|
||
|
}
|
||
|
return require('./reactGUI/initDOM')(el, opts);
|
||
|
};
|
||
|
|
||
|
initWithoutGUI = function(el, opts) {
|
||
|
var drawingViewElement, lc, originalClassName;
|
||
|
originalClassName = el.className;
|
||
|
if ([' ', ' '].join(el.className).indexOf(' literally ') === -1) {
|
||
|
el.className = el.className + ' literally';
|
||
|
}
|
||
|
el.className = el.className + ' toolbar-hidden';
|
||
|
drawingViewElement = document.createElement('div');
|
||
|
drawingViewElement.className = 'lc-drawing';
|
||
|
el.appendChild(drawingViewElement);
|
||
|
lc = new LiterallyCanvasModel(drawingViewElement, opts);
|
||
|
lc.teardown = function() {
|
||
|
var child, i, len, ref;
|
||
|
lc._teardown();
|
||
|
ref = el.children;
|
||
|
for (i = 0, len = ref.length; i < len; i++) {
|
||
|
child = ref[i];
|
||
|
el.removeChild(child);
|
||
|
}
|
||
|
return el.className = originalClassName;
|
||
|
};
|
||
|
if ('onInit' in opts) {
|
||
|
opts.onInit(lc);
|
||
|
}
|
||
|
return lc;
|
||
|
};
|
||
|
|
||
|
registerJQueryPlugin = function(_$) {
|
||
|
return _$.fn.literallycanvas = function(opts) {
|
||
|
if (opts == null) {
|
||
|
opts = {};
|
||
|
}
|
||
|
this.each((function(_this) {
|
||
|
return function(ix, el) {
|
||
|
return el.literallycanvas = init(el, opts);
|
||
|
};
|
||
|
})(this));
|
||
|
return this;
|
||
|
};
|
||
|
};
|
||
|
|
||
|
if (typeof window !== 'undefined') {
|
||
|
window.LC = {
|
||
|
init: init
|
||
|
};
|
||
|
if (window.$) {
|
||
|
registerJQueryPlugin(window.$);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
module.exports = {
|
||
|
init: init,
|
||
|
registerJQueryPlugin: registerJQueryPlugin,
|
||
|
util: util,
|
||
|
tools: tools,
|
||
|
setDefaultImageURLPrefix: setDefaultImageURLPrefix,
|
||
|
defaultTools: defaultTools,
|
||
|
defineOptionsStyle: defineOptionsStyle,
|
||
|
LiterallyCanvasReactComponent: LiterallyCanvasReactComponent,
|
||
|
defineShape: shapes.defineShape,
|
||
|
createShape: shapes.createShape,
|
||
|
JSONToShape: shapes.JSONToShape,
|
||
|
shapeToJSON: shapes.shapeToJSON,
|
||
|
defineCanvasRenderer: canvasRenderer.defineCanvasRenderer,
|
||
|
renderShapeToContext: canvasRenderer.renderShapeToContext,
|
||
|
renderShapeToCanvas: canvasRenderer.renderShapeToCanvas,
|
||
|
renderShapesToCanvas: util.renderShapes,
|
||
|
defineSVGRenderer: svgRenderer.defineSVGRenderer,
|
||
|
renderShapeToSVG: svgRenderer.renderShapeToSVG,
|
||
|
renderShapesToSVG: util.renderShapesToSVG,
|
||
|
snapshotToShapes: conversion.snapshotToShapes,
|
||
|
snapshotJSONToShapes: conversion.snapshotJSONToShapes,
|
||
|
renderSnapshotToImage: renderSnapshotToImage,
|
||
|
renderSnapshotToSVG: renderSnapshotToSVG,
|
||
|
localize: localize
|
||
|
};
|
||
|
|
||
|
|
||
|
},{"./core/LiterallyCanvas":5,"./core/canvasRenderer":9,"./core/defaultOptions":10,"./core/localization":13,"./core/renderSnapshotToImage":15,"./core/renderSnapshotToSVG":16,"./core/shapes":17,"./core/svgRenderer":18,"./core/util":19,"./ie_customevent":20,"./ie_setLineDash":21,"./optionsStyles/font":23,"./optionsStyles/line-options-and-stroke-width":24,"./optionsStyles/null":25,"./optionsStyles/optionsStyles":26,"./optionsStyles/polygon-and-stroke-width":27,"./optionsStyles/stroke-or-fill":28,"./optionsStyles/stroke-width":29,"./reactGUI/LiterallyCanvas":32,"./reactGUI/initDOM":42,"./tools/Ellipse":43,"./tools/Eraser":44,"./tools/Eyedropper":45,"./tools/Line":46,"./tools/Pan":47,"./tools/Pencil":48,"./tools/Polygon":49,"./tools/Rectangle":50,"./tools/SelectShape":51,"./tools/Text":52,"./tools/base":53}],23:[function(require,module,exports){
|
||
|
var ALL_FONTS, FONT_NAME_TO_VALUE, MONOSPACE_FONTS, OTHER_FONTS, React, SANS_SERIF_FONTS, SERIF_FONTS, _, defineOptionsStyle, i, j, l, len, len1, len2, len3, m, name, ref, ref1, ref2, ref3, value;
|
||
|
|
||
|
React = require('../reactGUI/React-shim');
|
||
|
|
||
|
defineOptionsStyle = require('./optionsStyles').defineOptionsStyle;
|
||
|
|
||
|
_ = require('../core/localization')._;
|
||
|
|
||
|
SANS_SERIF_FONTS = [['Arial', 'Arial,"Helvetica Neue",Helvetica,sans-serif'], ['Arial Black', '"Arial Black","Arial Bold",Gadget,sans-serif'], ['Arial Narrow', '"Arial Narrow",Arial,sans-serif'], ['Gill Sans', '"Gill Sans","Gill Sans MT",Calibri,sans-serif'], ['Helvetica', '"Helvetica Neue",Helvetica,Arial,sans-serif'], ['Impact', 'Impact,Haettenschweiler,"Franklin Gothic Bold",Charcoal,"Helvetica Inserat","Bitstream Vera Sans Bold","Arial Black",sans-serif'], ['Tahoma', 'Tahoma,Verdana,Segoe,sans-serif'], ['Trebuchet MS', '"Trebuchet MS","Lucida Grande","Lucida Sans Unicode","Lucida Sans",Tahoma,sans-serif'], ['Verdana', 'Verdana,Geneva,sans-serif']].map(function(arg) {
|
||
|
var name, value;
|
||
|
name = arg[0], value = arg[1];
|
||
|
return {
|
||
|
name: _(name),
|
||
|
value: value
|
||
|
};
|
||
|
});
|
||
|
|
||
|
SERIF_FONTS = [['Baskerville', 'Baskerville,"Baskerville Old Face","Hoefler Text",Garamond,"Times New Roman",serif'], ['Garamond', 'Garamond,Baskerville,"Baskerville Old Face","Hoefler Text","Times New Roman",serif'], ['Georgia', 'Georgia,Times,"Times New Roman",serif'], ['Hoefler Text', '"Hoefler Text","Baskerville Old Face",Garamond,"Times New Roman",serif'], ['Lucida Bright', '"Lucida Bright",Georgia,serif'], ['Palatino', 'Palatino,"Palatino Linotype","Palatino LT STD","Book Antiqua",Georgia,serif'], ['Times New Roman', 'TimesNewRoman,"Times New Roman",Times,Baskerville,Georgia,serif']].map(function(arg) {
|
||
|
var name, value;
|
||
|
name = arg[0], value = arg[1];
|
||
|
return {
|
||
|
name: _(name),
|
||
|
value: value
|
||
|
};
|
||
|
});
|
||
|
|
||
|
MONOSPACE_FONTS = [['Consolas/Monaco', 'Consolas,monaco,"Lucida Console",monospace'], ['Courier New', '"Courier New",Courier,"Lucida Sans Typewriter","Lucida Typewriter",monospace'], ['Lucida Sans Typewriter', '"Lucida Sans Typewriter","Lucida Console",monaco,"Bitstream Vera Sans Mono",monospace']].map(function(arg) {
|
||
|
var name, value;
|
||
|
name = arg[0], value = arg[1];
|
||
|
return {
|
||
|
name: _(name),
|
||
|
value: value
|
||
|
};
|
||
|
});
|
||
|
|
||
|
OTHER_FONTS = [['Copperplate', 'Copperplate,"Copperplate Gothic Light",fantasy'], ['Papyrus', 'Papyrus,fantasy'], ['Script', '"Brush Script MT",cursive']].map(function(arg) {
|
||
|
var name, value;
|
||
|
name = arg[0], value = arg[1];
|
||
|
return {
|
||
|
name: _(name),
|
||
|
value: value
|
||
|
};
|
||
|
});
|
||
|
|
||
|
ALL_FONTS = [[_('Sans Serif'), SANS_SERIF_FONTS], [_('Serif'), SERIF_FONTS], [_('Monospace'), MONOSPACE_FONTS], [_('Other'), OTHER_FONTS]];
|
||
|
|
||
|
FONT_NAME_TO_VALUE = {};
|
||
|
|
||
|
for (i = 0, len = SANS_SERIF_FONTS.length; i < len; i++) {
|
||
|
ref = SANS_SERIF_FONTS[i], name = ref.name, value = ref.value;
|
||
|
FONT_NAME_TO_VALUE[name] = value;
|
||
|
}
|
||
|
|
||
|
for (j = 0, len1 = SERIF_FONTS.length; j < len1; j++) {
|
||
|
ref1 = SERIF_FONTS[j], name = ref1.name, value = ref1.value;
|
||
|
FONT_NAME_TO_VALUE[name] = value;
|
||
|
}
|
||
|
|
||
|
for (l = 0, len2 = MONOSPACE_FONTS.length; l < len2; l++) {
|
||
|
ref2 = MONOSPACE_FONTS[l], name = ref2.name, value = ref2.value;
|
||
|
FONT_NAME_TO_VALUE[name] = value;
|
||
|
}
|
||
|
|
||
|
for (m = 0, len3 = OTHER_FONTS.length; m < len3; m++) {
|
||
|
ref3 = OTHER_FONTS[m], name = ref3.name, value = ref3.value;
|
||
|
FONT_NAME_TO_VALUE[name] = value;
|
||
|
}
|
||
|
|
||
|
defineOptionsStyle('font', React.createClass({
|
||
|
displayName: 'FontOptions',
|
||
|
getInitialState: function() {
|
||
|
return {
|
||
|
isItalic: false,
|
||
|
isBold: false,
|
||
|
fontName: 'Helvetica',
|
||
|
fontSizeIndex: 4
|
||
|
};
|
||
|
},
|
||
|
getFontSizes: function() {
|
||
|
return [9, 10, 12, 14, 18, 24, 36, 48, 64, 72, 96, 144, 288];
|
||
|
},
|
||
|
updateTool: function(newState) {
|
||
|
var fontSize, items, k;
|
||
|
if (newState == null) {
|
||
|
newState = {};
|
||
|
}
|
||
|
for (k in this.state) {
|
||
|
if (!(k in newState)) {
|
||
|
newState[k] = this.state[k];
|
||
|
}
|
||
|
}
|
||
|
fontSize = this.getFontSizes()[newState.fontSizeIndex];
|
||
|
items = [];
|
||
|
if (newState.isItalic) {
|
||
|
items.push('italic');
|
||
|
}
|
||
|
if (newState.isBold) {
|
||
|
items.push('bold');
|
||
|
}
|
||
|
items.push(fontSize + "px");
|
||
|
items.push(FONT_NAME_TO_VALUE[newState.fontName]);
|
||
|
this.props.lc.tool.font = items.join(' ');
|
||
|
return this.props.lc.trigger('setFont', items.join(' '));
|
||
|
},
|
||
|
handleFontSize: function(event) {
|
||
|
var newState;
|
||
|
newState = {
|
||
|
fontSizeIndex: event.target.value
|
||
|
};
|
||
|
this.setState(newState);
|
||
|
return this.updateTool(newState);
|
||
|
},
|
||
|
handleFontFamily: function(event) {
|
||
|
var newState;
|
||
|
newState = {
|
||
|
fontName: event.target.selectedOptions[0].innerHTML
|
||
|
};
|
||
|
this.setState(newState);
|
||
|
return this.updateTool(newState);
|
||
|
},
|
||
|
handleItalic: function(event) {
|
||
|
var newState;
|
||
|
newState = {
|
||
|
isItalic: !this.state.isItalic
|
||
|
};
|
||
|
this.setState(newState);
|
||
|
return this.updateTool(newState);
|
||
|
},
|
||
|
handleBold: function(event) {
|
||
|
var newState;
|
||
|
newState = {
|
||
|
isBold: !this.state.isBold
|
||
|
};
|
||
|
this.setState(newState);
|
||
|
return this.updateTool(newState);
|
||
|
},
|
||
|
componentDidMount: function() {
|
||
|
return this.updateTool();
|
||
|
},
|
||
|
render: function() {
|
||
|
var br, div, input, label, lc, optgroup, option, ref4, select, span;
|
||
|
lc = this.props.lc;
|
||
|
ref4 = React.DOM, div = ref4.div, input = ref4.input, select = ref4.select, option = ref4.option, br = ref4.br, label = ref4.label, span = ref4.span, optgroup = ref4.optgroup;
|
||
|
return div({
|
||
|
className: 'lc-font-settings'
|
||
|
}, select({
|
||
|
value: this.state.fontSizeIndex,
|
||
|
onChange: this.handleFontSize
|
||
|
}, this.getFontSizes().map((function(_this) {
|
||
|
return function(size, ix) {
|
||
|
return option({
|
||
|
value: ix,
|
||
|
key: ix
|
||
|
}, size + "px");
|
||
|
};
|
||
|
})(this))), select({
|
||
|
value: this.state.fontName,
|
||
|
onChange: this.handleFontFamily
|
||
|
}, ALL_FONTS.map((function(_this) {
|
||
|
return function(arg) {
|
||
|
var fonts, label;
|
||
|
label = arg[0], fonts = arg[1];
|
||
|
return optgroup({
|
||
|
key: label,
|
||
|
label: label
|
||
|
}, fonts.map(function(family, ix) {
|
||
|
return option({
|
||
|
value: family.name,
|
||
|
key: ix
|
||
|
}, family.name);
|
||
|
}));
|
||
|
};
|
||
|
})(this))), span({}, label({
|
||
|
htmlFor: 'italic'
|
||
|
}, _("italic")), input({
|
||
|
type: 'checkbox',
|
||
|
id: 'italic',
|
||
|
checked: this.state.isItalic,
|
||
|
onChange: this.handleItalic
|
||
|
})), span({}, label({
|
||
|
htmlFor: 'bold'
|
||
|
}, _("bold")), input({
|
||
|
type: 'checkbox',
|
||
|
id: 'bold',
|
||
|
checked: this.state.isBold,
|
||
|
onChange: this.handleBold
|
||
|
})));
|
||
|
}
|
||
|
}));
|
||
|
|
||
|
module.exports = {};
|
||
|
|
||
|
|
||
|
},{"../core/localization":13,"../reactGUI/React-shim":35,"./optionsStyles":26}],24:[function(require,module,exports){
|
||
|
var React, StrokeWidthPicker, classSet, createSetStateOnEventMixin, defineOptionsStyle;
|
||
|
|
||
|
React = require('../reactGUI/React-shim');
|
||
|
|
||
|
defineOptionsStyle = require('./optionsStyles').defineOptionsStyle;
|
||
|
|
||
|
StrokeWidthPicker = React.createFactory(require('../reactGUI/StrokeWidthPicker'));
|
||
|
|
||
|
createSetStateOnEventMixin = require('../reactGUI/createSetStateOnEventMixin');
|
||
|
|
||
|
classSet = require('../core/util').classSet;
|
||
|
|
||
|
defineOptionsStyle('line-options-and-stroke-width', React.createClass({
|
||
|
displayName: 'LineOptionsAndStrokeWidth',
|
||
|
getState: function() {
|
||
|
return {
|
||
|
strokeWidth: this.props.tool.strokeWidth,
|
||
|
isDashed: this.props.tool.isDashed,
|
||
|
hasEndArrow: this.props.tool.hasEndArrow
|
||
|
};
|
||
|
},
|
||
|
getInitialState: function() {
|
||
|
return this.getState();
|
||
|
},
|
||
|
mixins: [createSetStateOnEventMixin('toolChange')],
|
||
|
render: function() {
|
||
|
var arrowButtonClass, dashButtonClass, div, img, li, ref, style, toggleIsDashed, togglehasEndArrow, ul;
|
||
|
ref = React.DOM, div = ref.div, ul = ref.ul, li = ref.li, img = ref.img;
|
||
|
toggleIsDashed = (function(_this) {
|
||
|
return function() {
|
||
|
_this.props.tool.isDashed = !_this.props.tool.isDashed;
|
||
|
return _this.setState(_this.getState());
|
||
|
};
|
||
|
})(this);
|
||
|
togglehasEndArrow = (function(_this) {
|
||
|
return function() {
|
||
|
_this.props.tool.hasEndArrow = !_this.props.tool.hasEndArrow;
|
||
|
return _this.setState(_this.getState());
|
||
|
};
|
||
|
})(this);
|
||
|
dashButtonClass = classSet({
|
||
|
'square-toolbar-button': true,
|
||
|
'selected': this.state.isDashed
|
||
|
});
|
||
|
arrowButtonClass = classSet({
|
||
|
'square-toolbar-button': true,
|
||
|
'selected': this.state.hasEndArrow
|
||
|
});
|
||
|
style = {
|
||
|
float: 'left',
|
||
|
margin: 1
|
||
|
};
|
||
|
return div({}, div({
|
||
|
className: dashButtonClass,
|
||
|
onClick: toggleIsDashed,
|
||
|
style: style
|
||
|
}, img({
|
||
|
src: this.props.imageURLPrefix + "/dashed-line.png"
|
||
|
})), div({
|
||
|
className: arrowButtonClass,
|
||
|
onClick: togglehasEndArrow,
|
||
|
style: style
|
||
|
}, img({
|
||
|
src: this.props.imageURLPrefix + "/line-with-arrow.png"
|
||
|
})), StrokeWidthPicker({
|
||
|
tool: this.props.tool,
|
||
|
lc: this.props.lc
|
||
|
}));
|
||
|
}
|
||
|
}));
|
||
|
|
||
|
module.exports = {};
|
||
|
|
||
|
|
||
|
},{"../core/util":19,"../reactGUI/React-shim":35,"../reactGUI/StrokeWidthPicker":37,"../reactGUI/createSetStateOnEventMixin":40,"./optionsStyles":26}],25:[function(require,module,exports){
|
||
|
var React, defineOptionsStyle;
|
||
|
|
||
|
React = require('../reactGUI/React-shim');
|
||
|
|
||
|
defineOptionsStyle = require('./optionsStyles').defineOptionsStyle;
|
||
|
|
||
|
defineOptionsStyle('null', React.createClass({
|
||
|
displayName: 'NoOptions',
|
||
|
render: function() {
|
||
|
return React.DOM.div();
|
||
|
}
|
||
|
}));
|
||
|
|
||
|
module.exports = {};
|
||
|
|
||
|
|
||
|
},{"../reactGUI/React-shim":35,"./optionsStyles":26}],26:[function(require,module,exports){
|
||
|
var React, defineOptionsStyle, optionsStyles;
|
||
|
|
||
|
React = require('../reactGUI/React-shim');
|
||
|
|
||
|
optionsStyles = {};
|
||
|
|
||
|
defineOptionsStyle = function(name, style) {
|
||
|
return optionsStyles[name] = React.createFactory(style);
|
||
|
};
|
||
|
|
||
|
module.exports = {
|
||
|
optionsStyles: optionsStyles,
|
||
|
defineOptionsStyle: defineOptionsStyle
|
||
|
};
|
||
|
|
||
|
|
||
|
},{"../reactGUI/React-shim":35}],27:[function(require,module,exports){
|
||
|
var React, StrokeWidthPicker, createSetStateOnEventMixin, defineOptionsStyle;
|
||
|
|
||
|
React = require('../reactGUI/React-shim');
|
||
|
|
||
|
defineOptionsStyle = require('./optionsStyles').defineOptionsStyle;
|
||
|
|
||
|
StrokeWidthPicker = React.createFactory(require('../reactGUI/StrokeWidthPicker'));
|
||
|
|
||
|
createSetStateOnEventMixin = require('../reactGUI/createSetStateOnEventMixin');
|
||
|
|
||
|
defineOptionsStyle('polygon-and-stroke-width', React.createClass({
|
||
|
displayName: 'PolygonAndStrokeWidth',
|
||
|
getState: function() {
|
||
|
return {
|
||
|
strokeWidth: this.props.tool.strokeWidth,
|
||
|
inProgress: false
|
||
|
};
|
||
|
},
|
||
|
getInitialState: function() {
|
||
|
return this.getState();
|
||
|
},
|
||
|
mixins: [createSetStateOnEventMixin('toolChange')],
|
||
|
componentDidMount: function() {
|
||
|
var hidePolygonTools, showPolygonTools, unsubscribeFuncs;
|
||
|
unsubscribeFuncs = [];
|
||
|
this.unsubscribe = (function(_this) {
|
||
|
return function() {
|
||
|
var func, i, len, results;
|
||
|
results = [];
|
||
|
for (i = 0, len = unsubscribeFuncs.length; i < len; i++) {
|
||
|
func = unsubscribeFuncs[i];
|
||
|
results.push(func());
|
||
|
}
|
||
|
return results;
|
||
|
};
|
||
|
})(this);
|
||
|
showPolygonTools = (function(_this) {
|
||
|
return function() {
|
||
|
if (!_this.state.inProgress) {
|
||
|
return _this.setState({
|
||
|
inProgress: true
|
||
|
});
|
||
|
}
|
||
|
};
|
||
|
})(this);
|
||
|
hidePolygonTools = (function(_this) {
|
||
|
return function() {
|
||
|
return _this.setState({
|
||
|
inProgress: false
|
||
|
});
|
||
|
};
|
||
|
})(this);
|
||
|
unsubscribeFuncs.push(this.props.lc.on('lc-polygon-started', showPolygonTools));
|
||
|
return unsubscribeFuncs.push(this.props.lc.on('lc-polygon-stopped', hidePolygonTools));
|
||
|
},
|
||
|
componentWillUnmount: function() {
|
||
|
return this.unsubscribe();
|
||
|
},
|
||
|
render: function() {
|
||
|
var div, img, lc, polygonCancel, polygonFinishClosed, polygonFinishOpen, polygonToolStyle, ref;
|
||
|
lc = this.props.lc;
|
||
|
ref = React.DOM, div = ref.div, img = ref.img;
|
||
|
polygonFinishOpen = (function(_this) {
|
||
|
return function() {
|
||
|
return lc.trigger('lc-polygon-finishopen');
|
||
|
};
|
||
|
})(this);
|
||
|
polygonFinishClosed = (function(_this) {
|
||
|
return function() {
|
||
|
return lc.trigger('lc-polygon-finishclosed');
|
||
|
};
|
||
|
})(this);
|
||
|
polygonCancel = (function(_this) {
|
||
|
return function() {
|
||
|
return lc.trigger('lc-polygon-cancel');
|
||
|
};
|
||
|
})(this);
|
||
|
polygonToolStyle = {};
|
||
|
if (!this.state.inProgress) {
|
||
|
polygonToolStyle = {
|
||
|
display: 'none'
|
||
|
};
|
||
|
}
|
||
|
return div({}, div({
|
||
|
className: 'polygon-toolbar horz-toolbar',
|
||
|
style: polygonToolStyle
|
||
|
}, div({
|
||
|
className: 'square-toolbar-button',
|
||
|
onClick: polygonFinishOpen
|
||
|
}, img({
|
||
|
src: this.props.imageURLPrefix + "/polygon-open.png"
|
||
|
})), div({
|
||
|
className: 'square-toolbar-button',
|
||
|
onClick: polygonFinishClosed
|
||
|
}, img({
|
||
|
src: this.props.imageURLPrefix + "/polygon-closed.png"
|
||
|
})), div({
|
||
|
className: 'square-toolbar-button',
|
||
|
onClick: polygonCancel
|
||
|
}, img({
|
||
|
src: this.props.imageURLPrefix + "/polygon-cancel.png"
|
||
|
}))), div({}, StrokeWidthPicker({
|
||
|
tool: this.props.tool,
|
||
|
lc: this.props.lc
|
||
|
})));
|
||
|
}
|
||
|
}));
|
||
|
|
||
|
module.exports = {};
|
||
|
|
||
|
|
||
|
},{"../reactGUI/React-shim":35,"../reactGUI/StrokeWidthPicker":37,"../reactGUI/createSetStateOnEventMixin":40,"./optionsStyles":26}],28:[function(require,module,exports){
|
||
|
'use strict';
|
||
|
|
||
|
var React = require('../reactGUI/React-shim');
|
||
|
|
||
|
var _require = require('./optionsStyles');
|
||
|
|
||
|
var defineOptionsStyle = _require.defineOptionsStyle;
|
||
|
|
||
|
var createSetStateOnEventMixin = require('../reactGUI/createSetStateOnEventMixin');
|
||
|
var _ = require('../core/localization')._;
|
||
|
|
||
|
defineOptionsStyle('stroke-or-fill', React.createClass({
|
||
|
displayName: 'StrokeOrFillPicker',
|
||
|
getState: function getState() {
|
||
|
return { strokeOrFill: 'stroke' };
|
||
|
},
|
||
|
getInitialState: function getInitialState() {
|
||
|
return this.getState();
|
||
|
},
|
||
|
mixins: [createSetStateOnEventMixin('toolChange')],
|
||
|
|
||
|
onChange: function onChange(e) {
|
||
|
if (e.target.id == 'stroke-or-fill-stroke') {
|
||
|
this.props.lc.tool.strokeOrFill = 'stroke';
|
||
|
} else {
|
||
|
this.props.lc.tool.strokeOrFill = 'fill';
|
||
|
}
|
||
|
this.setState(this.getState());
|
||
|
},
|
||
|
|
||
|
render: function render() {
|
||
|
var lc = this.props.lc;
|
||
|
|
||
|
return React.createElement(
|
||
|
'form',
|
||
|
null,
|
||
|
React.createElement(
|
||
|
'span',
|
||
|
null,
|
||
|
' ',
|
||
|
_('Color to change:'),
|
||
|
' '
|
||
|
),
|
||
|
React.createElement(
|
||
|
'span',
|
||
|
null,
|
||
|
React.createElement('input', { type: 'radio', name: 'stroke-or-fill', value: 'stroke',
|
||
|
id: 'stroke-or-fill-stroke', onChange: this.onChange,
|
||
|
checked: lc.tool.strokeOrFill == 'stroke' }),
|
||
|
React.createElement(
|
||
|
'label',
|
||
|
{ htmlFor: 'stroke-or-fill-stroke', className: 'label' },
|
||
|
' ',
|
||
|
_("stroke")
|
||
|
)
|
||
|
),
|
||
|
React.createElement(
|
||
|
'span',
|
||
|
null,
|
||
|
React.createElement('input', { type: 'radio', name: 'stroke-or-fill', value: 'fill',
|
||
|
id: 'stroke-or-fill-fill', onChange: this.onChange,
|
||
|
checked: lc.tool.strokeOrFill == 'fill' }),
|
||
|
React.createElement(
|
||
|
'label',
|
||
|
{ htmlFor: 'stroke-or-fill-fill', className: 'label' },
|
||
|
' ',
|
||
|
_("fill")
|
||
|
)
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
}));
|
||
|
|
||
|
module.exports = {};
|
||
|
|
||
|
},{"../core/localization":13,"../reactGUI/React-shim":35,"../reactGUI/createSetStateOnEventMixin":40,"./optionsStyles":26}],29:[function(require,module,exports){
|
||
|
var StrokeWidthPicker, defineOptionsStyle;
|
||
|
|
||
|
defineOptionsStyle = require('./optionsStyles').defineOptionsStyle;
|
||
|
|
||
|
StrokeWidthPicker = require('../reactGUI/StrokeWidthPicker');
|
||
|
|
||
|
defineOptionsStyle('stroke-width', StrokeWidthPicker);
|
||
|
|
||
|
module.exports = {};
|
||
|
|
||
|
|
||
|
},{"../reactGUI/StrokeWidthPicker":37,"./optionsStyles":26}],30:[function(require,module,exports){
|
||
|
var ClearButton, React, _, classSet, createSetStateOnEventMixin;
|
||
|
|
||
|
React = require('./React-shim');
|
||
|
|
||
|
createSetStateOnEventMixin = require('./createSetStateOnEventMixin');
|
||
|
|
||
|
_ = require('../core/localization')._;
|
||
|
|
||
|
classSet = require('../core/util').classSet;
|
||
|
|
||
|
ClearButton = React.createClass({
|
||
|
displayName: 'ClearButton',
|
||
|
getState: function() {
|
||
|
return {
|
||
|
isEnabled: this.props.lc.canUndo()
|
||
|
};
|
||
|
},
|
||
|
getInitialState: function() {
|
||
|
return this.getState();
|
||
|
},
|
||
|
mixins: [createSetStateOnEventMixin('drawingChange')],
|
||
|
render: function() {
|
||
|
var className, div, lc, onClick;
|
||
|
div = React.DOM.div;
|
||
|
lc = this.props.lc;
|
||
|
className = classSet({
|
||
|
'lc-clear': true,
|
||
|
'toolbar-button': true,
|
||
|
'fat-button': true,
|
||
|
'disabled': !this.state.isEnabled
|
||
|
});
|
||
|
onClick = lc.canUndo() ? ((function(_this) {
|
||
|
return function() {
|
||
|
return lc.clear();
|
||
|
};
|
||
|
})(this)) : function() {};
|
||
|
return div({
|
||
|
className: className,
|
||
|
onClick: onClick
|
||
|
}, _('Clear'));
|
||
|
}
|
||
|
});
|
||
|
|
||
|
module.exports = ClearButton;
|
||
|
|
||
|
|
||
|
},{"../core/localization":13,"../core/util":19,"./React-shim":35,"./createSetStateOnEventMixin":40}],31:[function(require,module,exports){
|
||
|
var ColorGrid, ColorWell, PureRenderMixin, React, _, cancelAnimationFrame, classSet, getHSLAString, getHSLString, parseHSLAString, ref, requestAnimationFrame;
|
||
|
|
||
|
React = require('./React-shim');
|
||
|
|
||
|
PureRenderMixin = require('react-addons-pure-render-mixin');
|
||
|
|
||
|
ref = require('../core/util'), classSet = ref.classSet, requestAnimationFrame = ref.requestAnimationFrame, cancelAnimationFrame = ref.cancelAnimationFrame;
|
||
|
|
||
|
_ = require('../core/localization')._;
|
||
|
|
||
|
parseHSLAString = function(s) {
|
||
|
var components, firstParen, insideParens, lastParen;
|
||
|
if (s === 'transparent') {
|
||
|
return {
|
||
|
hue: 0,
|
||
|
sat: 0,
|
||
|
light: 0,
|
||
|
alpha: 0
|
||
|
};
|
||
|
}
|
||
|
if ((s != null ? s.substring(0, 4) : void 0) !== 'hsla') {
|
||
|
return null;
|
||
|
}
|
||
|
firstParen = s.indexOf('(');
|
||
|
lastParen = s.indexOf(')');
|
||
|
insideParens = s.substring(firstParen + 1, lastParen - firstParen + 4);
|
||
|
components = (function() {
|
||
|
var j, len, ref1, results;
|
||
|
ref1 = insideParens.split(',');
|
||
|
results = [];
|
||
|
for (j = 0, len = ref1.length; j < len; j++) {
|
||
|
s = ref1[j];
|
||
|
results.push(s.trim());
|
||
|
}
|
||
|
return results;
|
||
|
})();
|
||
|
return {
|
||
|
hue: parseInt(components[0], 10),
|
||
|
sat: parseInt(components[1].substring(0, components[1].length - 1), 10),
|
||
|
light: parseInt(components[2].substring(0, components[2].length - 1), 10),
|
||
|
alpha: parseFloat(components[3])
|
||
|
};
|
||
|
};
|
||
|
|
||
|
getHSLAString = function(arg) {
|
||
|
var alpha, hue, light, sat;
|
||
|
hue = arg.hue, sat = arg.sat, light = arg.light, alpha = arg.alpha;
|
||
|
return "hsla(" + hue + ", " + sat + "%, " + light + "%, " + alpha + ")";
|
||
|
};
|
||
|
|
||
|
getHSLString = function(arg) {
|
||
|
var hue, light, sat;
|
||
|
hue = arg.hue, sat = arg.sat, light = arg.light;
|
||
|
return "hsl(" + hue + ", " + sat + "%, " + light + "%)";
|
||
|
};
|
||
|
|
||
|
ColorGrid = React.createFactory(React.createClass({
|
||
|
displayName: 'ColorGrid',
|
||
|
mixins: [PureRenderMixin],
|
||
|
render: function() {
|
||
|
var div;
|
||
|
div = React.DOM.div;
|
||
|
return div({}, this.props.rows.map((function(_this) {
|
||
|
return function(row, ix) {
|
||
|
return div({
|
||
|
className: 'color-row',
|
||
|
key: ix,
|
||
|
style: {
|
||
|
width: 20 * row.length
|
||
|
}
|
||
|
}, row.map(function(cellColor, ix2) {
|
||
|
var alpha, className, colorString, colorStringNoAlpha, hue, light, sat, update;
|
||
|
hue = cellColor.hue, sat = cellColor.sat, light = cellColor.light, alpha = cellColor.alpha;
|
||
|
colorString = getHSLAString(cellColor);
|
||
|
colorStringNoAlpha = "hsl(" + hue + ", " + sat + "%, " + light + "%)";
|
||
|
className = classSet({
|
||
|
'color-cell': true,
|
||
|
'selected': _this.props.selectedColor === colorString
|
||
|
});
|
||
|
update = function(e) {
|
||
|
_this.props.onChange(cellColor, colorString);
|
||
|
e.stopPropagation();
|
||
|
return e.preventDefault();
|
||
|
};
|
||
|
return div({
|
||
|
className: className,
|
||
|
onTouchStart: update,
|
||
|
onTouchMove: update,
|
||
|
onClick: update,
|
||
|
style: {
|
||
|
backgroundColor: colorStringNoAlpha
|
||
|
},
|
||
|
key: ix2
|
||
|
});
|
||
|
}));
|
||
|
};
|
||
|
})(this)));
|
||
|
}
|
||
|
}));
|
||
|
|
||
|
ColorWell = React.createClass({
|
||
|
displayName: 'ColorWell',
|
||
|
mixins: [PureRenderMixin],
|
||
|
getInitialState: function() {
|
||
|
var colorString, hsla;
|
||
|
colorString = this.props.lc.colors[this.props.colorName];
|
||
|
hsla = parseHSLAString(colorString);
|
||
|
if (hsla == null) {
|
||
|
hsla = {};
|
||
|
}
|
||
|
if (hsla.alpha == null) {
|
||
|
hsla.alpha = 1;
|
||
|
}
|
||
|
if (hsla.sat == null) {
|
||
|
hsla.sat = 100;
|
||
|
}
|
||
|
if (hsla.hue == null) {
|
||
|
hsla.hue = 0;
|
||
|
}
|
||
|
if (hsla.light == null) {
|
||
|
hsla.light = 50;
|
||
|
}
|
||
|
return {
|
||
|
colorString: colorString,
|
||
|
alpha: hsla.alpha,
|
||
|
sat: hsla.sat === 0 ? 100 : hsla.sat,
|
||
|
isPickerVisible: false,
|
||
|
hsla: hsla
|
||
|
};
|
||
|
},
|
||
|
componentDidMount: function() {
|
||
|
return this.unsubscribe = this.props.lc.on(this.props.colorName + "ColorChange", (function(_this) {
|
||
|
return function() {
|
||
|
var colorString;
|
||
|
colorString = _this.props.lc.colors[_this.props.colorName];
|
||
|
_this.setState({
|
||
|
colorString: colorString
|
||
|
});
|
||
|
return _this.setHSLAFromColorString(colorString);
|
||
|
};
|
||
|
})(this));
|
||
|
},
|
||
|
componentWillUnmount: function() {
|
||
|
return this.unsubscribe();
|
||
|
},
|
||
|
setHSLAFromColorString: function(c) {
|
||
|
var hsla;
|
||
|
hsla = parseHSLAString(c);
|
||
|
if (hsla) {
|
||
|
return this.setState({
|
||
|
hsla: hsla,
|
||
|
alpha: hsla.alpha,
|
||
|
sat: hsla.sat
|
||
|
});
|
||
|
} else {
|
||
|
return this.setState({
|
||
|
hsla: null,
|
||
|
alpha: 1,
|
||
|
sat: 100
|
||
|
});
|
||
|
}
|
||
|
},
|
||
|
closePicker: function() {
|
||
|
return this.setState({
|
||
|
isPickerVisible: false
|
||
|
});
|
||
|
},
|
||
|
togglePicker: function() {
|
||
|
var isPickerVisible, shouldResetSat;
|
||
|
isPickerVisible = !this.state.isPickerVisible;
|
||
|
shouldResetSat = isPickerVisible && this.state.sat === 0;
|
||
|
this.setHSLAFromColorString(this.state.colorString);
|
||
|
return this.setState({
|
||
|
isPickerVisible: isPickerVisible,
|
||
|
sat: shouldResetSat ? 100 : this.state.sat
|
||
|
});
|
||
|
},
|
||
|
setColor: function(c) {
|
||
|
this.setState({
|
||
|
colorString: c
|
||
|
});
|
||
|
this.setHSLAFromColorString(c);
|
||
|
return this.props.lc.setColor(this.props.colorName, c);
|
||
|
},
|
||
|
setAlpha: function(alpha) {
|
||
|
var hsla;
|
||
|
this.setState({
|
||
|
alpha: alpha
|
||
|
});
|
||
|
if (this.state.hsla) {
|
||
|
hsla = this.state.hsla;
|
||
|
hsla.alpha = alpha;
|
||
|
this.setState({
|
||
|
hsla: hsla
|
||
|
});
|
||
|
return this.setColor(getHSLAString(hsla));
|
||
|
}
|
||
|
},
|
||
|
setSat: function(sat) {
|
||
|
var hsla;
|
||
|
this.setState({
|
||
|
sat: sat
|
||
|
});
|
||
|
if (isNaN(sat)) {
|
||
|
throw "SAT";
|
||
|
}
|
||
|
if (this.state.hsla) {
|
||
|
hsla = this.state.hsla;
|
||
|
hsla.sat = sat;
|
||
|
this.setState({
|
||
|
hsla: hsla
|
||
|
});
|
||
|
return this.setColor(getHSLAString(hsla));
|
||
|
}
|
||
|
},
|
||
|
render: function() {
|
||
|
var br, div, label, ref1;
|
||
|
ref1 = React.DOM, div = ref1.div, label = ref1.label, br = ref1.br;
|
||
|
return div({
|
||
|
className: classSet({
|
||
|
'color-well': true,
|
||
|
'open': this.state.isPickerVisible
|
||
|
}),
|
||
|
onMouseLeave: this.closePicker,
|
||
|
style: {
|
||
|
float: 'left',
|
||
|
textAlign: 'center'
|
||
|
}
|
||
|
}, label({
|
||
|
float: 'left'
|
||
|
}, this.props.label), br({}), div({
|
||
|
className: classSet({
|
||
|
'color-well-color-container': true,
|
||
|
'selected': this.state.isPickerVisible
|
||
|
}),
|
||
|
style: {
|
||
|
backgroundColor: 'white'
|
||
|
},
|
||
|
onClick: this.togglePicker
|
||
|
}, div({
|
||
|
className: 'color-well-checker color-well-checker-top-left'
|
||
|
}), div({
|
||
|
className: 'color-well-checker color-well-checker-bottom-right',
|
||
|
style: {
|
||
|
left: '50%',
|
||
|
top: '50%'
|
||
|
}
|
||
|
}), div({
|
||
|
className: 'color-well-color',
|
||
|
style: {
|
||
|
backgroundColor: this.state.colorString
|
||
|
}
|
||
|
}, " ")), this.renderPicker());
|
||
|
},
|
||
|
renderPicker: function() {
|
||
|
var div, hue, i, input, j, label, len, onSelectColor, ref1, ref2, renderColor, renderLabel, rows;
|
||
|
ref1 = React.DOM, div = ref1.div, label = ref1.label, input = ref1.input;
|
||
|
if (!this.state.isPickerVisible) {
|
||
|
return null;
|
||
|
}
|
||
|
renderLabel = (function(_this) {
|
||
|
return function(text) {
|
||
|
return div({
|
||
|
className: 'color-row label',
|
||
|
key: text,
|
||
|
style: {
|
||
|
lineHeight: '20px',
|
||
|
height: 16
|
||
|
}
|
||
|
}, text);
|
||
|
};
|
||
|
})(this);
|
||
|
renderColor = (function(_this) {
|
||
|
return function() {
|
||
|
var checkerboardURL;
|
||
|
checkerboardURL = _this.props.lc.opts.imageURLPrefix + "/checkerboard-8x8.png";
|
||
|
return div({
|
||
|
className: 'color-row',
|
||
|
key: "color",
|
||
|
style: {
|
||
|
position: 'relative',
|
||
|
backgroundImage: "url(" + checkerboardURL + ")",
|
||
|
backgroundRepeat: 'repeat',
|
||
|
height: 24
|
||
|
}
|
||
|
}, div({
|
||
|
style: {
|
||
|
position: 'absolute',
|
||
|
top: 0,
|
||
|
right: 0,
|
||
|
bottom: 0,
|
||
|
left: 0,
|
||
|
backgroundColor: _this.state.colorString
|
||
|
}
|
||
|
}));
|
||
|
};
|
||
|
})(this);
|
||
|
rows = [];
|
||
|
rows.push((function() {
|
||
|
var j, results;
|
||
|
results = [];
|
||
|
for (i = j = 0; j <= 100; i = j += 10) {
|
||
|
results.push({
|
||
|
hue: 0,
|
||
|
sat: 0,
|
||
|
light: i,
|
||
|
alpha: this.state.alpha
|
||
|
});
|
||
|
}
|
||
|
return results;
|
||
|
}).call(this));
|
||
|
ref2 = [0, 30, 60, 90, 120, 150, 180, 210, 240, 270, 300, 330];
|
||
|
for (j = 0, len = ref2.length; j < len; j++) {
|
||
|
hue = ref2[j];
|
||
|
rows.push((function() {
|
||
|
var k, results;
|
||
|
results = [];
|
||
|
for (i = k = 10; k <= 90; i = k += 8) {
|
||
|
results.push({
|
||
|
hue: hue,
|
||
|
sat: this.state.sat,
|
||
|
light: i,
|
||
|
alpha: this.state.alpha
|
||
|
});
|
||
|
}
|
||
|
return results;
|
||
|
}).call(this));
|
||
|
}
|
||
|
onSelectColor = (function(_this) {
|
||
|
return function(hsla, s) {
|
||
|
return _this.setColor(s);
|
||
|
};
|
||
|
})(this);
|
||
|
return div({
|
||
|
className: 'color-picker-popup'
|
||
|
}, renderColor(), renderLabel(_("alpha")), input({
|
||
|
type: 'range',
|
||
|
min: 0,
|
||
|
max: 1,
|
||
|
step: 0.01,
|
||
|
value: this.state.alpha,
|
||
|
onChange: (function(_this) {
|
||
|
return function(e) {
|
||
|
return _this.setAlpha(parseFloat(e.target.value));
|
||
|
};
|
||
|
})(this)
|
||
|
}), renderLabel(_("saturation")), input({
|
||
|
type: 'range',
|
||
|
min: 0,
|
||
|
max: 100,
|
||
|
value: this.state.sat,
|
||
|
max: 100,
|
||
|
onChange: (function(_this) {
|
||
|
return function(e) {
|
||
|
return _this.setSat(parseInt(e.target.value, 10));
|
||
|
};
|
||
|
})(this)
|
||
|
}), ColorGrid({
|
||
|
rows: rows,
|
||
|
selectedColor: this.state.colorString,
|
||
|
onChange: onSelectColor
|
||
|
}));
|
||
|
}
|
||
|
});
|
||
|
|
||
|
module.exports = ColorWell;
|
||
|
|
||
|
|
||
|
},{"../core/localization":13,"../core/util":19,"./React-shim":35,"react-addons-pure-render-mixin":2}],32:[function(require,module,exports){
|
||
|
'use strict';
|
||
|
|
||
|
var React = require('../reactGUI/React-shim');
|
||
|
|
||
|
var _require = require('../reactGUI/ReactDOM-shim');
|
||
|
|
||
|
var findDOMNode = _require.findDOMNode;
|
||
|
|
||
|
var _require2 = require('../core/util');
|
||
|
|
||
|
var classSet = _require2.classSet;
|
||
|
|
||
|
var Picker = require('./Picker');
|
||
|
var Options = require('./Options');
|
||
|
var createToolButton = require('./createToolButton');
|
||
|
var LiterallyCanvasModel = require('../core/LiterallyCanvas');
|
||
|
var defaultOptions = require('../core/defaultOptions');
|
||
|
|
||
|
require('../optionsStyles/font');
|
||
|
require('../optionsStyles/stroke-width');
|
||
|
require('../optionsStyles/line-options-and-stroke-width');
|
||
|
require('../optionsStyles/polygon-and-stroke-width');
|
||
|
require('../optionsStyles/null');
|
||
|
|
||
|
var CanvasContainer = React.createClass({
|
||
|
displayName: 'CanvasContainer',
|
||
|
shouldComponentUpdate: function shouldComponentUpdate() {
|
||
|
// Avoid React trying to control this DOM
|
||
|
return false;
|
||
|
},
|
||
|
render: function render() {
|
||
|
return React.createElement('div', { key: 'literallycanvas', className: 'lc-drawing with-gui' });
|
||
|
}
|
||
|
});
|
||
|
|
||
|
var LiterallyCanvas = React.createClass({
|
||
|
displayName: 'LiterallyCanvas',
|
||
|
|
||
|
getDefaultProps: function getDefaultProps() {
|
||
|
return defaultOptions;
|
||
|
},
|
||
|
bindToModel: function bindToModel() {
|
||
|
var canvasContainerEl = findDOMNode(this.canvas);
|
||
|
var opts = this.props;
|
||
|
this.lc.bindToElement(canvasContainerEl);
|
||
|
|
||
|
if (typeof this.lc.opts.onInit === 'function') {
|
||
|
this.lc.opts.onInit(this.lc);
|
||
|
}
|
||
|
},
|
||
|
componentWillMount: function componentWillMount() {
|
||
|
var _this = this;
|
||
|
|
||
|
if (this.lc) return;
|
||
|
|
||
|
if (this.props.lc) {
|
||
|
this.lc = this.props.lc;
|
||
|
} else {
|
||
|
this.lc = new LiterallyCanvasModel(this.props);
|
||
|
}
|
||
|
|
||
|
this.toolButtonComponents = this.lc.opts.tools.map(function (ToolClass) {
|
||
|
return createToolButton(new ToolClass(_this.lc));
|
||
|
});
|
||
|
},
|
||
|
componentDidMount: function componentDidMount() {
|
||
|
if (!this.lc.isBound) {
|
||
|
this.bindToModel();
|
||
|
}
|
||
|
},
|
||
|
componentWillUnmount: function componentWillUnmount() {
|
||
|
if (this.lc) {
|
||
|
this.lc._teardown();
|
||
|
}
|
||
|
},
|
||
|
render: function render() {
|
||
|
var _this2 = this;
|
||
|
|
||
|
var lc = this.lc;
|
||
|
var toolButtonComponents = this.toolButtonComponents;
|
||
|
var props = this.props;
|
||
|
var _lc$opts = this.lc.opts;
|
||
|
var imageURLPrefix = _lc$opts.imageURLPrefix;
|
||
|
var toolbarPosition = _lc$opts.toolbarPosition;
|
||
|
|
||
|
|
||
|
var pickerProps = { lc: lc, toolButtonComponents: toolButtonComponents, imageURLPrefix: imageURLPrefix };
|
||
|
var topOrBottomClassName = classSet({
|
||
|
'toolbar-at-top': toolbarPosition === 'top',
|
||
|
'toolbar-at-bottom': toolbarPosition === 'bottom',
|
||
|
'toolbar-hidden': toolbarPosition === 'hidden'
|
||
|
});
|
||
|
return React.createElement(
|
||
|
'div',
|
||
|
{ className: 'literally ' + topOrBottomClassName },
|
||
|
React.createElement(CanvasContainer, { ref: function ref(item) {
|
||
|
return _this2.canvas = item;
|
||
|
} }),
|
||
|
React.createElement(Picker, pickerProps),
|
||
|
React.createElement(Options, { lc: lc, imageURLPrefix: imageURLPrefix })
|
||
|
);
|
||
|
}
|
||
|
});
|
||
|
|
||
|
module.exports = LiterallyCanvas;
|
||
|
|
||
|
},{"../core/LiterallyCanvas":5,"../core/defaultOptions":10,"../core/util":19,"../optionsStyles/font":23,"../optionsStyles/line-options-and-stroke-width":24,"../optionsStyles/null":25,"../optionsStyles/polygon-and-stroke-width":27,"../optionsStyles/stroke-width":29,"../reactGUI/React-shim":35,"../reactGUI/ReactDOM-shim":36,"./Options":33,"./Picker":34,"./createToolButton":41}],33:[function(require,module,exports){
|
||
|
var Options, React, createSetStateOnEventMixin, optionsStyles;
|
||
|
|
||
|
React = require('./React-shim');
|
||
|
|
||
|
createSetStateOnEventMixin = require('./createSetStateOnEventMixin');
|
||
|
|
||
|
optionsStyles = require('../optionsStyles/optionsStyles').optionsStyles;
|
||
|
|
||
|
Options = React.createClass({
|
||
|
displayName: 'Options',
|
||
|
getState: function() {
|
||
|
var ref;
|
||
|
return {
|
||
|
style: (ref = this.props.lc.tool) != null ? ref.optionsStyle : void 0,
|
||
|
tool: this.props.lc.tool
|
||
|
};
|
||
|
},
|
||
|
getInitialState: function() {
|
||
|
return this.getState();
|
||
|
},
|
||
|
mixins: [createSetStateOnEventMixin('toolChange')],
|
||
|
renderBody: function() {
|
||
|
var style;
|
||
|
style = "" + this.state.style;
|
||
|
return optionsStyles[style] && optionsStyles[style]({
|
||
|
lc: this.props.lc,
|
||
|
tool: this.state.tool,
|
||
|
imageURLPrefix: this.props.imageURLPrefix
|
||
|
});
|
||
|
},
|
||
|
render: function() {
|
||
|
var div;
|
||
|
div = React.DOM.div;
|
||
|
return div({
|
||
|
className: 'lc-options horz-toolbar'
|
||
|
}, this.renderBody());
|
||
|
}
|
||
|
});
|
||
|
|
||
|
module.exports = Options;
|
||
|
|
||
|
|
||
|
},{"../optionsStyles/optionsStyles":26,"./React-shim":35,"./createSetStateOnEventMixin":40}],34:[function(require,module,exports){
|
||
|
var ClearButton, ColorPickers, ColorWell, Picker, React, UndoRedoButtons, ZoomButtons, _;
|
||
|
|
||
|
React = require('./React-shim');
|
||
|
|
||
|
ClearButton = React.createFactory(require('./ClearButton'));
|
||
|
|
||
|
UndoRedoButtons = React.createFactory(require('./UndoRedoButtons'));
|
||
|
|
||
|
ZoomButtons = React.createFactory(require('./ZoomButtons'));
|
||
|
|
||
|
_ = require('../core/localization')._;
|
||
|
|
||
|
ColorWell = React.createFactory(require('./ColorWell'));
|
||
|
|
||
|
ColorPickers = React.createFactory(React.createClass({
|
||
|
displayName: 'ColorPickers',
|
||
|
render: function() {
|
||
|
var div, lc;
|
||
|
lc = this.props.lc;
|
||
|
div = React.DOM.div;
|
||
|
return div({
|
||
|
className: 'lc-color-pickers'
|
||
|
}, ColorWell({
|
||
|
lc: lc,
|
||
|
colorName: 'primary',
|
||
|
label: _('stroke')
|
||
|
}), ColorWell({
|
||
|
lc: lc,
|
||
|
colorName: 'secondary',
|
||
|
label: _('fill')
|
||
|
}), ColorWell({
|
||
|
lc: lc,
|
||
|
colorName: 'background',
|
||
|
label: _('bg')
|
||
|
}));
|
||
|
}
|
||
|
}));
|
||
|
|
||
|
Picker = React.createClass({
|
||
|
displayName: 'Picker',
|
||
|
getInitialState: function() {
|
||
|
return {
|
||
|
selectedToolIndex: 0
|
||
|
};
|
||
|
},
|
||
|
renderBody: function() {
|
||
|
var div, imageURLPrefix, lc, ref, toolButtonComponents;
|
||
|
div = React.DOM.div;
|
||
|
ref = this.props, toolButtonComponents = ref.toolButtonComponents, lc = ref.lc, imageURLPrefix = ref.imageURLPrefix;
|
||
|
return div({
|
||
|
className: 'lc-picker-contents'
|
||
|
}, toolButtonComponents.map((function(_this) {
|
||
|
return function(component, ix) {
|
||
|
return component({
|
||
|
lc: lc,
|
||
|
imageURLPrefix: imageURLPrefix,
|
||
|
key: ix,
|
||
|
isSelected: ix === _this.state.selectedToolIndex,
|
||
|
onSelect: function(tool) {
|
||
|
lc.setTool(tool);
|
||
|
return _this.setState({
|
||
|
selectedToolIndex: ix
|
||
|
});
|
||
|
}
|
||
|
});
|
||
|
};
|
||
|
})(this)), toolButtonComponents.length % 2 !== 0 ? div({
|
||
|
className: 'toolbar-button thin-button disabled'
|
||
|
}) : void 0, div({
|
||
|
style: {
|
||
|
position: 'absolute',
|
||
|
bottom: 0,
|
||
|
left: 0,
|
||
|
right: 0
|
||
|
}
|
||
|
}, ColorPickers({
|
||
|
lc: this.props.lc
|
||
|
}), UndoRedoButtons({
|
||
|
lc: lc,
|
||
|
imageURLPrefix: imageURLPrefix
|
||
|
}), ZoomButtons({
|
||
|
lc: lc,
|
||
|
imageURLPrefix: imageURLPrefix
|
||
|
}), ClearButton({
|
||
|
lc: lc
|
||
|
})));
|
||
|
},
|
||
|
render: function() {
|
||
|
var div;
|
||
|
div = React.DOM.div;
|
||
|
return div({
|
||
|
className: 'lc-picker'
|
||
|
}, this.renderBody());
|
||
|
}
|
||
|
});
|
||
|
|
||
|
module.exports = Picker;
|
||
|
|
||
|
|
||
|
},{"../core/localization":13,"./ClearButton":30,"./ColorWell":31,"./React-shim":35,"./UndoRedoButtons":38,"./ZoomButtons":39}],35:[function(require,module,exports){
|
||
|
var React, error;
|
||
|
|
||
|
try {
|
||
|
React = require('react');
|
||
|
} catch (error) {
|
||
|
React = window.React;
|
||
|
}
|
||
|
|
||
|
if (React == null) {
|
||
|
throw "Can't find React";
|
||
|
}
|
||
|
|
||
|
module.exports = React;
|
||
|
|
||
|
|
||
|
},{"react":"react"}],36:[function(require,module,exports){
|
||
|
var ReactDOM, error, error1;
|
||
|
|
||
|
try {
|
||
|
ReactDOM = require('react-dom');
|
||
|
} catch (error) {
|
||
|
ReactDOM = window.ReactDOM;
|
||
|
}
|
||
|
|
||
|
if (ReactDOM == null) {
|
||
|
try {
|
||
|
ReactDOM = require('react');
|
||
|
} catch (error1) {
|
||
|
ReactDOM = window.React;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (ReactDOM == null) {
|
||
|
throw "Can't find ReactDOM";
|
||
|
}
|
||
|
|
||
|
module.exports = ReactDOM;
|
||
|
|
||
|
|
||
|
},{"react":"react","react-dom":"react-dom"}],37:[function(require,module,exports){
|
||
|
var React, classSet, createSetStateOnEventMixin;
|
||
|
|
||
|
React = require('./React-shim');
|
||
|
|
||
|
createSetStateOnEventMixin = require('../reactGUI/createSetStateOnEventMixin');
|
||
|
|
||
|
classSet = require('../core/util').classSet;
|
||
|
|
||
|
module.exports = React.createClass({
|
||
|
displayName: 'StrokeWidthPicker',
|
||
|
getState: function(tool) {
|
||
|
if (tool == null) {
|
||
|
tool = this.props.tool;
|
||
|
}
|
||
|
return {
|
||
|
strokeWidth: tool.strokeWidth
|
||
|
};
|
||
|
},
|
||
|
getInitialState: function() {
|
||
|
return this.getState();
|
||
|
},
|
||
|
mixins: [createSetStateOnEventMixin('toolDidUpdateOptions')],
|
||
|
componentWillReceiveProps: function(props) {
|
||
|
return this.setState(this.getState(props.tool));
|
||
|
},
|
||
|
render: function() {
|
||
|
var circle, div, li, ref, strokeWidths, svg, ul;
|
||
|
ref = React.DOM, ul = ref.ul, li = ref.li, svg = ref.svg, circle = ref.circle, div = ref.div;
|
||
|
strokeWidths = this.props.lc.opts.strokeWidths;
|
||
|
return div({}, strokeWidths.map((function(_this) {
|
||
|
return function(strokeWidth, ix) {
|
||
|
var buttonClassName, buttonSize;
|
||
|
buttonClassName = classSet({
|
||
|
'square-toolbar-button': true,
|
||
|
'selected': strokeWidth === _this.state.strokeWidth
|
||
|
});
|
||
|
buttonSize = 28;
|
||
|
return div({
|
||
|
key: strokeWidth
|
||
|
}, div({
|
||
|
className: buttonClassName,
|
||
|
onClick: function() {
|
||
|
return _this.props.lc.trigger('setStrokeWidth', strokeWidth);
|
||
|
}
|
||
|
}, svg({
|
||
|
width: buttonSize - 2,
|
||
|
height: buttonSize - 2,
|
||
|
viewPort: "0 0 " + strokeWidth + " " + strokeWidth,
|
||
|
version: "1.1",
|
||
|
xmlns: "http://www.w3.org/2000/svg"
|
||
|
}, circle({
|
||
|
cx: Math.ceil(buttonSize / 2 - 1),
|
||
|
cy: Math.ceil(buttonSize / 2 - 1),
|
||
|
r: strokeWidth / 2
|
||
|
}))));
|
||
|
};
|
||
|
})(this)));
|
||
|
}
|
||
|
});
|
||
|
|
||
|
|
||
|
},{"../core/util":19,"../reactGUI/createSetStateOnEventMixin":40,"./React-shim":35}],38:[function(require,module,exports){
|
||
|
var React, RedoButton, UndoButton, UndoRedoButtons, classSet, createSetStateOnEventMixin, createUndoRedoButtonComponent;
|
||
|
|
||
|
React = require('./React-shim');
|
||
|
|
||
|
createSetStateOnEventMixin = require('./createSetStateOnEventMixin');
|
||
|
|
||
|
classSet = require('../core/util').classSet;
|
||
|
|
||
|
createUndoRedoButtonComponent = function(undoOrRedo) {
|
||
|
return React.createClass({
|
||
|
displayName: undoOrRedo === 'undo' ? 'UndoButton' : 'RedoButton',
|
||
|
getState: function() {
|
||
|
return {
|
||
|
isEnabled: (function() {
|
||
|
switch (false) {
|
||
|
case undoOrRedo !== 'undo':
|
||
|
return this.props.lc.canUndo();
|
||
|
case undoOrRedo !== 'redo':
|
||
|
return this.props.lc.canRedo();
|
||
|
}
|
||
|
}).call(this)
|
||
|
};
|
||
|
},
|
||
|
getInitialState: function() {
|
||
|
return this.getState();
|
||
|
},
|
||
|
mixins: [createSetStateOnEventMixin('drawingChange')],
|
||
|
render: function() {
|
||
|
var className, div, imageURLPrefix, img, lc, onClick, ref, ref1, src, style, title;
|
||
|
ref = React.DOM, div = ref.div, img = ref.img;
|
||
|
ref1 = this.props, lc = ref1.lc, imageURLPrefix = ref1.imageURLPrefix;
|
||
|
title = undoOrRedo === 'undo' ? 'Undo' : 'Redo';
|
||
|
className = ("lc-" + undoOrRedo + " ") + classSet({
|
||
|
'toolbar-button': true,
|
||
|
'thin-button': true,
|
||
|
'disabled': !this.state.isEnabled
|
||
|
});
|
||
|
onClick = (function() {
|
||
|
switch (false) {
|
||
|
case !!this.state.isEnabled:
|
||
|
return function() {};
|
||
|
case undoOrRedo !== 'undo':
|
||
|
return function() {
|
||
|
return lc.undo();
|
||
|
};
|
||
|
case undoOrRedo !== 'redo':
|
||
|
return function() {
|
||
|
return lc.redo();
|
||
|
};
|
||
|
}
|
||
|
}).call(this);
|
||
|
src = imageURLPrefix + "/" + undoOrRedo + ".png";
|
||
|
style = {
|
||
|
backgroundImage: "url(" + src + ")"
|
||
|
};
|
||
|
return div({
|
||
|
className: className,
|
||
|
onClick: onClick,
|
||
|
title: title,
|
||
|
style: style
|
||
|
});
|
||
|
}
|
||
|
});
|
||
|
};
|
||
|
|
||
|
UndoButton = React.createFactory(createUndoRedoButtonComponent('undo'));
|
||
|
|
||
|
RedoButton = React.createFactory(createUndoRedoButtonComponent('redo'));
|
||
|
|
||
|
UndoRedoButtons = React.createClass({
|
||
|
displayName: 'UndoRedoButtons',
|
||
|
render: function() {
|
||
|
var div;
|
||
|
div = React.DOM.div;
|
||
|
return div({
|
||
|
className: 'lc-undo-redo'
|
||
|
}, UndoButton(this.props), RedoButton(this.props));
|
||
|
}
|
||
|
});
|
||
|
|
||
|
module.exports = UndoRedoButtons;
|
||
|
|
||
|
|
||
|
},{"../core/util":19,"./React-shim":35,"./createSetStateOnEventMixin":40}],39:[function(require,module,exports){
|
||
|
var React, ZoomButtons, ZoomInButton, ZoomOutButton, classSet, createSetStateOnEventMixin, createZoomButtonComponent;
|
||
|
|
||
|
React = require('./React-shim');
|
||
|
|
||
|
createSetStateOnEventMixin = require('./createSetStateOnEventMixin');
|
||
|
|
||
|
classSet = require('../core/util').classSet;
|
||
|
|
||
|
createZoomButtonComponent = function(inOrOut) {
|
||
|
return React.createClass({
|
||
|
displayName: inOrOut === 'in' ? 'ZoomInButton' : 'ZoomOutButton',
|
||
|
getState: function() {
|
||
|
return {
|
||
|
isEnabled: (function() {
|
||
|
switch (false) {
|
||
|
case inOrOut !== 'in':
|
||
|
return this.props.lc.scale < this.props.lc.config.zoomMax;
|
||
|
case inOrOut !== 'out':
|
||
|
return this.props.lc.scale > this.props.lc.config.zoomMin;
|
||
|
}
|
||
|
}).call(this)
|
||
|
};
|
||
|
},
|
||
|
getInitialState: function() {
|
||
|
return this.getState();
|
||
|
},
|
||
|
mixins: [createSetStateOnEventMixin('zoom')],
|
||
|
render: function() {
|
||
|
var className, div, imageURLPrefix, img, lc, onClick, ref, ref1, src, style, title;
|
||
|
ref = React.DOM, div = ref.div, img = ref.img;
|
||
|
ref1 = this.props, lc = ref1.lc, imageURLPrefix = ref1.imageURLPrefix;
|
||
|
title = inOrOut === 'in' ? 'Zoom in' : 'Zoom out';
|
||
|
className = ("lc-zoom-" + inOrOut + " ") + classSet({
|
||
|
'toolbar-button': true,
|
||
|
'thin-button': true,
|
||
|
'disabled': !this.state.isEnabled
|
||
|
});
|
||
|
onClick = (function() {
|
||
|
switch (false) {
|
||
|
case !!this.state.isEnabled:
|
||
|
return function() {};
|
||
|
case inOrOut !== 'in':
|
||
|
return function() {
|
||
|
return lc.zoom(lc.config.zoomStep);
|
||
|
};
|
||
|
case inOrOut !== 'out':
|
||
|
return function() {
|
||
|
return lc.zoom(-lc.config.zoomStep);
|
||
|
};
|
||
|
}
|
||
|
}).call(this);
|
||
|
src = imageURLPrefix + "/zoom-" + inOrOut + ".png";
|
||
|
style = {
|
||
|
backgroundImage: "url(" + src + ")"
|
||
|
};
|
||
|
return div({
|
||
|
className: className,
|
||
|
onClick: onClick,
|
||
|
title: title,
|
||
|
style: style
|
||
|
});
|
||
|
}
|
||
|
});
|
||
|
};
|
||
|
|
||
|
ZoomOutButton = React.createFactory(createZoomButtonComponent('out'));
|
||
|
|
||
|
ZoomInButton = React.createFactory(createZoomButtonComponent('in'));
|
||
|
|
||
|
ZoomButtons = React.createClass({
|
||
|
displayName: 'ZoomButtons',
|
||
|
render: function() {
|
||
|
var div;
|
||
|
div = React.DOM.div;
|
||
|
return div({
|
||
|
className: 'lc-zoom'
|
||
|
}, ZoomOutButton(this.props), ZoomInButton(this.props));
|
||
|
}
|
||
|
});
|
||
|
|
||
|
module.exports = ZoomButtons;
|
||
|
|
||
|
|
||
|
},{"../core/util":19,"./React-shim":35,"./createSetStateOnEventMixin":40}],40:[function(require,module,exports){
|
||
|
var React, createSetStateOnEventMixin;
|
||
|
|
||
|
React = require('./React-shim');
|
||
|
|
||
|
module.exports = createSetStateOnEventMixin = function(eventName) {
|
||
|
return {
|
||
|
componentDidMount: function() {
|
||
|
return this.unsubscribe = this.props.lc.on(eventName, (function(_this) {
|
||
|
return function() {
|
||
|
return _this.setState(_this.getState());
|
||
|
};
|
||
|
})(this));
|
||
|
},
|
||
|
componentWillUnmount: function() {
|
||
|
return this.unsubscribe();
|
||
|
}
|
||
|
};
|
||
|
};
|
||
|
|
||
|
|
||
|
},{"./React-shim":35}],41:[function(require,module,exports){
|
||
|
var React, _, classSet, createToolButton;
|
||
|
|
||
|
React = require('./React-shim');
|
||
|
|
||
|
classSet = require('../core/util').classSet;
|
||
|
|
||
|
_ = require('../core/localization')._;
|
||
|
|
||
|
createToolButton = function(tool) {
|
||
|
var displayName, imageName;
|
||
|
displayName = tool.name;
|
||
|
imageName = tool.iconName;
|
||
|
return React.createFactory(React.createClass({
|
||
|
displayName: displayName,
|
||
|
getDefaultProps: function() {
|
||
|
return {
|
||
|
isSelected: false,
|
||
|
lc: null
|
||
|
};
|
||
|
},
|
||
|
componentWillMount: function() {
|
||
|
if (this.props.isSelected) {
|
||
|
return this.props.lc.setTool(tool);
|
||
|
}
|
||
|
},
|
||
|
render: function() {
|
||
|
var className, div, imageURLPrefix, img, isSelected, onSelect, ref, ref1, src;
|
||
|
ref = React.DOM, div = ref.div, img = ref.img;
|
||
|
ref1 = this.props, imageURLPrefix = ref1.imageURLPrefix, isSelected = ref1.isSelected, onSelect = ref1.onSelect;
|
||
|
className = classSet({
|
||
|
'lc-pick-tool': true,
|
||
|
'toolbar-button': true,
|
||
|
'thin-button': true,
|
||
|
'selected': isSelected
|
||
|
});
|
||
|
src = imageURLPrefix + "/" + imageName + ".png";
|
||
|
return div({
|
||
|
className: className,
|
||
|
style: {
|
||
|
'backgroundImage': "url(" + src + ")"
|
||
|
},
|
||
|
onClick: (function() {
|
||
|
return onSelect(tool);
|
||
|
}),
|
||
|
title: _(displayName)
|
||
|
});
|
||
|
}
|
||
|
}));
|
||
|
};
|
||
|
|
||
|
module.exports = createToolButton;
|
||
|
|
||
|
|
||
|
},{"../core/localization":13,"../core/util":19,"./React-shim":35}],42:[function(require,module,exports){
|
||
|
'use strict';
|
||
|
|
||
|
var React = require('./React-shim');
|
||
|
var ReactDOM = require('./ReactDOM-shim');
|
||
|
var LiterallyCanvasModel = require('../core/LiterallyCanvas');
|
||
|
var LiterallyCanvasReactComponent = require('./LiterallyCanvas');
|
||
|
|
||
|
function init(el, opts) {
|
||
|
var originalClassName = el.className;
|
||
|
var lc = new LiterallyCanvasModel(opts);
|
||
|
ReactDOM.render(React.createElement(LiterallyCanvasReactComponent, { lc: lc }), el);
|
||
|
lc.teardown = function () {
|
||
|
lc._teardown();
|
||
|
for (var i = 0; i < el.children.length; i++) {
|
||
|
el.removeChild(el.children[i]);
|
||
|
}
|
||
|
el.className = originalClassName;
|
||
|
};
|
||
|
return lc;
|
||
|
}
|
||
|
|
||
|
module.exports = init;
|
||
|
|
||
|
},{"../core/LiterallyCanvas":5,"./LiterallyCanvas":32,"./React-shim":35,"./ReactDOM-shim":36}],43:[function(require,module,exports){
|
||
|
var Ellipse, ToolWithStroke, createShape,
|
||
|
extend = function(child, parent) { for (var key in parent) { if (hasProp.call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; },
|
||
|
hasProp = {}.hasOwnProperty;
|
||
|
|
||
|
ToolWithStroke = require('./base').ToolWithStroke;
|
||
|
|
||
|
createShape = require('../core/shapes').createShape;
|
||
|
|
||
|
module.exports = Ellipse = (function(superClass) {
|
||
|
extend(Ellipse, superClass);
|
||
|
|
||
|
function Ellipse() {
|
||
|
return Ellipse.__super__.constructor.apply(this, arguments);
|
||
|
}
|
||
|
|
||
|
Ellipse.prototype.name = 'Ellipse';
|
||
|
|
||
|
Ellipse.prototype.iconName = 'ellipse';
|
||
|
|
||
|
Ellipse.prototype.begin = function(x, y, lc) {
|
||
|
return this.currentShape = createShape('Ellipse', {
|
||
|
x: x,
|
||
|
y: y,
|
||
|
strokeWidth: this.strokeWidth,
|
||
|
strokeColor: lc.getColor('primary'),
|
||
|
fillColor: lc.getColor('secondary')
|
||
|
});
|
||
|
};
|
||
|
|
||
|
Ellipse.prototype["continue"] = function(x, y, lc) {
|
||
|
this.currentShape.width = x - this.currentShape.x;
|
||
|
this.currentShape.height = y - this.currentShape.y;
|
||
|
return lc.drawShapeInProgress(this.currentShape);
|
||
|
};
|
||
|
|
||
|
Ellipse.prototype.end = function(x, y, lc) {
|
||
|
return lc.saveShape(this.currentShape);
|
||
|
};
|
||
|
|
||
|
return Ellipse;
|
||
|
|
||
|
})(ToolWithStroke);
|
||
|
|
||
|
|
||
|
},{"../core/shapes":17,"./base":53}],44:[function(require,module,exports){
|
||
|
var Eraser, Pencil, createShape,
|
||
|
extend = function(child, parent) { for (var key in parent) { if (hasProp.call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; },
|
||
|
hasProp = {}.hasOwnProperty;
|
||
|
|
||
|
Pencil = require('./Pencil');
|
||
|
|
||
|
createShape = require('../core/shapes').createShape;
|
||
|
|
||
|
module.exports = Eraser = (function(superClass) {
|
||
|
extend(Eraser, superClass);
|
||
|
|
||
|
function Eraser() {
|
||
|
return Eraser.__super__.constructor.apply(this, arguments);
|
||
|
}
|
||
|
|
||
|
Eraser.prototype.name = 'Eraser';
|
||
|
|
||
|
Eraser.prototype.iconName = 'eraser';
|
||
|
|
||
|
Eraser.prototype.makePoint = function(x, y, lc) {
|
||
|
return createShape('Point', {
|
||
|
x: x,
|
||
|
y: y,
|
||
|
size: this.strokeWidth,
|
||
|
color: '#000'
|
||
|
});
|
||
|
};
|
||
|
|
||
|
Eraser.prototype.makeShape = function() {
|
||
|
return createShape('ErasedLinePath');
|
||
|
};
|
||
|
|
||
|
return Eraser;
|
||
|
|
||
|
})(Pencil);
|
||
|
|
||
|
|
||
|
},{"../core/shapes":17,"./Pencil":48}],45:[function(require,module,exports){
|
||
|
var Eyedropper, Tool, getPixel,
|
||
|
extend = function(child, parent) { for (var key in parent) { if (hasProp.call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; },
|
||
|
hasProp = {}.hasOwnProperty;
|
||
|
|
||
|
Tool = require('./base').Tool;
|
||
|
|
||
|
getPixel = function(ctx, arg) {
|
||
|
var pixel, x, y;
|
||
|
x = arg.x, y = arg.y;
|
||
|
pixel = ctx.getImageData(x, y, 1, 1).data;
|
||
|
if (pixel[3]) {
|
||
|
return "rgb(" + pixel[0] + ", " + pixel[1] + ", " + pixel[2] + ")";
|
||
|
} else {
|
||
|
return null;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
module.exports = Eyedropper = (function(superClass) {
|
||
|
extend(Eyedropper, superClass);
|
||
|
|
||
|
Eyedropper.prototype.name = 'Eyedropper';
|
||
|
|
||
|
Eyedropper.prototype.iconName = 'eyedropper';
|
||
|
|
||
|
Eyedropper.prototype.optionsStyle = 'stroke-or-fill';
|
||
|
|
||
|
function Eyedropper(lc) {
|
||
|
Eyedropper.__super__.constructor.call(this, lc);
|
||
|
this.strokeOrFill = 'stroke';
|
||
|
}
|
||
|
|
||
|
Eyedropper.prototype.readColor = function(x, y, lc) {
|
||
|
var canvas, color, newColor, offset;
|
||
|
offset = lc.getDefaultImageRect();
|
||
|
canvas = lc.getImage();
|
||
|
newColor = getPixel(canvas.getContext('2d'), {
|
||
|
x: x - offset.x,
|
||
|
y: y - offset.y
|
||
|
});
|
||
|
color = newColor || lc.getColor('background');
|
||
|
if (this.strokeOrFill === 'stroke') {
|
||
|
return lc.setColor('primary', newColor);
|
||
|
} else {
|
||
|
return lc.setColor('secondary', newColor);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
Eyedropper.prototype.begin = function(x, y, lc) {
|
||
|
return this.readColor(x, y, lc);
|
||
|
};
|
||
|
|
||
|
Eyedropper.prototype["continue"] = function(x, y, lc) {
|
||
|
return this.readColor(x, y, lc);
|
||
|
};
|
||
|
|
||
|
return Eyedropper;
|
||
|
|
||
|
})(Tool);
|
||
|
|
||
|
|
||
|
},{"./base":53}],46:[function(require,module,exports){
|
||
|
var Line, ToolWithStroke, createShape,
|
||
|
extend = function(child, parent) { for (var key in parent) { if (hasProp.call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; },
|
||
|
hasProp = {}.hasOwnProperty;
|
||
|
|
||
|
ToolWithStroke = require('./base').ToolWithStroke;
|
||
|
|
||
|
createShape = require('../core/shapes').createShape;
|
||
|
|
||
|
module.exports = Line = (function(superClass) {
|
||
|
extend(Line, superClass);
|
||
|
|
||
|
function Line() {
|
||
|
return Line.__super__.constructor.apply(this, arguments);
|
||
|
}
|
||
|
|
||
|
Line.prototype.name = 'Line';
|
||
|
|
||
|
Line.prototype.iconName = 'line';
|
||
|
|
||
|
Line.prototype.optionsStyle = 'line-options-and-stroke-width';
|
||
|
|
||
|
Line.prototype.begin = function(x, y, lc) {
|
||
|
return this.currentShape = createShape('Line', {
|
||
|
x1: x,
|
||
|
y1: y,
|
||
|
x2: x,
|
||
|
y2: y,
|
||
|
strokeWidth: this.strokeWidth,
|
||
|
dash: (function() {
|
||
|
switch (false) {
|
||
|
case !this.isDashed:
|
||
|
return [this.strokeWidth * 2, this.strokeWidth * 4];
|
||
|
default:
|
||
|
return null;
|
||
|
}
|
||
|
}).call(this),
|
||
|
endCapShapes: this.hasEndArrow ? [null, 'arrow'] : null,
|
||
|
color: lc.getColor('primary')
|
||
|
});
|
||
|
};
|
||
|
|
||
|
Line.prototype["continue"] = function(x, y, lc) {
|
||
|
this.currentShape.x2 = x;
|
||
|
this.currentShape.y2 = y;
|
||
|
return lc.drawShapeInProgress(this.currentShape);
|
||
|
};
|
||
|
|
||
|
Line.prototype.end = function(x, y, lc) {
|
||
|
return lc.saveShape(this.currentShape);
|
||
|
};
|
||
|
|
||
|
return Line;
|
||
|
|
||
|
})(ToolWithStroke);
|
||
|
|
||
|
|
||
|
},{"../core/shapes":17,"./base":53}],47:[function(require,module,exports){
|
||
|
var Pan, Tool, createShape,
|
||
|
extend = function(child, parent) { for (var key in parent) { if (hasProp.call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; },
|
||
|
hasProp = {}.hasOwnProperty;
|
||
|
|
||
|
Tool = require('./base').Tool;
|
||
|
|
||
|
createShape = require('../core/shapes').createShape;
|
||
|
|
||
|
module.exports = Pan = (function(superClass) {
|
||
|
extend(Pan, superClass);
|
||
|
|
||
|
function Pan() {
|
||
|
return Pan.__super__.constructor.apply(this, arguments);
|
||
|
}
|
||
|
|
||
|
Pan.prototype.name = 'Pan';
|
||
|
|
||
|
Pan.prototype.iconName = 'pan';
|
||
|
|
||
|
Pan.prototype.usesSimpleAPI = false;
|
||
|
|
||
|
Pan.prototype.didBecomeActive = function(lc) {
|
||
|
var unsubscribeFuncs;
|
||
|
unsubscribeFuncs = [];
|
||
|
this.unsubscribe = (function(_this) {
|
||
|
return function() {
|
||
|
var func, i, len, results;
|
||
|
results = [];
|
||
|
for (i = 0, len = unsubscribeFuncs.length; i < len; i++) {
|
||
|
func = unsubscribeFuncs[i];
|
||
|
results.push(func());
|
||
|
}
|
||
|
return results;
|
||
|
};
|
||
|
})(this);
|
||
|
unsubscribeFuncs.push(lc.on('lc-pointerdown', (function(_this) {
|
||
|
return function(arg) {
|
||
|
var rawX, rawY;
|
||
|
rawX = arg.rawX, rawY = arg.rawY;
|
||
|
_this.oldPosition = lc.position;
|
||
|
return _this.pointerStart = {
|
||
|
x: rawX,
|
||
|
y: rawY
|
||
|
};
|
||
|
};
|
||
|
})(this)));
|
||
|
return unsubscribeFuncs.push(lc.on('lc-pointerdrag', (function(_this) {
|
||
|
return function(arg) {
|
||
|
var dp, rawX, rawY;
|
||
|
rawX = arg.rawX, rawY = arg.rawY;
|
||
|
dp = {
|
||
|
x: (rawX - _this.pointerStart.x) * lc.backingScale,
|
||
|
y: (rawY - _this.pointerStart.y) * lc.backingScale
|
||
|
};
|
||
|
return lc.setPan(_this.oldPosition.x + dp.x, _this.oldPosition.y + dp.y);
|
||
|
};
|
||
|
})(this)));
|
||
|
};
|
||
|
|
||
|
Pan.prototype.willBecomeInactive = function(lc) {
|
||
|
return this.unsubscribe();
|
||
|
};
|
||
|
|
||
|
return Pan;
|
||
|
|
||
|
})(Tool);
|
||
|
|
||
|
|
||
|
},{"../core/shapes":17,"./base":53}],48:[function(require,module,exports){
|
||
|
var Pencil, ToolWithStroke, createShape,
|
||
|
extend = function(child, parent) { for (var key in parent) { if (hasProp.call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; },
|
||
|
hasProp = {}.hasOwnProperty;
|
||
|
|
||
|
ToolWithStroke = require('./base').ToolWithStroke;
|
||
|
|
||
|
createShape = require('../core/shapes').createShape;
|
||
|
|
||
|
module.exports = Pencil = (function(superClass) {
|
||
|
extend(Pencil, superClass);
|
||
|
|
||
|
function Pencil() {
|
||
|
return Pencil.__super__.constructor.apply(this, arguments);
|
||
|
}
|
||
|
|
||
|
Pencil.prototype.name = 'Pencil';
|
||
|
|
||
|
Pencil.prototype.iconName = 'pencil';
|
||
|
|
||
|
Pencil.prototype.eventTimeThreshold = 10;
|
||
|
|
||
|
Pencil.prototype.begin = function(x, y, lc) {
|
||
|
this.color = lc.getColor('primary');
|
||
|
this.currentShape = this.makeShape();
|
||
|
this.currentShape.addPoint(this.makePoint(x, y, lc));
|
||
|
return this.lastEventTime = Date.now();
|
||
|
};
|
||
|
|
||
|
Pencil.prototype["continue"] = function(x, y, lc) {
|
||
|
var timeDiff;
|
||
|
timeDiff = Date.now() - this.lastEventTime;
|
||
|
if (timeDiff > this.eventTimeThreshold) {
|
||
|
this.lastEventTime += timeDiff;
|
||
|
this.currentShape.addPoint(this.makePoint(x, y, lc));
|
||
|
return lc.drawShapeInProgress(this.currentShape);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
Pencil.prototype.end = function(x, y, lc) {
|
||
|
lc.saveShape(this.currentShape);
|
||
|
return this.currentShape = void 0;
|
||
|
};
|
||
|
|
||
|
Pencil.prototype.makePoint = function(x, y, lc) {
|
||
|
return createShape('Point', {
|
||
|
x: x,
|
||
|
y: y,
|
||
|
size: this.strokeWidth,
|
||
|
color: this.color
|
||
|
});
|
||
|
};
|
||
|
|
||
|
Pencil.prototype.makeShape = function() {
|
||
|
return createShape('LinePath');
|
||
|
};
|
||
|
|
||
|
return Pencil;
|
||
|
|
||
|
})(ToolWithStroke);
|
||
|
|
||
|
|
||
|
},{"../core/shapes":17,"./base":53}],49:[function(require,module,exports){
|
||
|
var Polygon, ToolWithStroke, createShape,
|
||
|
extend = function(child, parent) { for (var key in parent) { if (hasProp.call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; },
|
||
|
hasProp = {}.hasOwnProperty;
|
||
|
|
||
|
ToolWithStroke = require('./base').ToolWithStroke;
|
||
|
|
||
|
createShape = require('../core/shapes').createShape;
|
||
|
|
||
|
module.exports = Polygon = (function(superClass) {
|
||
|
extend(Polygon, superClass);
|
||
|
|
||
|
function Polygon() {
|
||
|
return Polygon.__super__.constructor.apply(this, arguments);
|
||
|
}
|
||
|
|
||
|
Polygon.prototype.name = 'Polygon';
|
||
|
|
||
|
Polygon.prototype.iconName = 'polygon';
|
||
|
|
||
|
Polygon.prototype.usesSimpleAPI = false;
|
||
|
|
||
|
Polygon.prototype.didBecomeActive = function(lc) {
|
||
|
var onDown, onMove, onUp, polygonCancel, polygonFinishClosed, polygonFinishOpen, polygonUnsubscribeFuncs;
|
||
|
Polygon.__super__.didBecomeActive.call(this, lc);
|
||
|
polygonUnsubscribeFuncs = [];
|
||
|
this.polygonUnsubscribe = (function(_this) {
|
||
|
return function() {
|
||
|
var func, i, len, results;
|
||
|
results = [];
|
||
|
for (i = 0, len = polygonUnsubscribeFuncs.length; i < len; i++) {
|
||
|
func = polygonUnsubscribeFuncs[i];
|
||
|
results.push(func());
|
||
|
}
|
||
|
return results;
|
||
|
};
|
||
|
})(this);
|
||
|
this.points = null;
|
||
|
this.maybePoint = null;
|
||
|
onUp = (function(_this) {
|
||
|
return function() {
|
||
|
if (_this._getWillFinish()) {
|
||
|
return _this._close(lc);
|
||
|
}
|
||
|
lc.trigger('lc-polygon-started');
|
||
|
if (_this.points) {
|
||
|
_this.points.push(_this.maybePoint);
|
||
|
} else {
|
||
|
_this.points = [_this.maybePoint];
|
||
|
}
|
||
|
_this.maybePoint = {
|
||
|
x: _this.maybePoint.x,
|
||
|
y: _this.maybePoint.y
|
||
|
};
|
||
|
lc.setShapesInProgress(_this._getShapes(lc));
|
||
|
return lc.repaintLayer('main');
|
||
|
};
|
||
|
})(this);
|
||
|
onMove = (function(_this) {
|
||
|
return function(arg) {
|
||
|
var x, y;
|
||
|
x = arg.x, y = arg.y;
|
||
|
if (_this.maybePoint) {
|
||
|
_this.maybePoint.x = x;
|
||
|
_this.maybePoint.y = y;
|
||
|
lc.setShapesInProgress(_this._getShapes(lc));
|
||
|
return lc.repaintLayer('main');
|
||
|
}
|
||
|
};
|
||
|
})(this);
|
||
|
onDown = (function(_this) {
|
||
|
return function(arg) {
|
||
|
var x, y;
|
||
|
x = arg.x, y = arg.y;
|
||
|
_this.maybePoint = {
|
||
|
x: x,
|
||
|
y: y
|
||
|
};
|
||
|
lc.setShapesInProgress(_this._getShapes(lc));
|
||
|
return lc.repaintLayer('main');
|
||
|
};
|
||
|
})(this);
|
||
|
polygonFinishOpen = (function(_this) {
|
||
|
return function() {
|
||
|
_this.maybePoint = {
|
||
|
x: Infinity,
|
||
|
y: Infinity
|
||
|
};
|
||
|
return _this._close(lc);
|
||
|
};
|
||
|
})(this);
|
||
|
polygonFinishClosed = (function(_this) {
|
||
|
return function() {
|
||
|
_this.maybePoint = _this.points[0];
|
||
|
return _this._close(lc);
|
||
|
};
|
||
|
})(this);
|
||
|
polygonCancel = (function(_this) {
|
||
|
return function() {
|
||
|
return _this._cancel(lc);
|
||
|
};
|
||
|
})(this);
|
||
|
polygonUnsubscribeFuncs.push(lc.on('drawingChange', (function(_this) {
|
||
|
return function() {
|
||
|
return _this._cancel(lc);
|
||
|
};
|
||
|
})(this)));
|
||
|
polygonUnsubscribeFuncs.push(lc.on('lc-pointerdown', onDown));
|
||
|
polygonUnsubscribeFuncs.push(lc.on('lc-pointerdrag', onMove));
|
||
|
polygonUnsubscribeFuncs.push(lc.on('lc-pointermove', onMove));
|
||
|
polygonUnsubscribeFuncs.push(lc.on('lc-pointerup', onUp));
|
||
|
polygonUnsubscribeFuncs.push(lc.on('lc-polygon-finishopen', polygonFinishOpen));
|
||
|
polygonUnsubscribeFuncs.push(lc.on('lc-polygon-finishclosed', polygonFinishClosed));
|
||
|
return polygonUnsubscribeFuncs.push(lc.on('lc-polygon-cancel', polygonCancel));
|
||
|
};
|
||
|
|
||
|
Polygon.prototype.willBecomeInactive = function(lc) {
|
||
|
Polygon.__super__.willBecomeInactive.call(this, lc);
|
||
|
if (this.points || this.maybePoint) {
|
||
|
this._cancel(lc);
|
||
|
}
|
||
|
return this.polygonUnsubscribe();
|
||
|
};
|
||
|
|
||
|
Polygon.prototype._getArePointsClose = function(a, b) {
|
||
|
return (Math.abs(a.x - b.x) + Math.abs(a.y - b.y)) < 10;
|
||
|
};
|
||
|
|
||
|
Polygon.prototype._getWillClose = function() {
|
||
|
if (!(this.points && this.points.length > 1)) {
|
||
|
return false;
|
||
|
}
|
||
|
if (!this.maybePoint) {
|
||
|
return false;
|
||
|
}
|
||
|
return this._getArePointsClose(this.points[0], this.maybePoint);
|
||
|
};
|
||
|
|
||
|
Polygon.prototype._getWillFinish = function() {
|
||
|
if (!(this.points && this.points.length > 1)) {
|
||
|
return false;
|
||
|
}
|
||
|
if (!this.maybePoint) {
|
||
|
return false;
|
||
|
}
|
||
|
return this._getArePointsClose(this.points[0], this.maybePoint) || this._getArePointsClose(this.points[this.points.length - 1], this.maybePoint);
|
||
|
};
|
||
|
|
||
|
Polygon.prototype._cancel = function(lc) {
|
||
|
lc.trigger('lc-polygon-stopped');
|
||
|
this.maybePoint = null;
|
||
|
this.points = null;
|
||
|
lc.setShapesInProgress([]);
|
||
|
return lc.repaintLayer('main');
|
||
|
};
|
||
|
|
||
|
Polygon.prototype._close = function(lc) {
|
||
|
lc.trigger('lc-polygon-stopped');
|
||
|
lc.setShapesInProgress([]);
|
||
|
if (this.points.length > 2) {
|
||
|
lc.saveShape(this._getShape(lc, false));
|
||
|
}
|
||
|
this.maybePoint = null;
|
||
|
return this.points = null;
|
||
|
};
|
||
|
|
||
|
Polygon.prototype._getShapes = function(lc, isInProgress) {
|
||
|
var shape;
|
||
|
if (isInProgress == null) {
|
||
|
isInProgress = true;
|
||
|
}
|
||
|
shape = this._getShape(lc, isInProgress);
|
||
|
if (shape) {
|
||
|
return [shape];
|
||
|
} else {
|
||
|
return [];
|
||
|
}
|
||
|
};
|
||
|
|
||
|
Polygon.prototype._getShape = function(lc, isInProgress) {
|
||
|
var points;
|
||
|
if (isInProgress == null) {
|
||
|
isInProgress = true;
|
||
|
}
|
||
|
points = [];
|
||
|
if (this.points) {
|
||
|
points = points.concat(this.points);
|
||
|
}
|
||
|
if ((!isInProgress) && points.length < 3) {
|
||
|
return null;
|
||
|
}
|
||
|
if (isInProgress && this.maybePoint) {
|
||
|
points.push(this.maybePoint);
|
||
|
}
|
||
|
if (points.length > 1) {
|
||
|
return createShape('Polygon', {
|
||
|
isClosed: this._getWillClose(),
|
||
|
strokeColor: lc.getColor('primary'),
|
||
|
fillColor: lc.getColor('secondary'),
|
||
|
strokeWidth: this.strokeWidth,
|
||
|
points: points.map(function(xy) {
|
||
|
return createShape('Point', xy);
|
||
|
})
|
||
|
});
|
||
|
} else {
|
||
|
return null;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
Polygon.prototype.optionsStyle = 'polygon-and-stroke-width';
|
||
|
|
||
|
return Polygon;
|
||
|
|
||
|
})(ToolWithStroke);
|
||
|
|
||
|
|
||
|
},{"../core/shapes":17,"./base":53}],50:[function(require,module,exports){
|
||
|
var Rectangle, ToolWithStroke, createShape,
|
||
|
extend = function(child, parent) { for (var key in parent) { if (hasProp.call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; },
|
||
|
hasProp = {}.hasOwnProperty;
|
||
|
|
||
|
ToolWithStroke = require('./base').ToolWithStroke;
|
||
|
|
||
|
createShape = require('../core/shapes').createShape;
|
||
|
|
||
|
module.exports = Rectangle = (function(superClass) {
|
||
|
extend(Rectangle, superClass);
|
||
|
|
||
|
function Rectangle() {
|
||
|
return Rectangle.__super__.constructor.apply(this, arguments);
|
||
|
}
|
||
|
|
||
|
Rectangle.prototype.name = 'Rectangle';
|
||
|
|
||
|
Rectangle.prototype.iconName = 'rectangle';
|
||
|
|
||
|
Rectangle.prototype.begin = function(x, y, lc) {
|
||
|
return this.currentShape = createShape('Rectangle', {
|
||
|
x: x,
|
||
|
y: y,
|
||
|
strokeWidth: this.strokeWidth,
|
||
|
strokeColor: lc.getColor('primary'),
|
||
|
fillColor: lc.getColor('secondary')
|
||
|
});
|
||
|
};
|
||
|
|
||
|
Rectangle.prototype["continue"] = function(x, y, lc) {
|
||
|
this.currentShape.width = x - this.currentShape.x;
|
||
|
this.currentShape.height = y - this.currentShape.y;
|
||
|
return lc.drawShapeInProgress(this.currentShape);
|
||
|
};
|
||
|
|
||
|
Rectangle.prototype.end = function(x, y, lc) {
|
||
|
return lc.saveShape(this.currentShape);
|
||
|
};
|
||
|
|
||
|
return Rectangle;
|
||
|
|
||
|
})(ToolWithStroke);
|
||
|
|
||
|
|
||
|
},{"../core/shapes":17,"./base":53}],51:[function(require,module,exports){
|
||
|
var SelectShape, Tool, createShape,
|
||
|
extend = function(child, parent) { for (var key in parent) { if (hasProp.call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; },
|
||
|
hasProp = {}.hasOwnProperty;
|
||
|
|
||
|
Tool = require('./base').Tool;
|
||
|
|
||
|
createShape = require('../core/shapes').createShape;
|
||
|
|
||
|
module.exports = SelectShape = (function(superClass) {
|
||
|
extend(SelectShape, superClass);
|
||
|
|
||
|
SelectShape.prototype.name = 'SelectShape';
|
||
|
|
||
|
SelectShape.prototype.usesSimpleAPI = false;
|
||
|
|
||
|
function SelectShape(lc) {
|
||
|
this.selectCanvas = document.createElement('canvas');
|
||
|
this.selectCanvas.style['background-color'] = 'transparent';
|
||
|
this.selectCtx = this.selectCanvas.getContext('2d');
|
||
|
}
|
||
|
|
||
|
SelectShape.prototype.didBecomeActive = function(lc) {
|
||
|
var onDown, onDrag, onUp, selectShapeUnsubscribeFuncs;
|
||
|
selectShapeUnsubscribeFuncs = [];
|
||
|
this._selectShapeUnsubscribe = (function(_this) {
|
||
|
return function() {
|
||
|
var func, j, len, results;
|
||
|
results = [];
|
||
|
for (j = 0, len = selectShapeUnsubscribeFuncs.length; j < len; j++) {
|
||
|
func = selectShapeUnsubscribeFuncs[j];
|
||
|
results.push(func());
|
||
|
}
|
||
|
return results;
|
||
|
};
|
||
|
})(this);
|
||
|
onDown = (function(_this) {
|
||
|
return function(arg) {
|
||
|
var br, shapeIndex, x, y;
|
||
|
x = arg.x, y = arg.y;
|
||
|
_this.didDrag = false;
|
||
|
shapeIndex = _this._getPixel(x, y, lc, _this.selectCtx);
|
||
|
_this.selectedShape = lc.shapes[shapeIndex];
|
||
|
if (_this.selectedShape != null) {
|
||
|
lc.trigger('shapeSelected', {
|
||
|
selectedShape: _this.selectedShape
|
||
|
});
|
||
|
lc.setShapesInProgress([
|
||
|
_this.selectedShape, createShape('SelectionBox', {
|
||
|
shape: _this.selectedShape,
|
||
|
handleSize: 0
|
||
|
})
|
||
|
]);
|
||
|
lc.repaintLayer('main');
|
||
|
br = _this.selectedShape.getBoundingRect();
|
||
|
return _this.dragOffset = {
|
||
|
x: x - br.x,
|
||
|
y: y - br.y
|
||
|
};
|
||
|
}
|
||
|
};
|
||
|
})(this);
|
||
|
onDrag = (function(_this) {
|
||
|
return function(arg) {
|
||
|
var x, y;
|
||
|
x = arg.x, y = arg.y;
|
||
|
if (_this.selectedShape != null) {
|
||
|
_this.didDrag = true;
|
||
|
_this.selectedShape.setUpperLeft({
|
||
|
x: x - _this.dragOffset.x,
|
||
|
y: y - _this.dragOffset.y
|
||
|
});
|
||
|
lc.setShapesInProgress([
|
||
|
_this.selectedShape, createShape('SelectionBox', {
|
||
|
shape: _this.selectedShape,
|
||
|
handleSize: 0
|
||
|
})
|
||
|
]);
|
||
|
return lc.repaintLayer('main');
|
||
|
}
|
||
|
};
|
||
|
})(this);
|
||
|
onUp = (function(_this) {
|
||
|
return function(arg) {
|
||
|
var x, y;
|
||
|
x = arg.x, y = arg.y;
|
||
|
if (_this.didDrag) {
|
||
|
_this.didDrag = false;
|
||
|
lc.trigger('shapeMoved', {
|
||
|
shape: _this.selectedShape
|
||
|
});
|
||
|
lc.trigger('drawingChange', {});
|
||
|
lc.repaintLayer('main');
|
||
|
return _this._drawSelectCanvas(lc);
|
||
|
}
|
||
|
};
|
||
|
})(this);
|
||
|
selectShapeUnsubscribeFuncs.push(lc.on('lc-pointerdown', onDown));
|
||
|
selectShapeUnsubscribeFuncs.push(lc.on('lc-pointerdrag', onDrag));
|
||
|
selectShapeUnsubscribeFuncs.push(lc.on('lc-pointerup', onUp));
|
||
|
return this._drawSelectCanvas(lc);
|
||
|
};
|
||
|
|
||
|
SelectShape.prototype.willBecomeInactive = function(lc) {
|
||
|
this._selectShapeUnsubscribe();
|
||
|
return lc.setShapesInProgress([]);
|
||
|
};
|
||
|
|
||
|
SelectShape.prototype._drawSelectCanvas = function(lc) {
|
||
|
var shapes;
|
||
|
this.selectCanvas.width = lc.canvas.width;
|
||
|
this.selectCanvas.height = lc.canvas.height;
|
||
|
this.selectCtx.clearRect(0, 0, this.selectCanvas.width, this.selectCanvas.height);
|
||
|
shapes = lc.shapes.map((function(_this) {
|
||
|
return function(shape, index) {
|
||
|
return createShape('SelectionBox', {
|
||
|
shape: shape,
|
||
|
handleSize: 0,
|
||
|
backgroundColor: "#" + (_this._intToHex(index))
|
||
|
});
|
||
|
};
|
||
|
})(this));
|
||
|
return lc.draw(shapes, this.selectCtx);
|
||
|
};
|
||
|
|
||
|
SelectShape.prototype._intToHex = function(i) {
|
||
|
return ("000000" + (i.toString(16))).slice(-6);
|
||
|
};
|
||
|
|
||
|
SelectShape.prototype._getPixel = function(x, y, lc, ctx) {
|
||
|
var p, pixel;
|
||
|
p = lc.drawingCoordsToClientCoords(x, y);
|
||
|
pixel = ctx.getImageData(p.x, p.y, 1, 1).data;
|
||
|
if (pixel[3]) {
|
||
|
return parseInt(this._rgbToHex(pixel[0], pixel[1], pixel[2]), 16);
|
||
|
} else {
|
||
|
return null;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
SelectShape.prototype._componentToHex = function(c) {
|
||
|
var hex;
|
||
|
hex = c.toString(16);
|
||
|
return ("0" + hex).slice(-2);
|
||
|
};
|
||
|
|
||
|
SelectShape.prototype._rgbToHex = function(r, g, b) {
|
||
|
return "" + (this._componentToHex(r)) + (this._componentToHex(g)) + (this._componentToHex(b));
|
||
|
};
|
||
|
|
||
|
return SelectShape;
|
||
|
|
||
|
})(Tool);
|
||
|
|
||
|
|
||
|
},{"../core/shapes":17,"./base":53}],52:[function(require,module,exports){
|
||
|
var Text, Tool, createShape, getIsPointInBox,
|
||
|
extend = function(child, parent) { for (var key in parent) { if (hasProp.call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; },
|
||
|
hasProp = {}.hasOwnProperty;
|
||
|
|
||
|
Tool = require('./base').Tool;
|
||
|
|
||
|
createShape = require('../core/shapes').createShape;
|
||
|
|
||
|
getIsPointInBox = function(point, box) {
|
||
|
if (point.x < box.x) {
|
||
|
return false;
|
||
|
}
|
||
|
if (point.y < box.y) {
|
||
|
return false;
|
||
|
}
|
||
|
if (point.x > box.x + box.width) {
|
||
|
return false;
|
||
|
}
|
||
|
if (point.y > box.y + box.height) {
|
||
|
return false;
|
||
|
}
|
||
|
return true;
|
||
|
};
|
||
|
|
||
|
module.exports = Text = (function(superClass) {
|
||
|
extend(Text, superClass);
|
||
|
|
||
|
Text.prototype.name = 'Text';
|
||
|
|
||
|
Text.prototype.iconName = 'text';
|
||
|
|
||
|
function Text() {
|
||
|
this.text = '';
|
||
|
this.font = 'bold 18px sans-serif';
|
||
|
this.currentShape = null;
|
||
|
this.currentShapeState = null;
|
||
|
this.initialShapeBoundingRect = null;
|
||
|
this.dragAction = null;
|
||
|
this.didDrag = false;
|
||
|
}
|
||
|
|
||
|
Text.prototype.didBecomeActive = function(lc) {
|
||
|
var switchAway, unsubscribeFuncs, updateInputEl;
|
||
|
unsubscribeFuncs = [];
|
||
|
this.unsubscribe = (function(_this) {
|
||
|
return function() {
|
||
|
var func, i, len, results;
|
||
|
results = [];
|
||
|
for (i = 0, len = unsubscribeFuncs.length; i < len; i++) {
|
||
|
func = unsubscribeFuncs[i];
|
||
|
results.push(func());
|
||
|
}
|
||
|
return results;
|
||
|
};
|
||
|
})(this);
|
||
|
switchAway = (function(_this) {
|
||
|
return function() {
|
||
|
_this._ensureNotEditing(lc);
|
||
|
_this._clearCurrentShape(lc);
|
||
|
return lc.repaintLayer('main');
|
||
|
};
|
||
|
})(this);
|
||
|
updateInputEl = (function(_this) {
|
||
|
return function() {
|
||
|
return _this._updateInputEl(lc);
|
||
|
};
|
||
|
})(this);
|
||
|
unsubscribeFuncs.push(lc.on('drawingChange', switchAway));
|
||
|
unsubscribeFuncs.push(lc.on('zoom', updateInputEl));
|
||
|
unsubscribeFuncs.push(lc.on('imageSizeChange', updateInputEl));
|
||
|
unsubscribeFuncs.push(lc.on('snapshotLoad', (function(_this) {
|
||
|
return function() {
|
||
|
_this._clearCurrentShape(lc);
|
||
|
return lc.repaintLayer('main');
|
||
|
};
|
||
|
})(this)));
|
||
|
unsubscribeFuncs.push(lc.on('primaryColorChange', (function(_this) {
|
||
|
return function(newColor) {
|
||
|
if (!_this.currentShape) {
|
||
|
return;
|
||
|
}
|
||
|
_this.currentShape.color = newColor;
|
||
|
_this._updateInputEl(lc);
|
||
|
return lc.repaintLayer('main');
|
||
|
};
|
||
|
})(this)));
|
||
|
return unsubscribeFuncs.push(lc.on('setFont', (function(_this) {
|
||
|
return function(font) {
|
||
|
if (!_this.currentShape) {
|
||
|
return;
|
||
|
}
|
||
|
_this.font = font;
|
||
|
_this.currentShape.setFont(font);
|
||
|
_this._setShapesInProgress(lc);
|
||
|
_this._updateInputEl(lc);
|
||
|
return lc.repaintLayer('main');
|
||
|
};
|
||
|
})(this)));
|
||
|
};
|
||
|
|
||
|
Text.prototype.willBecomeInactive = function(lc) {
|
||
|
if (this.currentShape) {
|
||
|
this._ensureNotEditing(lc);
|
||
|
this.commit(lc);
|
||
|
}
|
||
|
return this.unsubscribe();
|
||
|
};
|
||
|
|
||
|
Text.prototype.setText = function(text) {
|
||
|
return this.text = text;
|
||
|
};
|
||
|
|
||
|
Text.prototype._ensureNotEditing = function(lc) {
|
||
|
if (this.currentShapeState === 'editing') {
|
||
|
return this._exitEditingState(lc);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
Text.prototype._clearCurrentShape = function(lc) {
|
||
|
this.currentShape = null;
|
||
|
this.initialShapeBoundingRect = null;
|
||
|
this.currentShapeState = null;
|
||
|
return lc.setShapesInProgress([]);
|
||
|
};
|
||
|
|
||
|
Text.prototype.commit = function(lc) {
|
||
|
if (this.currentShape.text) {
|
||
|
lc.saveShape(this.currentShape);
|
||
|
}
|
||
|
this._clearCurrentShape(lc);
|
||
|
return lc.repaintLayer('main');
|
||
|
};
|
||
|
|
||
|
Text.prototype._getSelectionShape = function(ctx, backgroundColor) {
|
||
|
if (backgroundColor == null) {
|
||
|
backgroundColor = null;
|
||
|
}
|
||
|
return createShape('SelectionBox', {
|
||
|
shape: this.currentShape,
|
||
|
ctx: ctx,
|
||
|
backgroundColor: backgroundColor
|
||
|
});
|
||
|
};
|
||
|
|
||
|
Text.prototype._setShapesInProgress = function(lc) {
|
||
|
switch (this.currentShapeState) {
|
||
|
case 'selected':
|
||
|
return lc.setShapesInProgress([this._getSelectionShape(lc.ctx), this.currentShape]);
|
||
|
case 'editing':
|
||
|
return lc.setShapesInProgress([this._getSelectionShape(lc.ctx, '#fff')]);
|
||
|
default:
|
||
|
return lc.setShapesInProgress([this.currentShape]);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
Text.prototype.begin = function(x, y, lc) {
|
||
|
var br, point, selectionBox, selectionShape;
|
||
|
this.dragAction = 'none';
|
||
|
this.didDrag = false;
|
||
|
if (this.currentShapeState === 'selected' || this.currentShapeState === 'editing') {
|
||
|
br = this.currentShape.getBoundingRect(lc.ctx);
|
||
|
selectionShape = this._getSelectionShape(lc.ctx);
|
||
|
selectionBox = selectionShape.getBoundingRect();
|
||
|
point = {
|
||
|
x: x,
|
||
|
y: y
|
||
|
};
|
||
|
if (getIsPointInBox(point, br)) {
|
||
|
this.dragAction = 'move';
|
||
|
}
|
||
|
if (getIsPointInBox(point, selectionShape.getBottomRightHandleRect())) {
|
||
|
this.dragAction = 'resizeBottomRight';
|
||
|
}
|
||
|
if (getIsPointInBox(point, selectionShape.getTopLeftHandleRect())) {
|
||
|
this.dragAction = 'resizeTopLeft';
|
||
|
}
|
||
|
if (getIsPointInBox(point, selectionShape.getBottomLeftHandleRect())) {
|
||
|
this.dragAction = 'resizeBottomLeft';
|
||
|
}
|
||
|
if (getIsPointInBox(point, selectionShape.getTopRightHandleRect())) {
|
||
|
this.dragAction = 'resizeTopRight';
|
||
|
}
|
||
|
if (this.dragAction === 'none' && this.currentShapeState === 'editing') {
|
||
|
this.dragAction = 'stop-editing';
|
||
|
this._exitEditingState(lc);
|
||
|
}
|
||
|
} else {
|
||
|
this.color = lc.getColor('primary');
|
||
|
this.currentShape = createShape('Text', {
|
||
|
x: x,
|
||
|
y: y,
|
||
|
text: this.text,
|
||
|
color: this.color,
|
||
|
font: this.font,
|
||
|
v: 1
|
||
|
});
|
||
|
this.dragAction = 'place';
|
||
|
this.currentShapeState = 'selected';
|
||
|
}
|
||
|
if (this.dragAction === 'none') {
|
||
|
this.commit(lc);
|
||
|
return;
|
||
|
}
|
||
|
this.initialShapeBoundingRect = this.currentShape.getBoundingRect(lc.ctx);
|
||
|
this.dragOffset = {
|
||
|
x: x - this.initialShapeBoundingRect.x,
|
||
|
y: y - this.initialShapeBoundingRect.y
|
||
|
};
|
||
|
this._setShapesInProgress(lc);
|
||
|
return lc.repaintLayer('main');
|
||
|
};
|
||
|
|
||
|
Text.prototype["continue"] = function(x, y, lc) {
|
||
|
var br, brBottom, brRight;
|
||
|
if (this.dragAction === 'none') {
|
||
|
return;
|
||
|
}
|
||
|
br = this.initialShapeBoundingRect;
|
||
|
brRight = br.x + br.width;
|
||
|
brBottom = br.y + br.height;
|
||
|
switch (this.dragAction) {
|
||
|
case 'place':
|
||
|
this.currentShape.x = x;
|
||
|
this.currentShape.y = y;
|
||
|
this.didDrag = true;
|
||
|
break;
|
||
|
case 'move':
|
||
|
this.currentShape.x = x - this.dragOffset.x;
|
||
|
this.currentShape.y = y - this.dragOffset.y;
|
||
|
this.didDrag = true;
|
||
|
break;
|
||
|
case 'resizeBottomRight':
|
||
|
this.currentShape.setSize(x - (this.dragOffset.x - this.initialShapeBoundingRect.width) - br.x, y - (this.dragOffset.y - this.initialShapeBoundingRect.height) - br.y);
|
||
|
break;
|
||
|
case 'resizeTopLeft':
|
||
|
this.currentShape.setSize(brRight - x + this.dragOffset.x, brBottom - y + this.dragOffset.y);
|
||
|
this.currentShape.setPosition(x - this.dragOffset.x, y - this.dragOffset.y);
|
||
|
break;
|
||
|
case 'resizeBottomLeft':
|
||
|
this.currentShape.setSize(brRight - x + this.dragOffset.x, y - (this.dragOffset.y - this.initialShapeBoundingRect.height) - br.y);
|
||
|
this.currentShape.setPosition(x - this.dragOffset.x, this.currentShape.y);
|
||
|
break;
|
||
|
case 'resizeTopRight':
|
||
|
this.currentShape.setSize(x - (this.dragOffset.x - this.initialShapeBoundingRect.width) - br.x, brBottom - y + this.dragOffset.y);
|
||
|
this.currentShape.setPosition(this.currentShape.x, y - this.dragOffset.y);
|
||
|
}
|
||
|
this._setShapesInProgress(lc);
|
||
|
lc.repaintLayer('main');
|
||
|
return this._updateInputEl(lc);
|
||
|
};
|
||
|
|
||
|
Text.prototype.end = function(x, y, lc) {
|
||
|
if (!this.currentShape) {
|
||
|
return;
|
||
|
}
|
||
|
this.currentShape.setSize(this.currentShape.forcedWidth, 0);
|
||
|
if (this.currentShapeState === 'selected') {
|
||
|
if (this.dragAction === 'place' || (this.dragAction === 'move' && !this.didDrag)) {
|
||
|
this._enterEditingState(lc);
|
||
|
}
|
||
|
}
|
||
|
this._setShapesInProgress(lc);
|
||
|
lc.repaintLayer('main');
|
||
|
return this._updateInputEl(lc);
|
||
|
};
|
||
|
|
||
|
Text.prototype._enterEditingState = function(lc) {
|
||
|
var onChange;
|
||
|
this.currentShapeState = 'editing';
|
||
|
if (this.inputEl) {
|
||
|
throw "State error";
|
||
|
}
|
||
|
this.inputEl = document.createElement('textarea');
|
||
|
this.inputEl.className = 'text-tool-input';
|
||
|
this.inputEl.style.position = 'absolute';
|
||
|
this.inputEl.style.transformOrigin = '0px 0px';
|
||
|
this.inputEl.style.backgroundColor = 'transparent';
|
||
|
this.inputEl.style.border = 'none';
|
||
|
this.inputEl.style.outline = 'none';
|
||
|
this.inputEl.style.margin = '0';
|
||
|
this.inputEl.style.padding = '4px';
|
||
|
this.inputEl.style.zIndex = '1000';
|
||
|
this.inputEl.style.overflow = 'hidden';
|
||
|
this.inputEl.style.resize = 'none';
|
||
|
this.inputEl.value = this.currentShape.text;
|
||
|
this.inputEl.addEventListener('mousedown', function(e) {
|
||
|
return e.stopPropagation();
|
||
|
});
|
||
|
this.inputEl.addEventListener('touchstart', function(e) {
|
||
|
return e.stopPropagation();
|
||
|
});
|
||
|
onChange = (function(_this) {
|
||
|
return function(e) {
|
||
|
_this.currentShape.setText(e.target.value);
|
||
|
_this.currentShape.enforceMaxBoundingRect(lc);
|
||
|
_this._setShapesInProgress(lc);
|
||
|
lc.repaintLayer('main');
|
||
|
_this._updateInputEl(lc);
|
||
|
return e.stopPropagation();
|
||
|
};
|
||
|
})(this);
|
||
|
this.inputEl.addEventListener('keydown', (function(_this) {
|
||
|
return function() {
|
||
|
return _this._updateInputEl(lc, true);
|
||
|
};
|
||
|
})(this));
|
||
|
this.inputEl.addEventListener('keyup', onChange);
|
||
|
this.inputEl.addEventListener('change', onChange);
|
||
|
this._updateInputEl(lc);
|
||
|
lc.containerEl.appendChild(this.inputEl);
|
||
|
this.inputEl.focus();
|
||
|
return this._setShapesInProgress(lc);
|
||
|
};
|
||
|
|
||
|
Text.prototype._exitEditingState = function(lc) {
|
||
|
this.currentShapeState = 'selected';
|
||
|
lc.containerEl.removeChild(this.inputEl);
|
||
|
this.inputEl = null;
|
||
|
this._setShapesInProgress(lc);
|
||
|
return lc.repaintLayer('main');
|
||
|
};
|
||
|
|
||
|
Text.prototype._updateInputEl = function(lc, withMargin) {
|
||
|
var br, transformString;
|
||
|
if (withMargin == null) {
|
||
|
withMargin = false;
|
||
|
}
|
||
|
if (!this.inputEl) {
|
||
|
return;
|
||
|
}
|
||
|
br = this.currentShape.getBoundingRect(lc.ctx, true);
|
||
|
this.inputEl.style.font = this.currentShape.font;
|
||
|
this.inputEl.style.color = this.currentShape.color;
|
||
|
this.inputEl.style.left = (lc.position.x / lc.backingScale + br.x * lc.scale - 4) + "px";
|
||
|
this.inputEl.style.top = (lc.position.y / lc.backingScale + br.y * lc.scale - 4) + "px";
|
||
|
if (withMargin && !this.currentShape.forcedWidth) {
|
||
|
this.inputEl.style.width = (br.width + 10 + this.currentShape.renderer.emDashWidth) + "px";
|
||
|
} else {
|
||
|
this.inputEl.style.width = (br.width + 12) + "px";
|
||
|
}
|
||
|
if (withMargin) {
|
||
|
this.inputEl.style.height = (br.height + 10 + this.currentShape.renderer.metrics.leading) + "px";
|
||
|
} else {
|
||
|
this.inputEl.style.height = (br.height + 10) + "px";
|
||
|
}
|
||
|
transformString = "scale(" + lc.scale + ")";
|
||
|
this.inputEl.style.transform = transformString;
|
||
|
this.inputEl.style.webkitTransform = transformString;
|
||
|
this.inputEl.style.MozTransform = transformString;
|
||
|
this.inputEl.style.msTransform = transformString;
|
||
|
return this.inputEl.style.OTransform = transformString;
|
||
|
};
|
||
|
|
||
|
Text.prototype.optionsStyle = 'font';
|
||
|
|
||
|
return Text;
|
||
|
|
||
|
})(Tool);
|
||
|
|
||
|
|
||
|
},{"../core/shapes":17,"./base":53}],53:[function(require,module,exports){
|
||
|
var Tool, ToolWithStroke, tools,
|
||
|
extend = function(child, parent) { for (var key in parent) { if (hasProp.call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; },
|
||
|
hasProp = {}.hasOwnProperty;
|
||
|
|
||
|
tools = {};
|
||
|
|
||
|
tools.Tool = Tool = (function() {
|
||
|
function Tool() {}
|
||
|
|
||
|
Tool.prototype.name = null;
|
||
|
|
||
|
Tool.prototype.iconName = null;
|
||
|
|
||
|
Tool.prototype.usesSimpleAPI = true;
|
||
|
|
||
|
Tool.prototype.begin = function(x, y, lc) {};
|
||
|
|
||
|
Tool.prototype["continue"] = function(x, y, lc) {};
|
||
|
|
||
|
Tool.prototype.end = function(x, y, lc) {};
|
||
|
|
||
|
Tool.prototype.optionsStyle = null;
|
||
|
|
||
|
Tool.prototype.didBecomeActive = function(lc) {};
|
||
|
|
||
|
Tool.prototype.willBecomeInactive = function(lc) {};
|
||
|
|
||
|
return Tool;
|
||
|
|
||
|
})();
|
||
|
|
||
|
tools.ToolWithStroke = ToolWithStroke = (function(superClass) {
|
||
|
extend(ToolWithStroke, superClass);
|
||
|
|
||
|
function ToolWithStroke(lc) {
|
||
|
this.strokeWidth = lc.opts.defaultStrokeWidth;
|
||
|
}
|
||
|
|
||
|
ToolWithStroke.prototype.optionsStyle = 'stroke-width';
|
||
|
|
||
|
ToolWithStroke.prototype.didBecomeActive = function(lc) {
|
||
|
var unsubscribeFuncs;
|
||
|
unsubscribeFuncs = [];
|
||
|
this.unsubscribe = (function(_this) {
|
||
|
return function() {
|
||
|
var func, i, len, results;
|
||
|
results = [];
|
||
|
for (i = 0, len = unsubscribeFuncs.length; i < len; i++) {
|
||
|
func = unsubscribeFuncs[i];
|
||
|
results.push(func());
|
||
|
}
|
||
|
return results;
|
||
|
};
|
||
|
})(this);
|
||
|
return unsubscribeFuncs.push(lc.on('setStrokeWidth', (function(_this) {
|
||
|
return function(strokeWidth) {
|
||
|
_this.strokeWidth = strokeWidth;
|
||
|
return lc.trigger('toolDidUpdateOptions');
|
||
|
};
|
||
|
})(this)));
|
||
|
};
|
||
|
|
||
|
ToolWithStroke.prototype.willBecomeInactive = function(lc) {
|
||
|
return this.unsubscribe();
|
||
|
};
|
||
|
|
||
|
return ToolWithStroke;
|
||
|
|
||
|
})(Tool);
|
||
|
|
||
|
module.exports = tools;
|
||
|
|
||
|
|
||
|
},{}]},{},[22])(22)
|
||
|
});
|